Module: Sequel::Plugins::Privacy::ClassMethods
- Extended by:
- T::Helpers, T::Sig
- Defined in:
- lib/sequel/plugins/privacy.rb
Instance Method Summary collapse
- #_inject_privacy_eager_block(opts) ⇒ Object
- #allow_unsafe_access!(except: []) ⇒ Object
- #allow_unsafe_access?(name = nil) ⇒ Boolean
- #associate(type, name, opts = {}, &block) ⇒ Object
- #call(values) ⇒ Object
- #finalize_privacy! ⇒ Object
- #for_vc(vc) ⇒ Object
- #policies(action, *policy_chain) ⇒ Object
- #privacy(&block) ⇒ Object
- #privacy_association_policies ⇒ Object
- #privacy_fields ⇒ Object
- #privacy_finalized? ⇒ Boolean
- #privacy_policies ⇒ Object
- #privacy_vc_key ⇒ Object
- #protect_field(field, policy: nil) ⇒ Object
- #register_association_policies(assoc_name, action, policies) ⇒ Object
- #register_policies(action, policies) ⇒ Object
- #register_protected_field(field, policy_name) ⇒ Object
- #setup_association_privacy(assoc_name) ⇒ Object
Instance Method Details
#_inject_privacy_eager_block(opts) ⇒ Object
368 369 370 371 372 373 374 375 376 377 378 379 380 |
# File 'lib/sequel/plugins/privacy.rb', line 368 def _inject_privacy_eager_block(opts) original = opts[:eager_block] wrapped = proc do |ds| ds = original.call(ds) if original vc = Thread.current[DatasetMethods::EAGER_VC_KEY] if vc && T.unsafe(ds).model.respond_to?(:privacy_vc_key) T.unsafe(ds).for_vc(vc) else ds end end opts.merge(eager_block: wrapped) end |
#allow_unsafe_access!(except: []) ⇒ Object
156 157 158 159 160 |
# File 'lib/sequel/plugins/privacy.rb', line 156 def allow_unsafe_access!(except: []) @allow_unsafe_access = T.let(true, T.nilable(T::Boolean)) @unsafe_access_except = T.let(except.map(&:to_sym), T.nilable(T::Array[Symbol])) Sequel::Privacy.logger&.warn("#{self} allows unsafe access - migrate to use for_vc()") end |
#allow_unsafe_access?(name = nil) ⇒ Boolean
164 165 166 167 168 169 |
# File 'lib/sequel/plugins/privacy.rb', line 164 def allow_unsafe_access?(name = nil) return false unless @allow_unsafe_access == true return true if name.nil? !(@unsafe_access_except || []).include?(name) end |
#associate(type, name, opts = {}, &block) ⇒ Object
347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 |
# File 'lib/sequel/plugins/privacy.rb', line 347 def associate(type, name, opts = {}, &block) opts = _inject_privacy_eager_block(opts) result = super case type when :many_to_one, :one_to_one _override_singular_association(name) _override_association_dataset(name) when :one_to_many, :many_to_many _override_plural_association(name) _override_association_dataset(name) setup_association_privacy(name) if privacy_association_policies[name] end result end |
#call(values) ⇒ Object
181 182 183 184 185 186 187 188 189 190 |
# File 'lib/sequel/plugins/privacy.rb', line 181 def call(values) vc = Thread.current[privacy_vc_key] unless vc || allow_unsafe_access? Kernel.raise Sequel::Privacy::MissingViewerContext, "#{self} requires a ViewerContext. Use #{self}.for_vc(vc) or call #{self}.allow_unsafe_access!" end super end |
#finalize_privacy! ⇒ Object
321 322 323 |
# File 'lib/sequel/plugins/privacy.rb', line 321 def finalize_privacy! @privacy_finalized = T.let(true, T.nilable(T::Boolean)) end |
#for_vc(vc) ⇒ Object
342 343 344 |
# File 'lib/sequel/plugins/privacy.rb', line 342 def for_vc(vc) dataset.for_vc(vc) end |
#policies(action, *policy_chain) ⇒ Object
327 328 329 330 |
# File 'lib/sequel/plugins/privacy.rb', line 327 def policies(action, *policy_chain) Kernel.warn "DEPRECATED: #{self}.policies is deprecated. Use `privacy do; can :#{action}, ...; end` instead" register_policies(action, policy_chain) end |
#privacy(&block) ⇒ Object
221 222 223 224 225 226 227 228 |
# File 'lib/sequel/plugins/privacy.rb', line 221 def privacy(&block) if privacy_finalized? Kernel.raise Sequel::Privacy::PrivacyAlreadyFinalizedError, "Privacy already finalized for #{self}" end dsl = PrivacyDSL.new(self) dsl.instance_eval(&block) end |
#privacy_association_policies ⇒ Object
203 204 205 |
# File 'lib/sequel/plugins/privacy.rb', line 203 def privacy_association_policies @privacy_association_policies ||= T.let({}, T.nilable(T::Hash[Symbol, T::Hash[Symbol, T::Array[T.untyped]]])) end |
#privacy_fields ⇒ Object
198 199 200 |
# File 'lib/sequel/plugins/privacy.rb', line 198 def privacy_fields @privacy_fields ||= T.let({}, T.nilable(T::Hash[Symbol, Symbol])) end |
#privacy_finalized? ⇒ Boolean
208 209 210 |
# File 'lib/sequel/plugins/privacy.rb', line 208 def privacy_finalized? @privacy_finalized == true end |
#privacy_policies ⇒ Object
193 194 195 |
# File 'lib/sequel/plugins/privacy.rb', line 193 def privacy_policies @privacy_policies ||= T.let({}, T.nilable(T::Hash[Symbol, T::Array[T.untyped]])) end |
#privacy_vc_key ⇒ Object
174 175 176 |
# File 'lib/sequel/plugins/privacy.rb', line 174 def privacy_vc_key :"#{self}_privacy_vc" end |
#protect_field(field, policy: nil) ⇒ Object
334 335 336 337 338 |
# File 'lib/sequel/plugins/privacy.rb', line 334 def protect_field(field, policy: nil) Kernel.warn "DEPRECATED: #{self}.protect_field is deprecated. Use `privacy do; field :#{field}, ...; end` instead" policy_name = policy || :"view_#{field}" register_protected_field(field, policy_name) end |
#register_association_policies(assoc_name, action, policies) ⇒ Object
272 273 274 275 276 277 278 279 |
# File 'lib/sequel/plugins/privacy.rb', line 272 def register_association_policies(assoc_name, action, policies) Kernel.raise "Privacy policies have been finalized for #{self}" if privacy_finalized? privacy_association_policies[assoc_name] ||= {} assoc_hash = T.must(privacy_association_policies[assoc_name]) assoc_hash[action] ||= [] T.must(assoc_hash[action]).concat(policies) end |
#register_policies(action, policies) ⇒ Object
231 232 233 234 235 236 237 238 |
# File 'lib/sequel/plugins/privacy.rb', line 231 def register_policies(action, policies) if privacy_finalized? Kernel.raise Sequel::Privacy::PrivacyAlreadyFinalizedError, "Privacy already finalized for #{self}" end privacy_policies[action] ||= [] T.must(privacy_policies[action]).concat(policies) end |
#register_protected_field(field, policy_name) ⇒ Object
241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 |
# File 'lib/sequel/plugins/privacy.rb', line 241 def register_protected_field(field, policy_name) if privacy_finalized? Kernel.raise Sequel::Privacy::PrivacyAlreadyFinalizedError, "Privacy already finalized for #{self}" end privacy_fields[field] = policy_name original_method = instance_method(field) define_method(field) do return original_method.bind(self).() if Sequel::Privacy::Enforcer.in_policy_eval? vc = instance_variable_get(:@viewer_context) unless vc return original_method.bind(self).() if T.unsafe(self.class).allow_unsafe_access?(field) Kernel.raise Sequel::Privacy::MissingViewerContext, "#{self.class}##{field} requires a ViewerContext" end value = original_method.bind(self).() return unless T.cast(self, InstanceMethods).allow?(vc, policy_name) value end end |
#setup_association_privacy(assoc_name) ⇒ Object
284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 |
# File 'lib/sequel/plugins/privacy.rb', line 284 def setup_association_privacy(assoc_name) assoc_policies = privacy_association_policies[assoc_name] return unless assoc_policies reflection = association_reflection(assoc_name) return unless reflection @_wrapped_associations ||= T.let({}, T.nilable(T::Hash[Symbol, T::Boolean])) return if @_wrapped_associations[assoc_name] @_wrapped_associations[assoc_name] = true # Sequel derives mutator names by stripping a trailing 's' from # the association name: many_to_many :members → add_member, # one_to_many :memberships → add_membership. # # TODO: I'm not sure if this will break sometimes. singular_name = reflection[:name].to_s.chomp('s').to_sym add_policies = assoc_policies[:add] if add_policies && method_defined?(:"add_#{singular_name}") _wrap_association_add(assoc_name, singular_name, add_policies) end remove_policies = assoc_policies[:remove] if remove_policies && method_defined?(:"remove_#{singular_name}") _wrap_association_remove(assoc_name, singular_name, remove_policies) end remove_all_policies = assoc_policies[:remove_all] return unless remove_all_policies && method_defined?(:"remove_all_#{reflection[:name]}") _wrap_association_remove_all(assoc_name, reflection[:name], remove_all_policies) end |