Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

throw :forbidden #588

Merged
merged 5 commits into from
Jun 15, 2022
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Jump to
Jump to file
Failed to load files.
Diff view
Diff view
4 changes: 3 additions & 1 deletion app/channels/stimulus_reflex/channel.rb
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,8 @@ def receive(data)

if reflex.halted?
reflex.halted data: data
elsif reflex.forbidden?
reflex.forbidden data: data
else
begin
reflex.broadcast(reflex_data.selectors, data)
Expand Down Expand Up @@ -112,7 +114,7 @@ def delegate_call_to_reflex(reflex)
end

def commit_session(reflex)
store = reflex.request.session.instance_variable_get("@by")
store = reflex.request.session.instance_variable_get(:@by)
store.commit_session reflex.request, reflex.controller.response
rescue => exception
error = exception_with_backtrace(exception)
Expand Down
44 changes: 29 additions & 15 deletions javascript/callbacks.js
Original file line number Diff line number Diff line change
Expand Up @@ -86,8 +86,15 @@ const afterDOMUpdate = event => {
const routeReflexEvent = event => {
const { stimulusReflex, payload, name, body } = event.detail || {}
const eventType = name.split('-')[2]
if (!stimulusReflex || !['nothing', 'halted', 'error'].includes(eventType))
return

const eventTypes = {
nothing: nothing,
halted: halted,
forbidden: forbidden,
error: error
}

if (!stimulusReflex || !Object.keys(eventTypes).includes(eventType)) return

const { reflexId, xpathElement, xpathController } = stimulusReflex
const reflexElement = XPathToElement(xpathElement)
Expand All @@ -100,17 +107,7 @@ const routeReflexEvent = event => {
if (eventType === 'error') controllerElement.reflexError[reflexId] = body
}

switch (eventType) {
case 'nothing':
nothing(event, payload, promise, reflex, reflexElement)
break
case 'error':
error(event, payload, promise, reflex, reflexElement)
break
case 'halted':
halted(event, payload, promise, reflex, reflexElement)
break
}
eventTypes[eventType](event, payload, promise, reflex, reflexElement)

setTimeout(() =>
dispatchLifecycleEvent(
Expand All @@ -129,7 +126,7 @@ const routeReflexEvent = event => {
const nothing = (event, payload, promise, reflex, reflexElement) => {
reflex.finalStage = 'after'

Log.success(event, false)
Log.success(event)

setTimeout(() =>
promise.resolve({
Expand All @@ -146,7 +143,24 @@ const nothing = (event, payload, promise, reflex, reflexElement) => {
const halted = (event, payload, promise, reflex, reflexElement) => {
reflex.finalStage = 'halted'

Log.success(event, true)
Log.halted(event)

setTimeout(() =>
promise.resolve({
data: promise.data,
element: reflexElement,
event,
payload,
reflexId: promise.data.reflexId,
toString: () => ''
})
)
}

const forbidden = (event, payload, promise, reflex, reflexElement) => {
reflex.finalStage = 'forbidden'

Log.forbidden(event)

setTimeout(() =>
promise.resolve({
Expand Down
15 changes: 15 additions & 0 deletions javascript/lifecycle.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import { reflexes } from './reflex_store'
// * success
// * error
// * halted
// * forbidden
// * after
// * finalize
//
Expand Down Expand Up @@ -161,6 +162,19 @@ document.addEventListener(
true
)

document.addEventListener(
'stimulus-reflex:forbidden',
event =>
invokeLifecycleMethod(
'forbidden',
event.detail.element,
event.detail.controller.element,
event.detail.reflexId,
event.detail.payload
),
true
)

document.addEventListener(
'stimulus-reflex:after',
event =>
Expand Down Expand Up @@ -194,6 +208,7 @@ document.addEventListener(
// * success
// * error
// * halted
// * forbidden
// * after
// * finalize
//
Expand Down
38 changes: 35 additions & 3 deletions javascript/log.js
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ const request = (
})
}

const success = (event, halted) => {
const success = event => {
const { detail } = event || {}
const { selector, payload } = detail || {}
const { reflexId, target, morph } = detail.stimulusReflex || {}
Expand All @@ -40,10 +40,42 @@ const success = (event, halted) => {
.split('-')
.slice(1)
.join('_')
const output = { reflexId, morph, payload }
if (operation !== 'dispatch_event') output.operation = operation
console.log(
`\u2193 reflex \u2193 ${target} \u2192 ${selector ||
'\u221E'}${progress} ${duration}`,
{ reflexId, morph, operation, halted, payload }
output
)
}

const halted = event => {
leastbad marked this conversation as resolved.
Show resolved Hide resolved
const { detail } = event || {}
const { reflexId, target, payload } = detail.stimulusReflex || {}
const reflex = reflexes[reflexId]
if (Debug.disabled || reflex.promise.data.suppressLogging) return
const duration = reflex.timestamp
? `in ${new Date() - reflex.timestamp}ms`
: 'CLONED'
console.log(
`\u2193 reflex \u2193 ${target} ${duration} %cHALTED`,
'color: #ffa500;',
{ reflexId, payload }
)
}

const forbidden = event => {
const { detail } = event || {}
const { reflexId, target, payload } = detail.stimulusReflex || {}
const reflex = reflexes[reflexId]
if (Debug.disabled || reflex.promise.data.suppressLogging) return
const duration = reflex.timestamp
? `in ${new Date() - reflex.timestamp}ms`
: 'CLONED'
console.log(
`\u2193 reflex \u2193 ${target} ${duration} %cFORBIDDEN`,
'color: #BF40BF;',
{ reflexId, payload }
)
}

Expand All @@ -62,4 +94,4 @@ const error = event => {
)
}

export default { request, success, error }
export default { request, success, halted, forbidden, error }
6 changes: 1 addition & 5 deletions javascript/stimulus_reflex.js
Original file line number Diff line number Diff line change
Expand Up @@ -260,8 +260,4 @@ document.addEventListener('cable-ready:after-inner-html', afterDOMUpdate)
document.addEventListener('cable-ready:after-morph', afterDOMUpdate)
window.addEventListener('load', setupDeclarativeReflexes)

export {
initialize,
register,
useReflex
}
export { initialize, register, useReflex }
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,16 @@ export default class extends ApplicationController {
// element.innerText = "\nCouldn\'t dance!"
// }

// danceForbidden(element, reflex, noop, reflexId) {
// console.warn('danceForbidden');
// element.innerText = "\nDancing is forbidden in Bomont."
// }

// danceHalted(element, reflex, noop, reflexId) {
// console.warn('danceHalted');
// element.innerText = "\nNobody puts Baby in a corner."
// }

// afterDance(element, reflex, noop, reflexId) {
// element.innerText = '\nWhatever that was, it\'s over now.'
// }
Expand All @@ -89,6 +99,10 @@ export default class extends ApplicationController {
// console.warn("<%= action %> halted", element, reflex, reflexId)
// }

// <%= "#{action}_forbidden".camelize(:lower) %>(element, reflex, noop, reflexId) {
// console.warn("<%= action %> forbidden", element, reflex, reflexId)
// }

// <%= "after_#{action}".camelize(:lower) %>(element, reflex, noop, reflexId) {
// console.log("after <%= action %>", element, reflex, reflexId)
// }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,12 @@ export default class extends Controller {
// show error message
}

reflexHalted (element, reflex, error, reflexId) {
reflexForbidden (element, reflex, noop, reflexId) {
// Reflex action did not have permission to run
// window.location = '/'
}

reflexHalted (element, reflex, noop, reflexId) {
// handle aborted Reflex action
}

Expand Down
9 changes: 9 additions & 0 deletions lib/stimulus_reflex/broadcasters/broadcaster.rb
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,15 @@ def halted(data: {})
).broadcast
end

def forbidden(data: {})
leastbad marked this conversation as resolved.
Show resolved Hide resolved
operations << ["document", :dispatch_event]
cable_ready.dispatch_event(
name: "stimulus-reflex:morph-forbidden",
payload: payload,
stimulus_reflex: data.merge(morph: to_sym)
).broadcast
end

def error(data: {}, body: nil)
operations << ["document", :dispatch_event]
cable_ready.dispatch_event(
Expand Down
16 changes: 15 additions & 1 deletion lib/stimulus_reflex/callbacks.rb
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,21 @@ module Callbacks

included do
include ActiveSupport::Callbacks
define_callbacks :process, skip_after_callbacks_if_terminated: true
define_callbacks :process, skip_after_callbacks_if_terminated: true, terminator: ->(target, result_lambda) do
halted = true
forbidden = true
catch(:abort) do
catch(:forbidden) do
result_lambda.call
julianrubisch marked this conversation as resolved.
Show resolved Hide resolved
forbidden = false
end
halted = false
end
forbidden = false if halted == true
target.instance_variable_set(:@halted, halted)
target.instance_variable_set(:@forbidden, forbidden)
halted || forbidden
end
end

class_methods do
Expand Down
14 changes: 7 additions & 7 deletions lib/stimulus_reflex/reflex.rb
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ class VersionMismatchError < StandardError; end

delegate :connection, :stream_name, to: :channel
delegate :controller_class, :flash, :session, to: :request
delegate :broadcast, :halted, :error, to: :broadcaster
delegate :broadcast, :halted, :forbidden, :error, to: :broadcaster
delegate :reflex_id, :tab_id, :reflex_controller, :xpath_controller, :xpath_element, :permanent_attribute_name, :version, :suppress_logging, to: :client_attributes

def initialize(channel, url: nil, element: nil, selectors: [], method_name: nil, params: {}, client_attributes: {})
Expand Down Expand Up @@ -128,12 +128,7 @@ def render(*args)

# Invoke the reflex action specified by `name` and run all callbacks
def process(name, *args)
reflex_invoked = false
result = run_callbacks(:process) {
public_send(name, *args).tap { reflex_invoked = true }
}
@halted ||= result == false && !reflex_invoked
result
run_callbacks(:process) { public_send(name, *args) }
end

# Indicates if the callback chain was halted via a throw(:abort) in a before_reflex callback.
Expand All @@ -143,6 +138,11 @@ def halted?
!!@halted
end

# Indicates if the callback chain was halted via a throw(:forbidden) in a before_reflex callback.
def forbidden?
!!@forbidden
end

def default_reflex
# noop default reflex to force page reloads
end
Expand Down