glib-web 0.6.1 → 0.6.5
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/app/channels/glib/channel/is_typing_channel.rb +34 -34
- data/app/channels/glib/channel/online_channel.rb +36 -36
- data/app/controllers/concerns/glib/analytics/funnel.rb +66 -61
- data/app/controllers/concerns/glib/auth/policy.rb +149 -149
- data/app/controllers/concerns/glib/json/dynamic_text.rb +126 -126
- data/app/controllers/concerns/glib/json/libs.rb +149 -149
- data/app/controllers/concerns/glib/json/new_dynamic_text.rb +122 -122
- data/app/controllers/concerns/glib/json/transformation.rb +11 -11
- data/app/controllers/concerns/glib/json/traversal.rb +92 -92
- data/app/controllers/concerns/glib/json/ui.rb +88 -88
- data/app/controllers/concerns/glib/json/validation.rb +13 -13
- data/app/controllers/glib/home_controller.rb +54 -54
- data/app/helpers/glib/app_feature_support_helper.rb +16 -16
- data/app/helpers/glib/dynamic_images_helper.rb +55 -55
- data/app/helpers/glib/dynamic_texts_helper.rb +42 -42
- data/app/helpers/glib/enum_helper.rb +18 -18
- data/app/helpers/glib/forms_helper.rb +15 -15
- data/app/helpers/glib/json_ui/abstract_builder.rb +309 -309
- data/app/helpers/glib/json_ui/action_builder/dialogs.rb +58 -58
- data/app/helpers/glib/json_ui/action_builder/http.rb +39 -39
- data/app/helpers/glib/json_ui/action_builder/iap.rb +15 -15
- data/app/helpers/glib/json_ui/action_builder/panels.rb +14 -14
- data/app/helpers/glib/json_ui/action_builder/sheets.rb +15 -15
- data/app/helpers/glib/json_ui/action_builder/snackbars.rb +41 -41
- data/app/helpers/glib/json_ui/action_builder/windows.rb +38 -38
- data/app/helpers/glib/json_ui/action_builder.rb +140 -140
- data/app/helpers/glib/json_ui/analytics_helper.rb +17 -17
- data/app/helpers/glib/json_ui/dynamic_field_builders.rb +25 -25
- data/app/helpers/glib/json_ui/generic_builders.rb +28 -28
- data/app/helpers/glib/json_ui/list_builders.rb +110 -110
- data/app/helpers/glib/json_ui/menu_builder.rb +96 -96
- data/app/helpers/glib/json_ui/page_helper.rb +221 -221
- data/app/helpers/glib/json_ui/response_helper.rb +25 -25
- data/app/helpers/glib/json_ui/styling_helper.rb +55 -55
- data/app/helpers/glib/json_ui/table_builders.rb +74 -74
- data/app/helpers/glib/json_ui/view_builder/banners.rb +26 -26
- data/app/helpers/glib/json_ui/view_builder/charts.rb +49 -49
- data/app/helpers/glib/json_ui/view_builder/fields.rb +296 -296
- data/app/helpers/glib/json_ui/view_builder/iap.rb +11 -11
- data/app/helpers/glib/json_ui/view_builder/panels.rb +262 -262
- data/app/helpers/glib/json_ui/view_builder.rb +263 -262
- data/app/helpers/glib/urls_helper.rb +12 -12
- data/app/models/concerns/glib/soft_deletable.rb +73 -73
- data/app/models/glib/active_storage/attachment.rb +9 -9
- data/app/models/glib/active_storage/blob.rb +9 -9
- data/app/models/glib/application_record.rb +18 -18
- data/app/models/glib/dynamic_text_record.rb +9 -9
- data/app/models/glib/text.rb +95 -95
- data/app/policies/glib/application_policy.rb +191 -191
- data/app/validators/email_typo_validator.rb +38 -38
- data/app/validators/email_validator.rb +7 -7
- data/app/validators/url_validator.rb +20 -20
- data/app/views/json_ui/garage/_nav_menu.json.jbuilder +72 -72
- data/app/views/json_ui/garage/actions/_dialogs.json.jbuilder +104 -104
- data/app/views/json_ui/garage/actions/_http.json.jbuilder +24 -24
- data/app/views/json_ui/garage/actions/_panels.json.jbuilder +18 -18
- data/app/views/json_ui/garage/actions/_reload.json.jbuilder +17 -17
- data/app/views/json_ui/garage/actions/_sheets.json.jbuilder +18 -18
- data/app/views/json_ui/garage/actions/_snackbars.json.jbuilder +33 -33
- data/app/views/json_ui/garage/actions/_timeouts.json.jbuilder +24 -24
- data/app/views/json_ui/garage/actions/_windows.json.jbuilder +24 -24
- data/app/views/json_ui/garage/actions/dialogs_oauth_post.json.jbuilder +6 -6
- data/app/views/json_ui/garage/actions/index.json.jbuilder +24 -24
- data/app/views/json_ui/garage/forms/_alert_post_data.json.jbuilder +10 -10
- data/app/views/json_ui/garage/forms/basic.json.jbuilder +21 -21
- data/app/views/json_ui/garage/forms/basic_post.json.jbuilder +8 -8
- data/app/views/json_ui/garage/forms/checkboxes.json.jbuilder +27 -27
- data/app/views/json_ui/garage/forms/conditional_value.json.jbuilder +38 -38
- data/app/views/json_ui/garage/forms/dynamic_group.json.jbuilder +39 -39
- data/app/views/json_ui/garage/forms/dynamic_select.json.jbuilder +24 -24
- data/app/views/json_ui/garage/forms/dynamic_select_data.json.jbuilder +38 -38
- data/app/views/json_ui/garage/forms/file_upload.json.jbuilder +37 -37
- data/app/views/json_ui/garage/forms/floating_submit.json.jbuilder +19 -19
- data/app/views/json_ui/garage/forms/generic_post.json.jbuilder +3 -3
- data/app/views/json_ui/garage/forms/get_request.json.jbuilder +27 -27
- data/app/views/json_ui/garage/forms/index.json.jbuilder +120 -120
- data/app/views/json_ui/garage/forms/new_rich_text.json.jbuilder +40 -40
- data/app/views/json_ui/garage/forms/online_participant1.json.jbuilder +25 -25
- data/app/views/json_ui/garage/forms/online_participant2.json.jbuilder +25 -25
- data/app/views/json_ui/garage/forms/payments.json.jbuilder +34 -34
- data/app/views/json_ui/garage/forms/pickers.json.jbuilder +89 -89
- data/app/views/json_ui/garage/forms/ratings.json.jbuilder +49 -49
- data/app/views/json_ui/garage/forms/rich_text.json.jbuilder +40 -40
- data/app/views/json_ui/garage/forms/selects.json.jbuilder +91 -91
- data/app/views/json_ui/garage/forms/show_hide.json.jbuilder +154 -154
- data/app/views/json_ui/garage/forms/styled_boxes.json.jbuilder +35 -35
- data/app/views/json_ui/garage/forms/submission_flow.json.jbuilder +17 -17
- data/app/views/json_ui/garage/forms/submission_flow_post.json.jbuilder +26 -26
- data/app/views/json_ui/garage/forms/submission_indicator.json.jbuilder +63 -63
- data/app/views/json_ui/garage/forms/submission_indicator_post.json.jbuilder +25 -25
- data/app/views/json_ui/garage/forms/submit_on_change.json.jbuilder +28 -28
- data/app/views/json_ui/garage/forms/text_validation.json.jbuilder +65 -65
- data/app/views/json_ui/garage/forms/timers.json.jbuilder +120 -120
- data/app/views/json_ui/garage/home/blank.json.jbuilder +10 -10
- data/app/views/json_ui/garage/home/index.json.jbuilder +36 -36
- data/app/views/json_ui/garage/home/slow.json.jbuilder +11 -11
- data/app/views/json_ui/garage/lists/_autoload_section.json.jbuilder +30 -30
- data/app/views/json_ui/garage/lists/autoload_all.json.jbuilder +32 -32
- data/app/views/json_ui/garage/lists/autoload_as_needed.json.jbuilder +35 -35
- data/app/views/json_ui/garage/lists/autoload_as_needed_responsive_columns.json.jbuilder +27 -27
- data/app/views/json_ui/garage/lists/chat_ui.json.jbuilder +94 -94
- data/app/views/json_ui/garage/lists/edit_actions.json.jbuilder +62 -62
- data/app/views/json_ui/garage/lists/fab.json.jbuilder +12 -12
- data/app/views/json_ui/garage/lists/index.json.jbuilder +41 -41
- data/app/views/json_ui/garage/lists/reordering.json.jbuilder +34 -34
- data/app/views/json_ui/garage/lists/templating.json.jbuilder +35 -35
- data/app/views/json_ui/garage/notifications/action_cable.json.jbuilder +114 -114
- data/app/views/json_ui/garage/notifications/android_post.json.jbuilder +48 -48
- data/app/views/json_ui/garage/notifications/index.json.jbuilder +36 -36
- data/app/views/json_ui/garage/notifications/web_socket.json.jbuilder +60 -60
- data/app/views/json_ui/garage/pages/flat_centered.json.jbuilder +29 -29
- data/app/views/json_ui/garage/pages/full_width.json.jbuilder +29 -29
- data/app/views/json_ui/garage/pages/full_width_height.json.jbuilder +16 -16
- data/app/views/json_ui/garage/pages/index.json.jbuilder +62 -62
- data/app/views/json_ui/garage/pages/layout.json.jbuilder +18 -18
- data/app/views/json_ui/garage/pages/lifecycle_hooks.json.jbuilder +13 -13
- data/app/views/json_ui/garage/pages/loading_indicator.json.jbuilder +10 -10
- data/app/views/json_ui/garage/pages/nav_buttons.json.jbuilder +21 -21
- data/app/views/json_ui/garage/pages/nested_scroll.json.jbuilder +40 -40
- data/app/views/json_ui/garage/pages/tab_bar.json.jbuilder +92 -92
- data/app/views/json_ui/garage/panels/_styled.json.jbuilder +78 -78
- data/app/views/json_ui/garage/panels/card.json.jbuilder +4 -4
- data/app/views/json_ui/garage/panels/carousel.json.jbuilder +38 -38
- data/app/views/json_ui/garage/panels/custom.json.jbuilder +17 -17
- data/app/views/json_ui/garage/panels/flow.json.jbuilder +59 -59
- data/app/views/json_ui/garage/panels/horizontal.json.jbuilder +101 -101
- data/app/views/json_ui/garage/panels/index.json.jbuilder +138 -138
- data/app/views/json_ui/garage/panels/outlined.json.jbuilder +4 -4
- data/app/views/json_ui/garage/panels/responsive.json.jbuilder +98 -98
- data/app/views/json_ui/garage/panels/split.json.jbuilder +182 -182
- data/app/views/json_ui/garage/panels/ul.json.jbuilder +33 -33
- data/app/views/json_ui/garage/panels/vertical.json.jbuilder +55 -55
- data/app/views/json_ui/garage/panels/web.json.jbuilder +15 -15
- data/app/views/json_ui/garage/services/dynamic_text.json.jbuilder +13 -13
- data/app/views/json_ui/garage/services/image.json.jbuilder +47 -47
- data/app/views/json_ui/garage/services/index.json.jbuilder +17 -17
- data/app/views/json_ui/garage/tables/_autoload_section.json.jbuilder +16 -16
- data/app/views/json_ui/garage/tables/autoload_all.json.jbuilder +43 -43
- data/app/views/json_ui/garage/tables/autoload_as_needed.json.jbuilder +50 -50
- data/app/views/json_ui/garage/tables/export_import.json.jbuilder +29 -29
- data/app/views/json_ui/garage/tables/horizontal_scroll.json.jbuilder +26 -26
- data/app/views/json_ui/garage/tables/index.json.jbuilder +25 -25
- data/app/views/json_ui/garage/tables/layout.json.jbuilder +40 -40
- data/app/views/json_ui/garage/views/_chart_data.json.jbuilder +17 -17
- data/app/views/json_ui/garage/views/banners.json.jbuilder +67 -67
- data/app/views/json_ui/garage/views/calendar_data.json.jbuilder +30 -30
- data/app/views/json_ui/garage/views/charts.json.jbuilder +125 -125
- data/app/views/json_ui/garage/views/controls.json.jbuilder +60 -60
- data/app/views/json_ui/garage/views/iap.json.jbuilder +21 -21
- data/app/views/json_ui/garage/views/icon_names.json.jbuilder +1450 -1450
- data/app/views/json_ui/garage/views/icons.json.jbuilder +15 -15
- data/app/views/json_ui/garage/views/images.json.jbuilder +89 -89
- data/app/views/json_ui/garage/views/index.json.jbuilder +83 -83
- data/app/views/json_ui/garage/views/links.json.jbuilder +70 -70
- data/app/views/json_ui/garage/views/map_cluster_data.json.jbuilder +41 -41
- data/app/views/json_ui/garage/views/map_data.json.jbuilder +51 -51
- data/app/views/json_ui/garage/views/maps.json.jbuilder +31 -31
- data/app/views/json_ui/garage/views/markdowns.json.jbuilder +41 -41
- data/app/views/json_ui/garage/views/misc.json.jbuilder +34 -34
- data/app/views/json_ui/garage/views/progress.json.jbuilder +31 -31
- data/app/views/json_ui/garage/views/texts.json.jbuilder +35 -35
- data/app/views/layouts/json_ui/renderer.html.erb +35 -35
- data/config/routes.rb +7 -7
- data/lib/generators/glib/install_generator.rb +24 -24
- data/lib/generators/templates/20191017062519_create_texts.rb +12 -12
- data/lib/generators/templates/20191024063257_add_scope_to_texts.rb +7 -7
- data/lib/generators/templates/20191112095018_add_lang_to_texts.rb +7 -7
- data/lib/generators/templates/20191126071051_create_active_storage_tables.active_storage.rb +27 -27
- data/lib/generators/templates/database.yml +107 -107
- data/lib/generators/templates/dynamic_text.rb +2 -2
- data/lib/glib/crypt/utils.rb +26 -26
- data/lib/glib/crypt.rb +0 -0
- data/lib/glib/dynamic_text/config.rb +21 -21
- data/lib/glib/dynamic_text.rb +0 -0
- data/lib/glib/engine.rb +7 -7
- data/lib/glib/json_crawler/action_crawler.rb +23 -23
- data/lib/glib/json_crawler/action_crawlers/action_http.rb +11 -11
- data/lib/glib/json_crawler/action_crawlers/forms_submit.rb +48 -48
- data/lib/glib/json_crawler/action_crawlers/menu.rb +12 -12
- data/lib/glib/json_crawler/action_crawlers/nav_initiate.rb +19 -19
- data/lib/glib/json_crawler/action_crawlers/run_multiple.rb +13 -13
- data/lib/glib/json_crawler/action_crawlers/windows_open.rb +33 -33
- data/lib/glib/json_crawler/coverage.rb +20 -20
- data/lib/glib/json_crawler/http.rb +120 -120
- data/lib/glib/json_crawler/router.rb +102 -90
- data/lib/glib/json_crawler.rb +11 -11
- data/lib/glib/mailer_tester.rb +36 -36
- data/lib/glib/test_helpers.rb +52 -52
- data/lib/glib/value.rb +7 -7
- data/lib/glib/version.rb +5 -5
- data/lib/glib-web.rb +9 -9
- data/lib/tasks/db.rake +95 -95
- metadata +1 -1
@@ -1,191 +1,191 @@
|
|
1
|
-
# The main purpose of this is for security. If it is important to display useful error message or to provide a "banana", then
|
2
|
-
# it's better to perform an explicit check (e.g. as a validation in the model or using a before_action).
|
3
|
-
module Glib
|
4
|
-
class ApplicationPolicy
|
5
|
-
attr_reader :user, :record, :policy_name, :controller, :request, :params
|
6
|
-
class_attribute :should_exist_attributes, instance_writer: false, default: []
|
7
|
-
|
8
|
-
private
|
9
|
-
def initialize(user, record, policy_name, controller, request, params, attributes: {})
|
10
|
-
@user = user
|
11
|
-
@record = record
|
12
|
-
@controller = controller
|
13
|
-
@request = request
|
14
|
-
# Don't get params from request because we might not have a proper request object. This might execute in Sidekiq.
|
15
|
-
# See Presenter::Model::inside_mock_controller()
|
16
|
-
@params = params
|
17
|
-
@policy_name = policy_name
|
18
|
-
|
19
|
-
if attributes.present? && controller.action_name != 'index'
|
20
|
-
self.class.module_eval { attr_accessor(*attributes.keys) }
|
21
|
-
attributes.each do |key, value|
|
22
|
-
send("#{key}=", value)
|
23
|
-
end
|
24
|
-
end
|
25
|
-
|
26
|
-
if controller.user_signed_in?
|
27
|
-
should_exist_attributes.each do |attribute|
|
28
|
-
if try(attribute).blank? && !['index'].include?(controller.action_name)
|
29
|
-
raise "Attribute #{attribute} is blank, policy will not working properly"
|
30
|
-
end
|
31
|
-
end
|
32
|
-
end
|
33
|
-
end
|
34
|
-
|
35
|
-
class << self
|
36
|
-
attr_reader :catch_all
|
37
|
-
|
38
|
-
def inherited(base)
|
39
|
-
base.should_exist_attributes = should_exist_attributes.dup
|
40
|
-
super
|
41
|
-
end
|
42
|
-
|
43
|
-
# This is to define the authorization logic for an action (or a group of actions). It's different from controller's
|
44
|
-
# authorize().
|
45
|
-
private # Used by child
|
46
|
-
def authorize(*actions, &block)
|
47
|
-
actions.each do |action|
|
48
|
-
if action == :glib_all
|
49
|
-
# Serve as a catch-all to all actions that have not been specified in the policy.
|
50
|
-
@catch_all = block
|
51
|
-
else
|
52
|
-
method_name = "#{action}?"
|
53
|
-
# Avoid accidentally redefining multiple times from child policies. But it's okay if the child policy
|
54
|
-
# wants to override the parent's authorization method.
|
55
|
-
raise "Action authorization has been declared: #{action}" if instance_methods(false).include?(method_name.to_sym)
|
56
|
-
define_method method_name, &block
|
57
|
-
end
|
58
|
-
end
|
59
|
-
end
|
60
|
-
|
61
|
-
def should_exist(*attributes)
|
62
|
-
should_exist_attributes.push(*attributes)
|
63
|
-
end
|
64
|
-
end
|
65
|
-
|
66
|
-
private
|
67
|
-
def catch_all
|
68
|
-
self.class.catch_all
|
69
|
-
end
|
70
|
-
|
71
|
-
private
|
72
|
-
# To ensure the block is called on the policy's instance instead class.
|
73
|
-
def call_catch_all
|
74
|
-
instance_eval(&catch_all)
|
75
|
-
end
|
76
|
-
|
77
|
-
authorize :index do
|
78
|
-
# We need this line because in `index` action, this method will be called instead of method_missing().
|
79
|
-
# Having this line ensures that the catch_all behaviour works according to the priority below:
|
80
|
-
# - child_policy#index?
|
81
|
-
# - child_policy#manage? -- catch_all
|
82
|
-
# - application_policy@index?
|
83
|
-
return call_catch_all if catch_all
|
84
|
-
|
85
|
-
false
|
86
|
-
end
|
87
|
-
|
88
|
-
authorize :show do
|
89
|
-
return call_catch_all if catch_all
|
90
|
-
|
91
|
-
scope.where(id: record.id).exists?
|
92
|
-
end
|
93
|
-
|
94
|
-
authorize :create do
|
95
|
-
return call_catch_all if catch_all
|
96
|
-
|
97
|
-
false
|
98
|
-
end
|
99
|
-
|
100
|
-
authorize :new do
|
101
|
-
return call_catch_all if catch_all
|
102
|
-
|
103
|
-
create?
|
104
|
-
end
|
105
|
-
|
106
|
-
authorize :update do
|
107
|
-
return call_catch_all if catch_all
|
108
|
-
|
109
|
-
false
|
110
|
-
end
|
111
|
-
|
112
|
-
authorize :edit do
|
113
|
-
return call_catch_all if catch_all
|
114
|
-
|
115
|
-
update?
|
116
|
-
end
|
117
|
-
|
118
|
-
authorize :destroy do
|
119
|
-
return call_catch_all if catch_all
|
120
|
-
|
121
|
-
false
|
122
|
-
end
|
123
|
-
|
124
|
-
public
|
125
|
-
def method_missing(name, *args, &block)
|
126
|
-
if name.to_s.end_with?('?') && catch_all
|
127
|
-
call_catch_all
|
128
|
-
else
|
129
|
-
super
|
130
|
-
end
|
131
|
-
end
|
132
|
-
|
133
|
-
public
|
134
|
-
def scope
|
135
|
-
policy_scope_class = Pundit::PolicyFinder.new(@policy_name).scope
|
136
|
-
return unless policy_scope_class
|
137
|
-
|
138
|
-
controller.policy_scope(record.class, policy_scope_class: policy_scope_class)
|
139
|
-
end
|
140
|
-
|
141
|
-
public
|
142
|
-
def self.args_builder
|
143
|
-
Proc.new { |controller| [] }
|
144
|
-
end
|
145
|
-
|
146
|
-
public
|
147
|
-
def helpers
|
148
|
-
controller.helpers
|
149
|
-
end
|
150
|
-
|
151
|
-
# TODO: Remove. Deprecated
|
152
|
-
# private # Used by child
|
153
|
-
# def public?
|
154
|
-
# true
|
155
|
-
# end
|
156
|
-
|
157
|
-
class Scope
|
158
|
-
attr_reader :user, :scope
|
159
|
-
|
160
|
-
def initialize(user, scope)
|
161
|
-
@user = user
|
162
|
-
@scope = scope
|
163
|
-
end
|
164
|
-
|
165
|
-
# def current_user
|
166
|
-
# user
|
167
|
-
# end
|
168
|
-
|
169
|
-
# To be overridden
|
170
|
-
def resolve
|
171
|
-
scope.none
|
172
|
-
end
|
173
|
-
end
|
174
|
-
|
175
|
-
private # Used by child
|
176
|
-
def everyone
|
177
|
-
true
|
178
|
-
end
|
179
|
-
|
180
|
-
# def current_user
|
181
|
-
# user
|
182
|
-
# end
|
183
|
-
|
184
|
-
# TODO: Bad pattern. Implement explicit policy parameter instead.
|
185
|
-
# - E.g. can? :destroy, :service_subscription_auto_renewal, { service_subscription: @service_subscription }
|
186
|
-
# - E.g. super class: :service_subscription_auto_renewal, { service_subscription: @service_subscription }
|
187
|
-
def controller_var(name)
|
188
|
-
controller.instance_variable_get(:"@#{name}")
|
189
|
-
end
|
190
|
-
end
|
191
|
-
end
|
1
|
+
# The main purpose of this is for security. If it is important to display useful error message or to provide a "banana", then
|
2
|
+
# it's better to perform an explicit check (e.g. as a validation in the model or using a before_action).
|
3
|
+
module Glib
|
4
|
+
class ApplicationPolicy
|
5
|
+
attr_reader :user, :record, :policy_name, :controller, :request, :params
|
6
|
+
class_attribute :should_exist_attributes, instance_writer: false, default: []
|
7
|
+
|
8
|
+
private
|
9
|
+
def initialize(user, record, policy_name, controller, request, params, attributes: {})
|
10
|
+
@user = user
|
11
|
+
@record = record
|
12
|
+
@controller = controller
|
13
|
+
@request = request
|
14
|
+
# Don't get params from request because we might not have a proper request object. This might execute in Sidekiq.
|
15
|
+
# See Presenter::Model::inside_mock_controller()
|
16
|
+
@params = params
|
17
|
+
@policy_name = policy_name
|
18
|
+
|
19
|
+
if attributes.present? && controller.action_name != 'index'
|
20
|
+
self.class.module_eval { attr_accessor(*attributes.keys) }
|
21
|
+
attributes.each do |key, value|
|
22
|
+
send("#{key}=", value)
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
if controller.user_signed_in?
|
27
|
+
should_exist_attributes.each do |attribute|
|
28
|
+
if try(attribute).blank? && !['index'].include?(controller.action_name)
|
29
|
+
raise "Attribute #{attribute} is blank, policy will not working properly"
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
class << self
|
36
|
+
attr_reader :catch_all
|
37
|
+
|
38
|
+
def inherited(base)
|
39
|
+
base.should_exist_attributes = should_exist_attributes.dup
|
40
|
+
super
|
41
|
+
end
|
42
|
+
|
43
|
+
# This is to define the authorization logic for an action (or a group of actions). It's different from controller's
|
44
|
+
# authorize().
|
45
|
+
private # Used by child
|
46
|
+
def authorize(*actions, &block)
|
47
|
+
actions.each do |action|
|
48
|
+
if action == :glib_all
|
49
|
+
# Serve as a catch-all to all actions that have not been specified in the policy.
|
50
|
+
@catch_all = block
|
51
|
+
else
|
52
|
+
method_name = "#{action}?"
|
53
|
+
# Avoid accidentally redefining multiple times from child policies. But it's okay if the child policy
|
54
|
+
# wants to override the parent's authorization method.
|
55
|
+
raise "Action authorization has been declared: #{action}" if instance_methods(false).include?(method_name.to_sym)
|
56
|
+
define_method method_name, &block
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
def should_exist(*attributes)
|
62
|
+
should_exist_attributes.push(*attributes)
|
63
|
+
end
|
64
|
+
end
|
65
|
+
|
66
|
+
private
|
67
|
+
def catch_all
|
68
|
+
self.class.catch_all
|
69
|
+
end
|
70
|
+
|
71
|
+
private
|
72
|
+
# To ensure the block is called on the policy's instance instead class.
|
73
|
+
def call_catch_all
|
74
|
+
instance_eval(&catch_all)
|
75
|
+
end
|
76
|
+
|
77
|
+
authorize :index do
|
78
|
+
# We need this line because in `index` action, this method will be called instead of method_missing().
|
79
|
+
# Having this line ensures that the catch_all behaviour works according to the priority below:
|
80
|
+
# - child_policy#index?
|
81
|
+
# - child_policy#manage? -- catch_all
|
82
|
+
# - application_policy@index?
|
83
|
+
return call_catch_all if catch_all
|
84
|
+
|
85
|
+
false
|
86
|
+
end
|
87
|
+
|
88
|
+
authorize :show do
|
89
|
+
return call_catch_all if catch_all
|
90
|
+
|
91
|
+
scope.where(id: record.id).exists?
|
92
|
+
end
|
93
|
+
|
94
|
+
authorize :create do
|
95
|
+
return call_catch_all if catch_all
|
96
|
+
|
97
|
+
false
|
98
|
+
end
|
99
|
+
|
100
|
+
authorize :new do
|
101
|
+
return call_catch_all if catch_all
|
102
|
+
|
103
|
+
create?
|
104
|
+
end
|
105
|
+
|
106
|
+
authorize :update do
|
107
|
+
return call_catch_all if catch_all
|
108
|
+
|
109
|
+
false
|
110
|
+
end
|
111
|
+
|
112
|
+
authorize :edit do
|
113
|
+
return call_catch_all if catch_all
|
114
|
+
|
115
|
+
update?
|
116
|
+
end
|
117
|
+
|
118
|
+
authorize :destroy do
|
119
|
+
return call_catch_all if catch_all
|
120
|
+
|
121
|
+
false
|
122
|
+
end
|
123
|
+
|
124
|
+
public
|
125
|
+
def method_missing(name, *args, &block)
|
126
|
+
if name.to_s.end_with?('?') && catch_all
|
127
|
+
call_catch_all
|
128
|
+
else
|
129
|
+
super
|
130
|
+
end
|
131
|
+
end
|
132
|
+
|
133
|
+
public
|
134
|
+
def scope
|
135
|
+
policy_scope_class = Pundit::PolicyFinder.new(@policy_name).scope
|
136
|
+
return unless policy_scope_class
|
137
|
+
|
138
|
+
controller.policy_scope(record.class, policy_scope_class: policy_scope_class)
|
139
|
+
end
|
140
|
+
|
141
|
+
public
|
142
|
+
def self.args_builder
|
143
|
+
Proc.new { |controller| [] }
|
144
|
+
end
|
145
|
+
|
146
|
+
public
|
147
|
+
def helpers
|
148
|
+
controller.helpers
|
149
|
+
end
|
150
|
+
|
151
|
+
# TODO: Remove. Deprecated
|
152
|
+
# private # Used by child
|
153
|
+
# def public?
|
154
|
+
# true
|
155
|
+
# end
|
156
|
+
|
157
|
+
class Scope
|
158
|
+
attr_reader :user, :scope
|
159
|
+
|
160
|
+
def initialize(user, scope)
|
161
|
+
@user = user
|
162
|
+
@scope = scope
|
163
|
+
end
|
164
|
+
|
165
|
+
# def current_user
|
166
|
+
# user
|
167
|
+
# end
|
168
|
+
|
169
|
+
# To be overridden
|
170
|
+
def resolve
|
171
|
+
scope.none
|
172
|
+
end
|
173
|
+
end
|
174
|
+
|
175
|
+
private # Used by child
|
176
|
+
def everyone
|
177
|
+
true
|
178
|
+
end
|
179
|
+
|
180
|
+
# def current_user
|
181
|
+
# user
|
182
|
+
# end
|
183
|
+
|
184
|
+
# TODO: Bad pattern. Implement explicit policy parameter instead.
|
185
|
+
# - E.g. can? :destroy, :service_subscription_auto_renewal, { service_subscription: @service_subscription }
|
186
|
+
# - E.g. super class: :service_subscription_auto_renewal, { service_subscription: @service_subscription }
|
187
|
+
def controller_var(name)
|
188
|
+
controller.instance_variable_get(:"@#{name}")
|
189
|
+
end
|
190
|
+
end
|
191
|
+
end
|
@@ -1,38 +1,38 @@
|
|
1
|
-
class EmailTypoValidator < ActiveModel::EachValidator
|
2
|
-
COMMON_FORMAT = {
|
3
|
-
'\.\.' => '.',
|
4
|
-
'\.con\z' => '.com',
|
5
|
-
'\.coj\z' => '.com',
|
6
|
-
'\.vom\z' => '.com',
|
7
|
-
'\.xom\z' => '.com',
|
8
|
-
'\.com[a-z]\z' => '.com',
|
9
|
-
'\.con\.' => '.com.',
|
10
|
-
'\.orf\z' => '.org',
|
11
|
-
'\.orf\.' => '.org.',
|
12
|
-
'@hitmail\.' => '@hotmail.',
|
13
|
-
'@htomail\.' => '@hotmail.',
|
14
|
-
'@hotnail\.' => '@hotmail.',
|
15
|
-
'@hotmil\.' => '@hotmail.',
|
16
|
-
'@hotmal\.' => '@hotmail.',
|
17
|
-
'@hoymail\.' => '@hotmail.',
|
18
|
-
'@hormail\.' => '@hotmail.',
|
19
|
-
'@hotmsil\.' => '@hotmail.',
|
20
|
-
'@gail\.' => '@gmail.',
|
21
|
-
'@gnail\.' => '@gmail.',
|
22
|
-
'@gmil\.' => '@gmail.',
|
23
|
-
'@gmal\.' => '@gmail.',
|
24
|
-
'@gmai\.' => '@gmail.',
|
25
|
-
'@yahho\.' => '@yahoo.',
|
26
|
-
'@yaho\.' => '@yahoo.',
|
27
|
-
}
|
28
|
-
|
29
|
-
def validate_each(record, attribute, value)
|
30
|
-
COMMON_FORMAT.keys.each do |format_key|
|
31
|
-
regex = Regexp.new(format_key)
|
32
|
-
if match = regex.match(value)
|
33
|
-
record.errors.add(attribute, "Invalid, replace #{match[0]} with #{COMMON_FORMAT[format_key]}")
|
34
|
-
break
|
35
|
-
end
|
36
|
-
end
|
37
|
-
end
|
38
|
-
end
|
1
|
+
class EmailTypoValidator < ActiveModel::EachValidator
|
2
|
+
COMMON_FORMAT = {
|
3
|
+
'\.\.' => '.',
|
4
|
+
'\.con\z' => '.com',
|
5
|
+
'\.coj\z' => '.com',
|
6
|
+
'\.vom\z' => '.com',
|
7
|
+
'\.xom\z' => '.com',
|
8
|
+
'\.com[a-z]\z' => '.com',
|
9
|
+
'\.con\.' => '.com.',
|
10
|
+
'\.orf\z' => '.org',
|
11
|
+
'\.orf\.' => '.org.',
|
12
|
+
'@hitmail\.' => '@hotmail.',
|
13
|
+
'@htomail\.' => '@hotmail.',
|
14
|
+
'@hotnail\.' => '@hotmail.',
|
15
|
+
'@hotmil\.' => '@hotmail.',
|
16
|
+
'@hotmal\.' => '@hotmail.',
|
17
|
+
'@hoymail\.' => '@hotmail.',
|
18
|
+
'@hormail\.' => '@hotmail.',
|
19
|
+
'@hotmsil\.' => '@hotmail.',
|
20
|
+
'@gail\.' => '@gmail.',
|
21
|
+
'@gnail\.' => '@gmail.',
|
22
|
+
'@gmil\.' => '@gmail.',
|
23
|
+
'@gmal\.' => '@gmail.',
|
24
|
+
'@gmai\.' => '@gmail.',
|
25
|
+
'@yahho\.' => '@yahoo.',
|
26
|
+
'@yaho\.' => '@yahoo.',
|
27
|
+
}
|
28
|
+
|
29
|
+
def validate_each(record, attribute, value)
|
30
|
+
COMMON_FORMAT.keys.each do |format_key|
|
31
|
+
regex = Regexp.new(format_key)
|
32
|
+
if match = regex.match(value)
|
33
|
+
record.errors.add(attribute, "Invalid, replace #{match[0]} with #{COMMON_FORMAT[format_key]}")
|
34
|
+
break
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
@@ -1,7 +1,7 @@
|
|
1
|
-
class EmailValidator < ActiveModel::EachValidator
|
2
|
-
def validate_each(record, attribute, value)
|
3
|
-
unless value =~ /\A([^@\s]+)@((?:[-a-z0-9]+\.)+[a-z]{2,})\z/i
|
4
|
-
record.errors.add(attribute, :invalid)
|
5
|
-
end
|
6
|
-
end
|
7
|
-
end
|
1
|
+
class EmailValidator < ActiveModel::EachValidator
|
2
|
+
def validate_each(record, attribute, value)
|
3
|
+
unless value =~ /\A([^@\s]+)@((?:[-a-z0-9]+\.)+[a-z]{2,})\z/i
|
4
|
+
record.errors.add(attribute, :invalid)
|
5
|
+
end
|
6
|
+
end
|
7
|
+
end
|
@@ -1,20 +1,20 @@
|
|
1
|
-
class UrlValidator < ActiveModel::EachValidator
|
2
|
-
def validate_each(record, attribute, value)
|
3
|
-
schemes = ['https', 'http']
|
4
|
-
|
5
|
-
begin
|
6
|
-
uri = URI.parse(value)
|
7
|
-
host = uri && uri.host
|
8
|
-
scheme = uri && uri.scheme
|
9
|
-
|
10
|
-
valid_scheme = host && scheme && schemes.include?(scheme)
|
11
|
-
valid_raw_url = scheme && value.match?(URI::regexp([scheme]))
|
12
|
-
|
13
|
-
unless valid_raw_url && valid_scheme
|
14
|
-
record.errors.add(attribute, :invalid)
|
15
|
-
end
|
16
|
-
rescue URI::InvalidURIError
|
17
|
-
record.errors.add(attribute, :invalid)
|
18
|
-
end
|
19
|
-
end
|
20
|
-
end
|
1
|
+
class UrlValidator < ActiveModel::EachValidator
|
2
|
+
def validate_each(record, attribute, value)
|
3
|
+
schemes = ['https', 'http']
|
4
|
+
|
5
|
+
begin
|
6
|
+
uri = URI.parse(value)
|
7
|
+
host = uri && uri.host
|
8
|
+
scheme = uri && uri.scheme
|
9
|
+
|
10
|
+
valid_scheme = host && scheme && schemes.include?(scheme)
|
11
|
+
valid_raw_url = scheme && value.match?(URI::regexp([scheme]))
|
12
|
+
|
13
|
+
unless valid_raw_url && valid_scheme
|
14
|
+
record.errors.add(attribute, :invalid)
|
15
|
+
end
|
16
|
+
rescue URI::InvalidURIError
|
17
|
+
record.errors.add(attribute, :invalid)
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
@@ -1,72 +1,72 @@
|
|
1
|
-
|
2
|
-
if local_assigns[:top_nav] || json_ui_app_is_web?
|
3
|
-
page.leftDrawer content: ->(drawer) do
|
4
|
-
drawer.header childViews: ->(header) do
|
5
|
-
header.button text: 'App', styleClasses: ['link', 'logo'], onClick: ->(action) do
|
6
|
-
action.windows_open url: root_url
|
7
|
-
end
|
8
|
-
end
|
9
|
-
|
10
|
-
drawer.rows builder: ->(menu) do
|
11
|
-
|
12
|
-
menu.button text: 'Pages', onClick: ->(action) do
|
13
|
-
action.windows_open url: json_ui_garage_url(path: 'pages/index')
|
14
|
-
end
|
15
|
-
|
16
|
-
menu.button text: 'Panels', onClick: ->(action) do
|
17
|
-
action.windows_open url: json_ui_garage_url(path: 'panels/index')
|
18
|
-
end
|
19
|
-
|
20
|
-
menu.button text: 'Lists', onClick: ->(action) do
|
21
|
-
action.windows_open url: json_ui_garage_url(path: 'lists/index')
|
22
|
-
end
|
23
|
-
|
24
|
-
menu.button text: 'Forms', onClick: ->(action) do
|
25
|
-
action.windows_open url: json_ui_garage_url(path: 'forms/index')
|
26
|
-
end
|
27
|
-
|
28
|
-
menu.button text: 'Views', onClick: ->(action) do
|
29
|
-
action.windows_open url: json_ui_garage_url(path: 'views/index')
|
30
|
-
end
|
31
|
-
|
32
|
-
menu.button text: 'Actions', onClick: ->(action) do
|
33
|
-
action.windows_open url: json_ui_garage_url(path: 'actions/index')
|
34
|
-
end
|
35
|
-
|
36
|
-
menu.button text: 'Tables (Web Only)', onClick: ->(action) do
|
37
|
-
action.windows_open url: json_ui_garage_url(path: 'tables/index')
|
38
|
-
end
|
39
|
-
|
40
|
-
menu.button text: 'Services', onClick: ->(action) do
|
41
|
-
action.windows_open url: json_ui_garage_url(path: 'services/index')
|
42
|
-
end
|
43
|
-
|
44
|
-
menu.divider
|
45
|
-
menu.label text: 'Misc Menu'
|
46
|
-
|
47
|
-
menu.button icon: 'notifications', text: 'Notifications', onClick: ->(action) do
|
48
|
-
action.windows_open url: json_ui_garage_url(path: 'notifications/index')
|
49
|
-
end
|
50
|
-
|
51
|
-
menu.button icon: 'list', text: 'UI Elements', onClick: ->(action) do
|
52
|
-
action.windows_open url: json_ui_garage_url
|
53
|
-
end
|
54
|
-
|
55
|
-
menu.button icon: 'refresh', text: 'Reload', onClick: ->(action) do
|
56
|
-
action.windows_reload
|
57
|
-
end
|
58
|
-
|
59
|
-
# Consider deprecating icon badge
|
60
|
-
# menu.button icon: { name: 'zoom_in', badge: '!' }, text: 'Diagnostics', onClick: ->(action) do
|
61
|
-
menu.button icon: 'zoom_in', text: 'Diagnostics', badgeContent: '!', onClick: ->(action) do
|
62
|
-
action.dialogs_alert message: %{
|
63
|
-
Bundle ID: #{json_ui_app_bundle_id || '<unknown>'}
|
64
|
-
App Version: #{json_ui_app_build_version || '<unknown>'}
|
65
|
-
Device OS: #{json_ui_app_device_os}
|
66
|
-
}
|
67
|
-
end
|
68
|
-
|
69
|
-
end
|
70
|
-
end
|
71
|
-
|
72
|
-
end
|
1
|
+
|
2
|
+
if local_assigns[:top_nav] || json_ui_app_is_web?
|
3
|
+
page.leftDrawer content: ->(drawer) do
|
4
|
+
drawer.header childViews: ->(header) do
|
5
|
+
header.button text: 'App', styleClasses: ['link', 'logo'], onClick: ->(action) do
|
6
|
+
action.windows_open url: root_url
|
7
|
+
end
|
8
|
+
end
|
9
|
+
|
10
|
+
drawer.rows builder: ->(menu) do
|
11
|
+
|
12
|
+
menu.button text: 'Pages', onClick: ->(action) do
|
13
|
+
action.windows_open url: json_ui_garage_url(path: 'pages/index')
|
14
|
+
end
|
15
|
+
|
16
|
+
menu.button text: 'Panels', onClick: ->(action) do
|
17
|
+
action.windows_open url: json_ui_garage_url(path: 'panels/index')
|
18
|
+
end
|
19
|
+
|
20
|
+
menu.button text: 'Lists', onClick: ->(action) do
|
21
|
+
action.windows_open url: json_ui_garage_url(path: 'lists/index')
|
22
|
+
end
|
23
|
+
|
24
|
+
menu.button text: 'Forms', onClick: ->(action) do
|
25
|
+
action.windows_open url: json_ui_garage_url(path: 'forms/index')
|
26
|
+
end
|
27
|
+
|
28
|
+
menu.button text: 'Views', onClick: ->(action) do
|
29
|
+
action.windows_open url: json_ui_garage_url(path: 'views/index')
|
30
|
+
end
|
31
|
+
|
32
|
+
menu.button text: 'Actions', onClick: ->(action) do
|
33
|
+
action.windows_open url: json_ui_garage_url(path: 'actions/index')
|
34
|
+
end
|
35
|
+
|
36
|
+
menu.button text: 'Tables (Web Only)', onClick: ->(action) do
|
37
|
+
action.windows_open url: json_ui_garage_url(path: 'tables/index')
|
38
|
+
end
|
39
|
+
|
40
|
+
menu.button text: 'Services', onClick: ->(action) do
|
41
|
+
action.windows_open url: json_ui_garage_url(path: 'services/index')
|
42
|
+
end
|
43
|
+
|
44
|
+
menu.divider
|
45
|
+
menu.label text: 'Misc Menu'
|
46
|
+
|
47
|
+
menu.button icon: 'notifications', text: 'Notifications', onClick: ->(action) do
|
48
|
+
action.windows_open url: json_ui_garage_url(path: 'notifications/index')
|
49
|
+
end
|
50
|
+
|
51
|
+
menu.button icon: 'list', text: 'UI Elements', onClick: ->(action) do
|
52
|
+
action.windows_open url: json_ui_garage_url
|
53
|
+
end
|
54
|
+
|
55
|
+
menu.button icon: 'refresh', text: 'Reload', onClick: ->(action) do
|
56
|
+
action.windows_reload
|
57
|
+
end
|
58
|
+
|
59
|
+
# Consider deprecating icon badge
|
60
|
+
# menu.button icon: { name: 'zoom_in', badge: '!' }, text: 'Diagnostics', onClick: ->(action) do
|
61
|
+
menu.button icon: 'zoom_in', text: 'Diagnostics', badgeContent: '!', onClick: ->(action) do
|
62
|
+
action.dialogs_alert message: %{
|
63
|
+
Bundle ID: #{json_ui_app_bundle_id || '<unknown>'}
|
64
|
+
App Version: #{json_ui_app_build_version || '<unknown>'}
|
65
|
+
Device OS: #{json_ui_app_device_os}
|
66
|
+
}
|
67
|
+
end
|
68
|
+
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
end
|