2019-05-12 22:42:48 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2015-06-25 12:25:15 -04:00
|
|
|
# name: discourse-data-explorer
|
|
|
|
# about: Interface for running analysis SQL queries on the live database
|
2021-03-01 10:51:48 -05:00
|
|
|
# version: 0.3
|
2015-06-25 12:25:15 -04:00
|
|
|
# authors: Riking
|
|
|
|
# url: https://github.com/discourse/discourse-data-explorer
|
|
|
|
|
|
|
|
enabled_site_setting :data_explorer_enabled
|
FEATURE: Add ability to soft delete (hide) queries and revert deletion with rake tasks (#54)
* FEATURE: Add hide button (toggleable) for all queries (frontend only)
* Switches between hide/unhide on click
* Works almost like the delete button, but toggles between the query's
hidden attribute instead
* So far this is only a frontend feature, the backend implementation
still needs work
* Revert "FEATURE: Add hide button (toggleable) for all queries (frontend only)"
This reverts commit a8771d2ad57083a91b7130df807fa54c26205d11.
REVERT: Remove button that hides queries (frontend)
* Prepare for migration of old frontend logic to backend
* We are going to reuse the existing delete button, but change its
backend logic to enable soft deletion. From the user's perspective
nothing will change, but any deletion mistakes can be reverted.
* DEV: Hide user queries upon deletion, but keep them in store
* Creating a new query will set its hidden attribute to false by
default
* Deleting a user-made query will not delete it from the store, but
set its hidden attribute to true
* User queries will not be indexed if they are hidden
* Undeleting a query will unhide it, and will be indexed
* Updating a hidden query will unhide it, and will be indexed
* SPEC: Add spec for hidden/deleted queries
* Hidden queries should not be shown
* FEATURE: Add ability to delete/hide system queries
* System queries are now able to be deleted from view, but will remain
in the backend for retrieval, if necessary
* FEATURE/DEV: Add rake commands for query soft deletion
* query:list_hidden - Shows a list of hidden queries
* query:hide_all[only_default] - Hides all queries, w/ boolean arg to
hide only default ones
* query:unhide[id] - Unhides a query by id
* query:unhide_all[exclude_default] - Unhides all hidden queries,
w/ boolean arg to exclude default ones
* Remove rails loggers
* UX/DEV: Update query rake tasks to be more user friendly
* Split query:hide_all[only_default] into two tasks:
* query:hide_all - Hides all queries
* query:hide_all:only_default - Hide only default queries
* Split query:unhide_all[exclude_default] into two tasks:
* query:unhide_all - Unhides all hidden queries
* query:unhide_all:exclude_default - Unhides all non-default
queries
* Rename file to match task name
* UX: query:unhide can accept multiple arguments
* Example: rake query:unhide[-5,-6,-7,3,5,6,-13,-14,-10]
* UX: Update query rake tasks to output cleaner messages
* Remove unneeded comment
* DEV: Keep only necessary rake tasks, use more specific naming
* UX/DEV: Add rake task for hard deletion, better console logs
* User is able to hard delete a query only if it is hidden, otherwise
output a message stating so
* Add commented examples above each task
* Add rainbow support for more readable console logs
* Successful messages will display green, failures display red,
additional info displays yellow
* Separate multiple queries with spaces instead of lines
* DEV: Remove rainbow colorizing in console logs
* Rainbow is a dependency of rubocop and it may go away in the future
* Rainbow is only used for dev and test environments
* DEV: Add Rails engine to enable rake tasks to be loaded at runtime
* DEV: Favor require - load files only if they are not already loaded
* SPEC: Add tests for data_explorer[id] rake command
* Test if a single query is hidden correctly
* Expect length of query list to not be modified
* Expect array of hidden queries to have exactly 1 element
* Expect that one element to have the same ID as the one invoked to
be hidden
* Test if multiple queries are hidden correctly
* Expect length of query list to not be modified
* Expect array of hidden queries to have the number of elements
equal to the number invoked to be hidden
* Expect the elements to have the same ID as the ones invoked to be
hidden
* Test if a query exists in PluginStore
* Expect query list to be empty
* DEV: Clear pre-existing tasks before redefining
* This prevents double invocation when user invokes the task once
* SPEC: Add tests for unhide_query rake task
* Test if a single query unhides correctly
* Expect length of query list to not be modified
* Expect array of hidden queries to have exactly 1 element after
unhiding 1 of 2 queries
* Expect remaining element to be hidden
* Test if multiple queries unhide correctly
* Expect length of query list to not be modified
* Expect array of hidden queries to have exactly 1 element after
unhiding 3 of 4 queries
* Expect remaining element to be hidden
* Test if a query exists in PluginStore
* Expect query list to not be modified
* SPEC: Add tests for hard_delete rake task
* Test if a single query hard deletes correctly
* Expect length of query list to be shorter by 1
* Expect array of hidden queries to have exactly 1 element after
hard deleting 1 of 2 queries
* Expect 1 remaining hidden element
* Test if multiple queries hard delete correctly
* Expect length of query list to be shorter by 3 after hard deleting
3 of 4 queries
* Expect array of hidden queries to have exactly 1 element after
hard deleting 3 of 4 queries
* Expect 1 remaining hidden element
* Test if a query exists in PluginStore
* Expect hidden query list to not be modified
* Test if a query is not hidden
* Expect query list to not be modified
* UX: Favor newline char in place of puts for logs
* Condensed console logs to output newline char instead of another puts
statement (reduces number of lines used significantly)
2020-07-29 02:50:24 -04:00
|
|
|
|
|
|
|
require File.expand_path('../lib/discourse_data_explorer/engine.rb', __FILE__)
|
2015-06-30 13:37:48 -04:00
|
|
|
register_asset 'stylesheets/explorer.scss'
|
2018-11-08 12:01:50 -05:00
|
|
|
|
2018-11-15 11:12:32 -05:00
|
|
|
if respond_to?(:register_svg_icon)
|
|
|
|
register_svg_icon "caret-down"
|
|
|
|
register_svg_icon "caret-right"
|
|
|
|
register_svg_icon "chevron-left"
|
|
|
|
register_svg_icon "exclamation-circle"
|
|
|
|
register_svg_icon "info"
|
|
|
|
register_svg_icon "pencil-alt"
|
|
|
|
register_svg_icon "upload"
|
|
|
|
end
|
2015-06-25 12:25:15 -04:00
|
|
|
|
2015-06-25 13:43:05 -04:00
|
|
|
# route: /admin/plugins/explorer
|
|
|
|
add_admin_route 'explorer.title', 'explorer'
|
|
|
|
|
|
|
|
module ::DataExplorer
|
2020-02-20 13:04:15 -05:00
|
|
|
QUERY_RESULT_DEFAULT_LIMIT = 1000
|
|
|
|
QUERY_RESULT_MAX_LIMIT = 10000
|
2019-01-21 03:51:53 -05:00
|
|
|
|
2015-06-25 13:43:05 -04:00
|
|
|
def self.plugin_name
|
2015-06-25 14:58:14 -04:00
|
|
|
'discourse-data-explorer'.freeze
|
2015-06-25 13:43:05 -04:00
|
|
|
end
|
|
|
|
end
|
2015-06-25 12:25:15 -04:00
|
|
|
|
2015-06-25 13:43:05 -04:00
|
|
|
after_initialize do
|
2019-09-11 10:09:41 -04:00
|
|
|
add_to_class(:guardian, :user_is_a_member_of_group?) do |group|
|
|
|
|
return false if !current_user
|
|
|
|
return true if current_user.admin?
|
|
|
|
return current_user.group_ids.include?(group.id)
|
|
|
|
end
|
|
|
|
|
2020-08-26 20:29:57 -04:00
|
|
|
add_to_class(:guardian, :user_can_access_query?) do |query|
|
|
|
|
return false if !current_user
|
|
|
|
return true if current_user.admin?
|
|
|
|
query.groups.blank? || query.groups.any? do |group|
|
|
|
|
user_is_a_member_of_group?(group)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
add_to_class(:guardian, :group_and_user_can_access_query?) do |group, query|
|
2019-09-11 10:09:41 -04:00
|
|
|
return false if !current_user
|
|
|
|
return true if current_user.admin?
|
2020-08-26 20:29:57 -04:00
|
|
|
return user_is_a_member_of_group?(group) && query.groups.exists?(id: group.id)
|
2019-09-11 10:09:41 -04:00
|
|
|
end
|
|
|
|
|
2021-04-08 12:47:44 -04:00
|
|
|
add_to_serializer(:group_show, :has_visible_data_explorer_queries, false) do
|
|
|
|
DataExplorer::Query.for_group(object).exists?
|
|
|
|
end
|
|
|
|
|
|
|
|
add_to_serializer(:group_show, :include_has_visible_data_explorer_queries?, false) do
|
|
|
|
SiteSetting.data_explorer_enabled && scope.user_is_a_member_of_group?(object)
|
|
|
|
end
|
|
|
|
|
2015-06-25 12:25:15 -04:00
|
|
|
module ::DataExplorer
|
|
|
|
class Engine < ::Rails::Engine
|
|
|
|
engine_name "data_explorer"
|
|
|
|
isolate_namespace DataExplorer
|
|
|
|
end
|
|
|
|
|
2019-01-21 05:24:21 -05:00
|
|
|
class ValidationError < StandardError
|
2015-07-08 16:45:13 -04:00
|
|
|
end
|
2015-06-25 14:58:14 -04:00
|
|
|
|
|
|
|
# Run a data explorer query on the currently connected database.
|
|
|
|
#
|
|
|
|
# @param [DataExplorer::Query] query the Query object to run
|
|
|
|
# @param [Hash] params the colon-style query parameters to pass to AR
|
|
|
|
# @param [Hash] opts hash of options
|
|
|
|
# explain - include a query plan in the result
|
|
|
|
# @return [Hash]
|
|
|
|
# error - any exception that was raised in the execution. Check this
|
|
|
|
# first before looking at any other fields.
|
|
|
|
# pg_result - the PG::Result object
|
|
|
|
# duration_nanos - the query duration, in nanoseconds
|
|
|
|
# explain - the query
|
2017-08-02 01:42:49 -04:00
|
|
|
def self.run_query(query, req_params = {}, opts = {})
|
2015-06-25 14:58:14 -04:00
|
|
|
# Safety checks
|
2015-07-15 16:20:42 -04:00
|
|
|
# see test 'doesn't allow you to modify the database #2'
|
2015-06-25 14:58:14 -04:00
|
|
|
if query.sql =~ /;/
|
|
|
|
err = DataExplorer::ValidationError.new(I18n.t('js.errors.explorer.no_semicolons'))
|
2017-08-02 01:42:49 -04:00
|
|
|
return { error: err, duration_nanos: 0 }
|
2015-06-25 12:25:15 -04:00
|
|
|
end
|
|
|
|
|
2015-07-14 16:22:53 -04:00
|
|
|
query_args = {}
|
|
|
|
begin
|
|
|
|
query_args = query.cast_params req_params
|
|
|
|
rescue DataExplorer::ValidationError => e
|
2017-08-02 01:42:49 -04:00
|
|
|
return { error: e, duration_nanos: 0 }
|
2015-06-30 22:51:38 -04:00
|
|
|
end
|
2015-07-14 16:22:53 -04:00
|
|
|
|
2015-06-25 17:53:03 -04:00
|
|
|
time_start, time_end, explain, err, result = nil
|
2015-06-25 14:58:14 -04:00
|
|
|
begin
|
|
|
|
ActiveRecord::Base.connection.transaction do
|
|
|
|
# Setting transaction to read only prevents shoot-in-foot actions like SELECT FOR UPDATE
|
2015-07-15 16:20:42 -04:00
|
|
|
# see test 'doesn't allow you to modify the database #1'
|
2018-07-12 22:42:11 -04:00
|
|
|
DB.exec "SET TRANSACTION READ ONLY"
|
2015-09-14 19:07:41 -04:00
|
|
|
# Set a statement timeout so we can't tie up the server
|
2018-07-12 22:42:11 -04:00
|
|
|
DB.exec "SET LOCAL statement_timeout = 10000"
|
2015-06-25 17:55:06 -04:00
|
|
|
|
2015-07-15 16:20:42 -04:00
|
|
|
# SQL comments are for the benefits of the slow queries log
|
|
|
|
sql = <<-SQL
|
2015-06-25 14:58:14 -04:00
|
|
|
/*
|
2015-06-25 17:55:06 -04:00
|
|
|
* DataExplorer Query
|
2015-06-30 22:51:38 -04:00
|
|
|
* Query: /admin/plugins/explorer?id=#{query.id}
|
2015-06-25 17:55:06 -04:00
|
|
|
* Started by: #{opts[:current_user]}
|
|
|
|
*/
|
2015-06-25 14:58:14 -04:00
|
|
|
WITH query AS (
|
|
|
|
#{query.sql}
|
|
|
|
) SELECT * FROM query
|
2020-02-20 13:04:15 -05:00
|
|
|
LIMIT #{opts[:limit] || DataExplorer::QUERY_RESULT_DEFAULT_LIMIT}
|
2015-06-25 14:58:14 -04:00
|
|
|
SQL
|
|
|
|
|
|
|
|
time_start = Time.now
|
2018-07-12 22:42:11 -04:00
|
|
|
|
|
|
|
# we probably want to rewrite this ... but for now reuse the working
|
|
|
|
# code we have
|
|
|
|
sql = DB.param_encoder.encode(sql, query_args)
|
|
|
|
|
|
|
|
result = ActiveRecord::Base.connection.raw_connection.async_exec(sql)
|
2015-07-09 15:02:47 -04:00
|
|
|
result.check # make sure it's done
|
2015-06-25 14:58:14 -04:00
|
|
|
time_end = Time.now
|
|
|
|
|
|
|
|
if opts[:explain]
|
2018-07-12 22:42:11 -04:00
|
|
|
explain = DB.query_hash("EXPLAIN #{query.sql}", query_args)
|
2017-08-02 01:42:49 -04:00
|
|
|
.map { |row| row["QUERY PLAN"] }.join "\n"
|
2015-06-25 14:58:14 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# All done. Issue a rollback anyways, just in case
|
2015-07-15 16:20:42 -04:00
|
|
|
# see test 'doesn't allow you to modify the database #1'
|
2015-06-25 14:58:14 -04:00
|
|
|
raise ActiveRecord::Rollback
|
|
|
|
end
|
|
|
|
rescue Exception => ex
|
|
|
|
err = ex
|
|
|
|
time_end = Time.now
|
|
|
|
end
|
2015-06-25 12:25:15 -04:00
|
|
|
|
2015-06-25 14:58:14 -04:00
|
|
|
{
|
|
|
|
error: err,
|
|
|
|
pg_result: result,
|
2015-07-09 15:02:47 -04:00
|
|
|
duration_secs: time_end - time_start,
|
2015-06-25 14:58:14 -04:00
|
|
|
explain: explain,
|
2018-07-12 22:42:11 -04:00
|
|
|
params_full: query_args
|
2015-06-25 14:58:14 -04:00
|
|
|
}
|
2015-06-25 12:25:15 -04:00
|
|
|
end
|
|
|
|
|
2015-08-05 19:40:00 -04:00
|
|
|
def self.extra_data_pluck_fields
|
|
|
|
@extra_data_pluck_fields ||= {
|
2017-08-02 01:42:49 -04:00
|
|
|
user: { class: User, fields: [:id, :username, :uploaded_avatar_id], serializer: BasicUserSerializer },
|
|
|
|
badge: { class: Badge, fields: [:id, :name, :badge_type_id, :description, :icon], include: [:badge_type], serializer: SmallBadgeSerializer },
|
|
|
|
post: { class: Post, fields: [:id, :topic_id, :post_number, :cooked, :user_id], include: [:user], serializer: SmallPostWithExcerptSerializer },
|
|
|
|
topic: { class: Topic, fields: [:id, :title, :slug, :posts_count], serializer: BasicTopicSerializer },
|
|
|
|
group: { class: Group, ignore: true },
|
|
|
|
category: { class: Category, ignore: true },
|
|
|
|
reltime: { ignore: true },
|
|
|
|
html: { ignore: true },
|
2015-08-05 19:40:00 -04:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2015-08-25 23:48:19 -04:00
|
|
|
def self.column_regexes
|
|
|
|
@column_regexes ||=
|
|
|
|
extra_data_pluck_fields.map do |key, val|
|
|
|
|
if val[:class]
|
|
|
|
/(#{val[:class].to_s.downcase})_id$/
|
|
|
|
end
|
|
|
|
end.compact
|
|
|
|
end
|
|
|
|
|
2015-08-05 19:40:00 -04:00
|
|
|
def self.add_extra_data(pg_result)
|
|
|
|
needed_classes = {}
|
2019-05-13 00:41:37 -04:00
|
|
|
ret = {}
|
|
|
|
col_map = {}
|
2015-08-05 19:40:00 -04:00
|
|
|
|
|
|
|
pg_result.fields.each_with_index do |col, idx|
|
2018-10-22 16:24:55 -04:00
|
|
|
rgx = column_regexes.find { |r| r.match col }
|
2015-08-25 23:48:19 -04:00
|
|
|
if rgx
|
|
|
|
cls = (rgx.match col)[1].to_sym
|
|
|
|
needed_classes[cls] ||= []
|
|
|
|
needed_classes[cls] << idx
|
|
|
|
elsif col =~ /^(\w+)\$/
|
|
|
|
cls = $1.to_sym
|
|
|
|
needed_classes[cls] ||= []
|
|
|
|
needed_classes[cls] << idx
|
2019-05-13 00:41:37 -04:00
|
|
|
elsif col =~ /^\w+_url$/
|
|
|
|
col_map[idx] = "url"
|
2015-08-05 19:40:00 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
needed_classes.each do |cls, column_nums|
|
|
|
|
next unless column_nums.present?
|
2015-08-25 23:48:19 -04:00
|
|
|
support_info = extra_data_pluck_fields[cls]
|
|
|
|
next unless support_info
|
|
|
|
|
|
|
|
column_nums.each do |col_n|
|
|
|
|
col_map[col_n] = cls
|
|
|
|
end
|
|
|
|
|
|
|
|
if support_info[:ignore]
|
|
|
|
ret[cls] = []
|
|
|
|
next
|
|
|
|
end
|
2015-08-05 19:40:00 -04:00
|
|
|
|
|
|
|
ids = Set.new
|
|
|
|
column_nums.each do |col_n|
|
|
|
|
ids.merge(pg_result.column_values(col_n))
|
|
|
|
end
|
|
|
|
ids.delete nil
|
|
|
|
ids.map! &:to_i
|
|
|
|
|
|
|
|
object_class = support_info[:class]
|
2015-09-21 17:43:23 -04:00
|
|
|
all_objs = object_class
|
|
|
|
all_objs = all_objs.with_deleted if all_objs.respond_to? :with_deleted
|
|
|
|
all_objs = all_objs
|
2017-08-02 01:42:49 -04:00
|
|
|
.select(support_info[:fields])
|
|
|
|
.where(id: ids.to_a.sort)
|
|
|
|
.includes(support_info[:include])
|
|
|
|
.order(:id)
|
2015-08-05 19:40:00 -04:00
|
|
|
|
|
|
|
ret[cls] = ActiveModel::ArraySerializer.new(all_objs, each_serializer: support_info[:serializer])
|
|
|
|
end
|
2015-08-25 23:48:19 -04:00
|
|
|
[ret, col_map]
|
2015-08-05 19:40:00 -04:00
|
|
|
end
|
|
|
|
|
2015-07-08 16:45:13 -04:00
|
|
|
def self.sensitive_column_names
|
|
|
|
%w(
|
2018-07-12 22:42:11 -04:00
|
|
|
#_IP_Addresses
|
|
|
|
topic_views.ip_address
|
|
|
|
users.ip_address
|
|
|
|
users.registration_ip_address
|
|
|
|
incoming_links.ip_address
|
|
|
|
topic_link_clicks.ip_address
|
|
|
|
user_histories.ip_address
|
|
|
|
|
|
|
|
#_Emails
|
|
|
|
email_tokens.email
|
|
|
|
users.email
|
|
|
|
invites.email
|
|
|
|
user_histories.email
|
|
|
|
email_logs.to_address
|
|
|
|
posts.raw_email
|
|
|
|
badge_posts.raw_email
|
|
|
|
|
|
|
|
#_Secret_Tokens
|
|
|
|
email_tokens.token
|
|
|
|
email_logs.reply_key
|
|
|
|
api_keys.key
|
|
|
|
site_settings.value
|
|
|
|
|
|
|
|
users.auth_token
|
|
|
|
users.password_hash
|
|
|
|
users.salt
|
|
|
|
|
|
|
|
#_Authentication_Info
|
|
|
|
user_open_ids.email
|
|
|
|
oauth2_user_infos.uid
|
|
|
|
oauth2_user_infos.email
|
|
|
|
facebook_user_infos.facebook_user_id
|
|
|
|
facebook_user_infos.email
|
|
|
|
twitter_user_infos.twitter_user_id
|
|
|
|
github_user_infos.github_user_id
|
|
|
|
single_sign_on_records.external_email
|
|
|
|
single_sign_on_records.external_id
|
|
|
|
google_user_infos.google_user_id
|
|
|
|
google_user_infos.email
|
2015-07-08 16:45:13 -04:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.schema
|
2015-07-28 14:18:22 -04:00
|
|
|
# No need to expire this, because the server processes get restarted on upgrade
|
2015-07-08 16:45:13 -04:00
|
|
|
# refer user to http://www.postgresql.org/docs/9.3/static/datatype.html
|
|
|
|
@schema ||= begin
|
2018-07-12 22:42:11 -04:00
|
|
|
results = DB.query_hash <<~SQL
|
|
|
|
select
|
|
|
|
c.column_name column_name,
|
|
|
|
c.data_type data_type,
|
|
|
|
c.character_maximum_length character_maximum_length,
|
|
|
|
c.is_nullable is_nullable,
|
|
|
|
c.column_default column_default,
|
|
|
|
c.table_name table_name,
|
|
|
|
pgd.description column_desc
|
|
|
|
from INFORMATION_SCHEMA.COLUMNS c
|
|
|
|
inner join pg_catalog.pg_statio_all_tables st on (c.table_schema = st.schemaname and c.table_name = st.relname)
|
|
|
|
left outer join pg_catalog.pg_description pgd on (pgd.objoid = st.relid and pgd.objsubid = c.ordinal_position)
|
|
|
|
where c.table_schema = 'public'
|
|
|
|
ORDER BY c.table_name, c.ordinal_position
|
|
|
|
SQL
|
|
|
|
|
2015-07-08 16:45:13 -04:00
|
|
|
by_table = {}
|
|
|
|
# Massage the results into a nicer form
|
|
|
|
results.each do |hash|
|
2015-07-09 18:46:19 -04:00
|
|
|
full_col_name = "#{hash['table_name']}.#{hash['column_name']}"
|
|
|
|
|
2015-07-08 16:45:13 -04:00
|
|
|
if hash['is_nullable'] == "YES"
|
|
|
|
hash['is_nullable'] = true
|
|
|
|
else
|
|
|
|
hash.delete('is_nullable')
|
|
|
|
end
|
|
|
|
clen = hash.delete 'character_maximum_length'
|
|
|
|
dt = hash['data_type']
|
2015-07-28 14:18:22 -04:00
|
|
|
if hash['column_name'] == 'id'
|
|
|
|
hash['data_type'] = 'serial'
|
|
|
|
hash['primary'] = true
|
|
|
|
elsif dt == 'character varying'
|
2015-07-08 16:45:13 -04:00
|
|
|
hash['data_type'] = "varchar(#{clen.to_i})"
|
|
|
|
elsif dt == 'timestamp without time zone'
|
|
|
|
hash['data_type'] = 'timestamp'
|
|
|
|
elsif dt == 'double precision'
|
|
|
|
hash['data_type'] = 'double'
|
|
|
|
end
|
|
|
|
default = hash['column_default']
|
|
|
|
if default.nil? || default =~ /^nextval\(/
|
|
|
|
hash.delete 'column_default'
|
|
|
|
elsif default =~ /^'(.*)'::(character varying|text)/
|
|
|
|
hash['column_default'] = $1
|
|
|
|
end
|
2015-07-27 19:05:10 -04:00
|
|
|
hash.delete('column_desc') unless hash['column_desc']
|
2015-07-08 16:45:13 -04:00
|
|
|
|
2015-07-09 18:46:19 -04:00
|
|
|
if sensitive_column_names.include? full_col_name
|
2015-07-08 16:45:13 -04:00
|
|
|
hash['sensitive'] = true
|
|
|
|
end
|
2015-07-09 18:46:19 -04:00
|
|
|
if enum_info.include? full_col_name
|
|
|
|
hash['enum'] = enum_info[full_col_name]
|
|
|
|
end
|
2015-07-28 14:18:22 -04:00
|
|
|
if denormalized_columns.include? full_col_name
|
|
|
|
hash['denormal'] = denormalized_columns[full_col_name]
|
|
|
|
end
|
2015-07-28 12:59:26 -04:00
|
|
|
fkey = fkey_info(hash['table_name'], hash['column_name'])
|
|
|
|
if fkey
|
|
|
|
hash['fkey_info'] = fkey
|
|
|
|
end
|
2015-07-08 16:45:13 -04:00
|
|
|
|
2015-07-28 12:59:26 -04:00
|
|
|
table_name = hash.delete('table_name')
|
|
|
|
by_table[table_name] ||= []
|
|
|
|
by_table[table_name] << hash
|
2015-07-08 16:45:13 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# this works for now, but no big loss if the tables aren't quite sorted
|
2015-07-09 17:22:54 -04:00
|
|
|
favored_order = %w(posts topics users categories badges groups notifications post_actions site_settings)
|
2015-07-08 16:45:13 -04:00
|
|
|
sorted_by_table = {}
|
2015-07-09 17:22:54 -04:00
|
|
|
favored_order.each do |tbl|
|
|
|
|
sorted_by_table[tbl] = by_table[tbl]
|
|
|
|
end
|
2015-07-08 16:45:13 -04:00
|
|
|
by_table.keys.sort.each do |tbl|
|
2015-07-09 17:22:54 -04:00
|
|
|
next if favored_order.include? tbl
|
2015-07-08 16:45:13 -04:00
|
|
|
sorted_by_table[tbl] = by_table[tbl]
|
|
|
|
end
|
|
|
|
sorted_by_table
|
|
|
|
end
|
|
|
|
end
|
2015-07-09 18:46:19 -04:00
|
|
|
|
|
|
|
def self.enums
|
2019-03-13 11:14:01 -04:00
|
|
|
return @enums if @enums
|
|
|
|
|
|
|
|
@enums = {
|
2019-12-08 22:20:34 -05:00
|
|
|
'application_requests.req_type': ApplicationRequest.req_types,
|
2017-08-02 01:42:49 -04:00
|
|
|
'badges.badge_type_id': Enum.new(:gold, :silver, :bronze, start: 1),
|
2020-10-16 02:41:39 -04:00
|
|
|
'bookmarks.auto_delete_preference': Bookmark.auto_delete_preferences,
|
2017-08-02 01:42:49 -04:00
|
|
|
'category_groups.permission_type': CategoryGroup.permission_types,
|
2019-12-08 22:20:34 -05:00
|
|
|
'category_users.notification_level': CategoryUser.notification_levels,
|
2017-08-02 01:42:49 -04:00
|
|
|
'directory_items.period_type': DirectoryItem.period_types,
|
2020-10-16 02:41:39 -04:00
|
|
|
'email_change_requests.change_state': EmailChangeRequest.states,
|
2017-08-02 01:42:49 -04:00
|
|
|
'groups.id': Group::AUTO_GROUPS,
|
2019-12-08 22:20:34 -05:00
|
|
|
'groups.mentionable_level': Group::ALIAS_LEVELS,
|
|
|
|
'groups.messageable_level': Group::ALIAS_LEVELS,
|
|
|
|
'groups.members_visibility_level': Group.visibility_levels,
|
|
|
|
'groups.visibility_level': Group.visibility_levels,
|
|
|
|
'groups.default_notification_level': GroupUser.notification_levels,
|
2020-10-16 02:41:39 -04:00
|
|
|
'group_histories.action': GroupHistory.actions,
|
2019-12-08 22:20:34 -05:00
|
|
|
'group_users.notification_level': GroupUser.notification_levels,
|
2020-10-16 02:41:39 -04:00
|
|
|
'imap_sync_logs.level': ImapSyncLog.levels,
|
|
|
|
'invites.emailed_status': Invite.emailed_status_types,
|
2017-08-02 01:42:49 -04:00
|
|
|
'notifications.notification_type': Notification.types,
|
2019-12-08 22:20:34 -05:00
|
|
|
'polls.results': Poll.results,
|
|
|
|
'polls.status': Poll.statuses,
|
|
|
|
'polls.type': Poll.types,
|
|
|
|
'polls.visibility': Poll.visibilities,
|
|
|
|
'post_action_types.id': PostActionType.types,
|
|
|
|
'post_actions.post_action_type_id': PostActionType.types,
|
2017-08-02 01:42:49 -04:00
|
|
|
'posts.cook_method': Post.cook_methods,
|
|
|
|
'posts.hidden_reason_id': Post.hidden_reasons,
|
|
|
|
'posts.post_type': Post.types,
|
2020-10-16 02:41:39 -04:00
|
|
|
'reviewables.status': Reviewable.statuses,
|
2019-12-08 22:20:34 -05:00
|
|
|
'reviewable_histories.reviewable_history_type': ReviewableHistory.types,
|
|
|
|
'reviewable_scores.status': ReviewableScore.statuses,
|
|
|
|
'screened_emails.action_type': ScreenedEmail.actions,
|
|
|
|
'screened_ip_addresses.action_type': ScreenedIpAddress.actions,
|
|
|
|
'screened_urls.action_type': ScreenedUrl.actions,
|
|
|
|
'search_logs.search_result_type': SearchLog.search_result_types,
|
|
|
|
'search_logs.search_type': SearchLog.search_types,
|
2017-08-02 01:42:49 -04:00
|
|
|
'site_settings.data_type': SiteSetting.types,
|
2019-12-08 22:20:34 -05:00
|
|
|
'skipped_email_logs.reason_type': SkippedEmailLog.reason_types,
|
|
|
|
'tag_group_permissions.permission_type': TagGroupPermission.permission_types,
|
2020-10-16 02:41:39 -04:00
|
|
|
'theme_fields.type_id': ThemeField.types,
|
2019-12-08 22:20:34 -05:00
|
|
|
'theme_settings.data_type': ThemeSetting.types,
|
|
|
|
'topic_timers.status_type': TopicTimer.types,
|
2017-08-02 01:42:49 -04:00
|
|
|
'topic_users.notification_level': TopicUser.notification_levels,
|
|
|
|
'topic_users.notifications_reason_id': TopicUser.notification_reasons,
|
2020-10-16 02:41:39 -04:00
|
|
|
'uploads.verification_status': Upload.verification_statuses,
|
2020-10-06 21:58:14 -04:00
|
|
|
'user_actions.action_type': UserAction.types,
|
2017-08-02 01:42:49 -04:00
|
|
|
'user_histories.action': UserHistory.actions,
|
2020-10-16 02:41:39 -04:00
|
|
|
'user_options.email_previous_replies': UserOption.previous_replies_type,
|
|
|
|
'user_options.like_notification_frequency': UserOption.like_notification_frequency_type,
|
|
|
|
'user_options.text_size_key': UserOption.text_sizes,
|
|
|
|
'user_options.title_count_mode_key': UserOption.title_count_modes,
|
|
|
|
'user_options.email_level': UserOption.email_level_types,
|
|
|
|
'user_options.email_messages_level': UserOption.email_level_types,
|
|
|
|
'user_second_factors.method': UserSecondFactor.methods,
|
2019-12-08 22:20:34 -05:00
|
|
|
'user_security_keys.factor_type': UserSecurityKey.factor_types,
|
2017-08-02 01:42:49 -04:00
|
|
|
'users.trust_level': TrustLevel.levels,
|
2020-10-16 02:41:39 -04:00
|
|
|
'watched_words.action': WatchedWord.actions,
|
2019-12-08 22:20:34 -05:00
|
|
|
'web_hooks.content_type': WebHook.content_types,
|
|
|
|
'web_hooks.last_delivery_status': WebHook.last_delivery_statuses,
|
2015-07-09 18:46:19 -04:00
|
|
|
}.with_indifferent_access
|
2019-03-13 11:14:01 -04:00
|
|
|
|
|
|
|
# QueuedPost is removed in recent Discourse releases
|
|
|
|
@enums['queued_posts.state'] = QueuedPost.states if defined?(QueuedPost)
|
|
|
|
|
|
|
|
@enums
|
2015-07-09 18:46:19 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.enum_info
|
|
|
|
@enum_info ||= begin
|
|
|
|
enum_info = {}
|
2017-08-02 01:42:49 -04:00
|
|
|
enums.map do |key, enum|
|
2015-07-09 18:46:19 -04:00
|
|
|
# https://stackoverflow.com/questions/10874356/reverse-a-hash-in-ruby
|
|
|
|
enum_info[key] = Hash[enum.to_a.map(&:reverse)]
|
|
|
|
end
|
|
|
|
enum_info
|
|
|
|
end
|
|
|
|
end
|
2015-07-28 12:59:26 -04:00
|
|
|
|
|
|
|
def self.fkey_info(table, column)
|
|
|
|
full_name = "#{table}.#{column}"
|
|
|
|
|
|
|
|
if fkey_defaults[column]
|
|
|
|
fkey_defaults[column]
|
|
|
|
elsif column =~ /_by_id$/ || column =~ /_user_id$/
|
|
|
|
:users
|
|
|
|
elsif foreign_keys[full_name]
|
|
|
|
foreign_keys[full_name]
|
|
|
|
else
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.foreign_keys
|
|
|
|
@fkey_columns ||= {
|
2017-08-02 01:42:49 -04:00
|
|
|
'posts.last_editor_id': :users,
|
|
|
|
'posts.version': :'post_revisions.number',
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'topics.featured_user1_id': :users,
|
|
|
|
'topics.featured_user2_id': :users,
|
|
|
|
'topics.featured_user3_id': :users,
|
|
|
|
'topics.featured_user4_id': :users,
|
|
|
|
'topics.featured_user5_id': :users,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'users.seen_notification_id': :notifications,
|
|
|
|
'users.uploaded_avatar_id': :uploads,
|
|
|
|
'users.primary_group_id': :groups,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'categories.latest_post_id': :posts,
|
|
|
|
'categories.latest_topic_id': :topics,
|
|
|
|
'categories.parent_category_id': :categories,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'badges.badge_grouping_id': :badge_groupings,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'post_actions.related_post_id': :posts,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'color_scheme_colors.color_scheme_id': :color_schemes,
|
|
|
|
'color_schemes.versioned_id': :color_schemes,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'incoming_links.incoming_referer_id': :incoming_referers,
|
|
|
|
'incoming_referers.incoming_domain_id': :incoming_domains,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'post_replies.reply_id': :posts,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'quoted_posts.quoted_post_id': :posts,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'topic_link_clicks.topic_link_id': :topic_links,
|
|
|
|
'topic_link_clicks.link_topic_id': :topics,
|
|
|
|
'topic_link_clicks.link_post_id': :posts,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'user_actions.target_topic_id': :topics,
|
|
|
|
'user_actions.target_post_id': :posts,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'user_avatars.custom_upload_id': :uploads,
|
|
|
|
'user_avatars.gravatar_upload_id': :uploads,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'user_badges.notification_id': :notifications,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
'user_profiles.card_image_badge_id': :badges,
|
2015-07-28 12:59:26 -04:00
|
|
|
}.with_indifferent_access
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.fkey_defaults
|
|
|
|
@fkey_defaults ||= {
|
2017-08-02 01:42:49 -04:00
|
|
|
user_id: :users,
|
2015-07-28 12:59:26 -04:00
|
|
|
# :*_by_id => :users,
|
|
|
|
# :*_user_id => :users,
|
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
category_id: :categories,
|
|
|
|
group_id: :groups,
|
|
|
|
post_id: :posts,
|
|
|
|
post_action_id: :post_actions,
|
|
|
|
topic_id: :topics,
|
|
|
|
upload_id: :uploads,
|
2015-07-28 12:59:26 -04:00
|
|
|
|
|
|
|
}.with_indifferent_access
|
|
|
|
end
|
2015-07-28 14:18:22 -04:00
|
|
|
|
|
|
|
def self.denormalized_columns
|
|
|
|
{
|
2017-08-02 01:42:49 -04:00
|
|
|
'posts.reply_count': :post_replies,
|
|
|
|
'posts.quote_count': :quoted_posts,
|
|
|
|
'posts.incoming_link_count': :topic_links,
|
|
|
|
'posts.word_count': :posts,
|
|
|
|
'posts.avg_time': :post_timings,
|
|
|
|
'posts.reads': :post_timings,
|
|
|
|
'posts.like_score': :post_actions,
|
|
|
|
|
|
|
|
'posts.like_count': :post_actions,
|
|
|
|
'posts.bookmark_count': :post_actions,
|
|
|
|
'posts.vote_count': :post_actions,
|
|
|
|
'posts.off_topic_count': :post_actions,
|
|
|
|
'posts.notify_moderators_count': :post_actions,
|
|
|
|
'posts.spam_count': :post_actions,
|
|
|
|
'posts.illegal_count': :post_actions,
|
|
|
|
'posts.inappropriate_count': :post_actions,
|
|
|
|
'posts.notify_user_count': :post_actions,
|
|
|
|
|
|
|
|
'topics.views': :topic_views,
|
|
|
|
'topics.posts_count': :posts,
|
|
|
|
'topics.reply_count': :posts,
|
|
|
|
'topics.incoming_link_count': :topic_links,
|
|
|
|
'topics.moderator_posts_count': :posts,
|
|
|
|
'topics.participant_count': :posts,
|
|
|
|
'topics.word_count': :posts,
|
|
|
|
'topics.last_posted_at': :posts,
|
|
|
|
'topics.last_post_user_idt': :posts,
|
|
|
|
'topics.avg_time': :post_timings,
|
|
|
|
'topics.highest_post_number': :posts,
|
|
|
|
'topics.image_url': :posts,
|
|
|
|
'topics.excerpt': :posts,
|
|
|
|
|
|
|
|
'topics.like_count': :post_actions,
|
|
|
|
'topics.bookmark_count': :post_actions,
|
|
|
|
'topics.vote_count': :post_actions,
|
|
|
|
'topics.off_topic_count': :post_actions,
|
|
|
|
'topics.notify_moderators_count': :post_actions,
|
|
|
|
'topics.spam_count': :post_actions,
|
|
|
|
'topics.illegal_count': :post_actions,
|
|
|
|
'topics.inappropriate_count': :post_actions,
|
|
|
|
'topics.notify_user_count': :post_actions,
|
|
|
|
|
|
|
|
'categories.topic_count': :topics,
|
|
|
|
'categories.post_count': :posts,
|
|
|
|
'categories.latest_post_id': :posts,
|
|
|
|
'categories.latest_topic_id': :topics,
|
|
|
|
'categories.description': :posts,
|
|
|
|
'categories.read_restricted': :category_groups,
|
|
|
|
'categories.topics_year': :topics,
|
|
|
|
'categories.topics_month': :topics,
|
|
|
|
'categories.topics_week': :topics,
|
|
|
|
'categories.topics_day': :topics,
|
|
|
|
'categories.posts_year': :posts,
|
|
|
|
'categories.posts_month': :posts,
|
|
|
|
'categories.posts_week': :posts,
|
|
|
|
'categories.posts_day': :posts,
|
|
|
|
|
|
|
|
'badges.grant_count': :user_badges,
|
|
|
|
'groups.user_count': :group_users,
|
|
|
|
|
|
|
|
'directory_items.likes_received': :post_actions,
|
|
|
|
'directory_items.likes_given': :post_actions,
|
|
|
|
'directory_items.topics_entered': :user_stats,
|
|
|
|
'directory_items.days_visited': :user_stats,
|
|
|
|
'directory_items.posts_read': :user_stats,
|
|
|
|
'directory_items.topic_count': :topics,
|
|
|
|
'directory_items.post_count': :posts,
|
|
|
|
|
|
|
|
'post_search_data.search_data': :posts,
|
|
|
|
|
|
|
|
'top_topics.yearly_posts_count': :posts,
|
|
|
|
'top_topics.monthly_posts_count': :posts,
|
|
|
|
'top_topics.weekly_posts_count': :posts,
|
|
|
|
'top_topics.daily_posts_count': :posts,
|
|
|
|
'top_topics.yearly_views_count': :topic_views,
|
|
|
|
'top_topics.monthly_views_count': :topic_views,
|
|
|
|
'top_topics.weekly_views_count': :topic_views,
|
|
|
|
'top_topics.daily_views_count': :topic_views,
|
|
|
|
'top_topics.yearly_likes_count': :post_actions,
|
|
|
|
'top_topics.monthly_likes_count': :post_actions,
|
|
|
|
'top_topics.weekly_likes_count': :post_actions,
|
|
|
|
'top_topics.daily_likes_count': :post_actions,
|
|
|
|
'top_topics.yearly_op_likes_count': :post_actions,
|
|
|
|
'top_topics.monthly_op_likes_count': :post_actions,
|
|
|
|
'top_topics.weekly_op_likes_count': :post_actions,
|
|
|
|
'top_topics.daily_op_likes_count': :post_actions,
|
|
|
|
'top_topics.all_score': :posts,
|
|
|
|
'top_topics.yearly_score': :posts,
|
|
|
|
'top_topics.monthly_score': :posts,
|
|
|
|
'top_topics.weekly_score': :posts,
|
|
|
|
'top_topics.daily_score': :posts,
|
|
|
|
|
|
|
|
'topic_links.clicks': :topic_link_clicks,
|
|
|
|
'topic_search_data.search_data': :topics,
|
|
|
|
|
|
|
|
'topic_users.liked': :post_actions,
|
|
|
|
'topic_users.bookmarked': :post_actions,
|
|
|
|
|
|
|
|
'user_stats.posts_read_count': :post_timings,
|
|
|
|
'user_stats.topic_reply_count': :posts,
|
|
|
|
'user_stats.first_post_created_at': :posts,
|
|
|
|
'user_stats.post_count': :posts,
|
|
|
|
'user_stats.topic_count': :topics,
|
|
|
|
'user_stats.likes_given': :post_actions,
|
|
|
|
'user_stats.likes_received': :post_actions,
|
|
|
|
|
|
|
|
'user_search_data.search_data': :user_profiles,
|
|
|
|
|
|
|
|
'users.last_posted_at': :posts,
|
|
|
|
'users.previous_visit_at': :user_visits,
|
2015-07-28 14:18:22 -04:00
|
|
|
}.with_indifferent_access
|
|
|
|
end
|
2015-06-25 12:25:15 -04:00
|
|
|
end
|
|
|
|
|
2015-07-14 16:22:53 -04:00
|
|
|
class DataExplorer::Parameter
|
|
|
|
attr_accessor :identifier, :type, :default, :nullable
|
|
|
|
|
|
|
|
def initialize(identifier, type, default, nullable)
|
|
|
|
raise DataExplorer::ValidationError.new('Parameter declaration error - identifier is missing') unless identifier
|
|
|
|
raise DataExplorer::ValidationError.new('Parameter declaration error - type is missing') unless type
|
|
|
|
# process aliases
|
|
|
|
type = type.to_sym
|
|
|
|
if DataExplorer::Parameter.type_aliases[type]
|
|
|
|
type = DataExplorer::Parameter.type_aliases[type]
|
|
|
|
end
|
|
|
|
raise DataExplorer::ValidationError.new("Parameter declaration error - unknown type #{type}") unless DataExplorer::Parameter.types[type]
|
|
|
|
|
|
|
|
@identifier = identifier
|
|
|
|
@type = type
|
|
|
|
@default = default
|
|
|
|
@nullable = nullable
|
|
|
|
begin
|
|
|
|
cast_to_ruby default unless default.blank?
|
|
|
|
rescue DataExplorer::ValidationError
|
|
|
|
raise DataExplorer::ValidationError.new("Parameter declaration error - the default value is not a valid #{type}")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_hash
|
|
|
|
{
|
|
|
|
identifier: @identifier,
|
|
|
|
type: @type,
|
|
|
|
default: @default,
|
|
|
|
nullable: @nullable,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.types
|
|
|
|
@types ||= Enum.new(
|
|
|
|
# Normal types
|
2015-07-14 19:49:23 -04:00
|
|
|
:int, :bigint, :boolean, :string, :date, :time, :datetime, :double,
|
2015-07-14 16:22:53 -04:00
|
|
|
# Selection help
|
|
|
|
:user_id, :post_id, :topic_id, :category_id, :group_id, :badge_id,
|
|
|
|
# Arrays
|
2015-07-14 19:49:23 -04:00
|
|
|
:int_list, :string_list, :user_list
|
2015-07-14 16:22:53 -04:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.type_aliases
|
|
|
|
@type_aliases ||= {
|
|
|
|
integer: :int,
|
|
|
|
text: :string,
|
2015-07-14 19:01:38 -04:00
|
|
|
timestamp: :datetime,
|
2015-07-14 16:22:53 -04:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def cast_to_ruby(string)
|
|
|
|
string = @default unless string
|
|
|
|
|
|
|
|
if string.blank?
|
|
|
|
if @nullable
|
|
|
|
return nil
|
|
|
|
else
|
|
|
|
raise DataExplorer::ValidationError.new("Missing parameter #{identifier} of type #{type}")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
if string.downcase == '#null'
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
def invalid_format(string, msg = nil)
|
2015-07-14 16:22:53 -04:00
|
|
|
if msg
|
|
|
|
raise DataExplorer::ValidationError.new("'#{string}' is an invalid #{type} - #{msg}")
|
|
|
|
else
|
|
|
|
raise DataExplorer::ValidationError.new("'#{string}' is an invalid value for #{type}")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
value = nil
|
|
|
|
|
|
|
|
case @type
|
2017-08-02 01:42:49 -04:00
|
|
|
when :int
|
|
|
|
invalid_format string, 'Not an integer' unless string =~ /^-?\d+$/
|
|
|
|
value = string.to_i
|
2017-09-04 02:06:19 -04:00
|
|
|
invalid_format string, 'Too large' unless Integer === value
|
2017-08-02 01:42:49 -04:00
|
|
|
when :bigint
|
|
|
|
invalid_format string, 'Not an integer' unless string =~ /^-?\d+$/
|
|
|
|
value = string.to_i
|
|
|
|
when :boolean
|
|
|
|
value = !!(string =~ /t|true|y|yes|1/i)
|
|
|
|
when :string
|
|
|
|
value = string
|
|
|
|
when :time
|
|
|
|
begin
|
|
|
|
value = Time.parse string
|
|
|
|
rescue ArgumentError => e
|
|
|
|
invalid_format string, e.message
|
|
|
|
end
|
|
|
|
when :date
|
|
|
|
begin
|
|
|
|
value = Date.parse string
|
|
|
|
rescue ArgumentError => e
|
|
|
|
invalid_format string, e.message
|
|
|
|
end
|
|
|
|
when :datetime
|
|
|
|
begin
|
|
|
|
value = DateTime.parse string
|
|
|
|
rescue ArgumentError => e
|
|
|
|
invalid_format string, e.message
|
|
|
|
end
|
|
|
|
when :double
|
|
|
|
if string =~ /-?\d*(\.\d+)/
|
|
|
|
value = Float(string)
|
|
|
|
elsif string =~ /^(-?)Inf(inity)?$/i
|
|
|
|
if $1
|
|
|
|
value = -Float::INFINITY
|
|
|
|
else
|
|
|
|
value = Float::INFINITY
|
2015-07-14 16:22:53 -04:00
|
|
|
end
|
2017-08-02 01:42:49 -04:00
|
|
|
elsif string =~ /^(-?)NaN$/i
|
|
|
|
if $1
|
|
|
|
value = -Float::NAN
|
|
|
|
else
|
|
|
|
value = Float::NAN
|
2015-07-14 19:01:38 -04:00
|
|
|
end
|
2017-08-02 01:42:49 -04:00
|
|
|
else
|
|
|
|
invalid_format string
|
|
|
|
end
|
|
|
|
when :category_id
|
|
|
|
if string =~ /(.*)\/(.*)/
|
|
|
|
parent_name = $1
|
|
|
|
child_name = $2
|
|
|
|
parent = Category.query_parent_category(parent_name)
|
|
|
|
invalid_format string, "Could not find category named #{parent_name}" unless parent
|
|
|
|
object = Category.query_category(child_name, parent)
|
|
|
|
invalid_format string, "Could not find subcategory of #{parent_name} named #{child_name}" unless object
|
|
|
|
else
|
|
|
|
object = Category.where(id: string.to_i).first || Category.where(slug: string).first || Category.where(name: string).first
|
|
|
|
invalid_format string, "Could not find category named #{string}" unless object
|
|
|
|
end
|
|
|
|
|
|
|
|
value = object.id
|
|
|
|
when :user_id, :post_id, :topic_id, :group_id, :badge_id
|
|
|
|
if string.gsub(/[ _]/, '') =~ /^-?\d+$/
|
|
|
|
clazz_name = (/^(.*)_id$/.match(type.to_s)[1].classify.to_sym)
|
2015-07-14 19:01:38 -04:00
|
|
|
begin
|
2017-08-02 01:42:49 -04:00
|
|
|
object = Object.const_get(clazz_name).with_deleted.find(string.gsub(/[ _]/, '').to_i)
|
|
|
|
value = object.id
|
|
|
|
rescue ActiveRecord::RecordNotFound
|
|
|
|
invalid_format string, "The specified #{clazz_name} was not found"
|
2015-07-14 19:01:38 -04:00
|
|
|
end
|
2017-08-02 01:42:49 -04:00
|
|
|
elsif type == :user_id
|
|
|
|
begin
|
|
|
|
object = User.find_by_username_or_email(string)
|
|
|
|
value = object.id
|
|
|
|
rescue ActiveRecord::RecordNotFound
|
|
|
|
invalid_format string, "The user named #{string} was not found"
|
2015-07-15 15:23:56 -04:00
|
|
|
end
|
2017-08-02 01:42:49 -04:00
|
|
|
elsif type == :post_id
|
|
|
|
if string =~ /(\d+)\/(\d+)(\?u=.*)?$/
|
|
|
|
object = Post.with_deleted.find_by(topic_id: $1, post_number: $2)
|
|
|
|
invalid_format string, "The post at topic:#{$1} post_number:#{$2} was not found" unless object
|
|
|
|
value = object.id
|
2015-07-14 19:49:23 -04:00
|
|
|
end
|
2017-08-02 01:42:49 -04:00
|
|
|
elsif type == :topic_id
|
|
|
|
if string =~ /\/t\/[^\/]+\/(\d+)/
|
2015-07-14 16:22:53 -04:00
|
|
|
begin
|
2017-08-02 01:42:49 -04:00
|
|
|
object = Topic.with_deleted.find($1)
|
2015-08-04 02:35:02 -04:00
|
|
|
value = object.id
|
2015-07-14 16:22:53 -04:00
|
|
|
rescue ActiveRecord::RecordNotFound
|
2017-08-02 01:42:49 -04:00
|
|
|
invalid_format string, "The topic with id #{$1} was not found"
|
2015-07-14 16:22:53 -04:00
|
|
|
end
|
|
|
|
end
|
2017-08-02 01:42:49 -04:00
|
|
|
elsif type == :group_id
|
|
|
|
object = Group.where(name: string).first
|
|
|
|
invalid_format string, "The group named #{string} was not found" unless object
|
|
|
|
value = object.id
|
2015-07-14 16:22:53 -04:00
|
|
|
else
|
2017-08-02 01:42:49 -04:00
|
|
|
invalid_format string
|
|
|
|
end
|
|
|
|
when :int_list
|
|
|
|
value = string.split(',').map { |s| s.downcase == '#null' ? nil : s.to_i }
|
|
|
|
invalid_format string, "can't be empty" if value.length == 0
|
|
|
|
when :string_list
|
|
|
|
value = string.split(',').map { |s| s.downcase == '#null' ? nil : s }
|
|
|
|
invalid_format string, "can't be empty" if value.length == 0
|
|
|
|
when :user_list
|
|
|
|
value = string.split(',').map { |s| User.find_by_username_or_email(s) }
|
|
|
|
invalid_format string, "can't be empty" if value.length == 0
|
|
|
|
else
|
|
|
|
raise TypeError.new('unknown parameter type??? should not get here')
|
2015-07-14 16:22:53 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
value
|
|
|
|
end
|
|
|
|
|
2017-08-02 01:42:49 -04:00
|
|
|
def self.create_from_sql(sql, opts = {})
|
2015-07-14 16:22:53 -04:00
|
|
|
in_params = false
|
|
|
|
ret_params = []
|
2017-12-18 07:46:02 -05:00
|
|
|
sql.lines.find do |line|
|
|
|
|
line.chomp!
|
|
|
|
|
2015-07-14 16:22:53 -04:00
|
|
|
if in_params
|
|
|
|
# -- (ident) :(ident) (= (ident))?
|
|
|
|
|
|
|
|
if line =~ /^\s*--\s*([a-zA-Z_ ]+)\s*:([a-z_]+)\s*(?:=\s+(.*)\s*)?$/
|
|
|
|
type = $1
|
|
|
|
ident = $2
|
|
|
|
default = $3
|
|
|
|
nullable = false
|
|
|
|
if type =~ /^(null)?(.*?)(null)?$/i
|
2017-08-02 01:42:49 -04:00
|
|
|
if $1 || $3
|
2015-07-14 16:22:53 -04:00
|
|
|
nullable = true
|
|
|
|
end
|
|
|
|
type = $2
|
|
|
|
end
|
|
|
|
type = type.strip
|
|
|
|
|
2015-07-14 19:01:38 -04:00
|
|
|
begin
|
|
|
|
ret_params << DataExplorer::Parameter.new(ident, type, default, nullable)
|
|
|
|
rescue
|
|
|
|
if opts[:strict]
|
|
|
|
raise
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-07-14 16:22:53 -04:00
|
|
|
false
|
|
|
|
elsif line =~ /^\s+$/
|
|
|
|
false
|
|
|
|
else
|
|
|
|
true
|
|
|
|
end
|
|
|
|
|
|
|
|
else
|
|
|
|
if line =~ /^\s*--\s*\[params\]\s*$/
|
|
|
|
in_params = true
|
|
|
|
end
|
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
2019-11-14 15:07:58 -05:00
|
|
|
ret_params
|
2015-07-14 16:22:53 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-06-25 12:25:15 -04:00
|
|
|
require_dependency 'application_controller'
|
2018-10-10 07:29:13 -04:00
|
|
|
require_dependency File.expand_path('../lib/queries.rb', __FILE__)
|
2015-06-25 12:25:15 -04:00
|
|
|
|
2015-06-25 14:58:14 -04:00
|
|
|
DataExplorer::Engine.routes.draw do
|
2015-06-25 16:26:31 -04:00
|
|
|
root to: "query#index"
|
2015-07-08 16:45:13 -04:00
|
|
|
get 'schema' => "query#schema"
|
2015-06-25 16:26:31 -04:00
|
|
|
get 'queries' => "query#index"
|
2019-11-14 13:39:55 -05:00
|
|
|
get 'groups' => "query#groups"
|
2015-06-29 15:10:24 -04:00
|
|
|
post 'queries' => "query#create"
|
2015-06-30 13:20:22 -04:00
|
|
|
get 'queries/:id' => "query#show"
|
|
|
|
put 'queries/:id' => "query#update"
|
|
|
|
delete 'queries/:id' => "query#destroy"
|
|
|
|
post 'queries/:id/run' => "query#run"
|
2015-06-25 12:25:15 -04:00
|
|
|
end
|
|
|
|
|
2015-06-25 14:58:14 -04:00
|
|
|
Discourse::Application.routes.append do
|
2019-09-11 10:09:41 -04:00
|
|
|
get '/g/:group_name/reports' => 'data_explorer/query#group_reports_index'
|
|
|
|
get '/g/:group_name/reports/:id' => 'data_explorer/query#group_reports_show'
|
|
|
|
post '/g/:group_name/reports/:id/run' => 'data_explorer/query#group_reports_run'
|
|
|
|
|
2015-06-25 16:26:31 -04:00
|
|
|
mount ::DataExplorer::Engine, at: '/admin/plugins/explorer', constraints: AdminConstraint.new
|
2015-06-25 12:25:15 -04:00
|
|
|
end
|
2015-07-28 12:59:26 -04:00
|
|
|
end
|