We previously presented a componentization model for cyclejs, which isolate the glueing of components into component combinators, from the application logic. We showed through miscellaneous examples that our component model is close in its syntax from React's component model, and is amenable to further syntactic sugaring (DSL, JSX, etc). The end result is an application logic which is more readable (i.e. easy to understand and reason about), more writable (i.e. easy to form a mental model and translate it into code).
The present effort focus on maintanability aspects linked to the chosen architecture. Maintainability is the ease by which a program can be altered to cope with reason for changes (in environment, specifications, etc.), and defects (bugs, poor performance, etc.).
Specifically, we are focusing here on adding capabilities in the program which allow to trace and debug a reactive system built with the chosen architecture.
We have :
run :: App -> Drivers -> Record {Sources, Sinks}
App :: Component
Drivers :: HashMap<DriverName, Driver>
Driver :: Sink -> Source
We use a traceRun
function which will take the run
function as parameter, and add the tracing
aspect to it. This is done by adding a sources.traceInputs
and sources.traceOutputs
dependencies into the sources
parameter.
Those trace
dependencies know how to trace inputs and outputs from any source or sink.
traceRun :: TraceFns -> Runnable -> TracedRunnable
traceRun :: TraceFns -> Runnable -> App -> Drivers -> Record {Sources, Sinks}
TraceFns :: HashMap<DriverName, TraceFn>
In m
, I can only trace the sources, not the stuff derived from the sources cause I dont know
what they are, so no way to attach an input to a component, except time precedence and same
context (same app/process/session/window)
But I can emit a log for each input by intervening the sources.source
. Just I won't know where
in the component tree that input is being read.
Unless I trace every sources in m
before passing it ?? could be
So :
m(Specs, Settings, Tree) :
-
Specs has trace information : NO, has to go through settings? yes otherwise I have to find a way to have a flatmap for log, i.e. traceInputs(tracedInputs)... merging the Specs.trace - more complex. Incorporate than in
m
instead?? separate it rather is better design -
m
body:- Tree.apply(traceInputs(Specs.trace, sources, settings)) (YES BOTH, I also need to include the settings in the log!)
- think about how the nesting would work, would not order be changed? to test
-
Remains the issue of tracing leaves of the component tree, i.e. Component
- is there a way to automate it ? Does not seem like so
- user should use traceInputs() himself to do so or a version of that
- same for traceOutputs
-
put trace info in settings._meta.trace etc.
function f (sources, settings){ const tracedSources = sources.traceInputs(sources, settings);
const events = makeEvents(sources, settings); // state can use events, for DRY reasons const state = makeState(events, sources, settings); const actions = makeActions(events, state, settings);
// settings can be used to parameterize the trace const tracedActions = sources.traceSinks(actions, settings);
return { sinkName : projectActionOnSink(sinkName, tracedActions), ... } }
Note that in cycle, drivers are called first with subjects. Then app with drivers sinks as source
traceRun :: TraceSpecs -> Runnable -> TracedRunnable
traceRun :: TraceSpecs -> Runnable -> App -> Drivers -> Record {Sources, Sinks}
TraceSpecs :: HashMap<DriverName, TraceFn>
const traceSpecs = { DriverName : traceFn, default? // will be necessary because of Pipe for instance... } const tracedRun = traceRun(traceSpecs, run); const {tracedSources, tracedSinks} = tracedRun(App, Drivers);
function traceRun (traceSpecs, run) { const driverNames = keys(traceSpecs);
return function tracedRun(app, drivers) { // weave the trace aspect in the run function // NOTE : could be wrote with a AOP library : this is output transformation function tracedApp (sources, settings){ // trace the inputs at top level : not super useful but done for full tracability const tracedSources = driverNames.reduce((acc, driverName) => { // TODO : guard against edge case (read or write only drivers) acc[driverName] = apply(traceSpecs[driverName].tracedInputs, topLevelTrace, sources[driverName]); return acc }, {}); // TO WRITE, should be about the same as previous line, so refactor later tracedSources.traceInputs = ... take all sources and add before advice tracedSources.traceSinks = ... take all sinks (should be easier, only observable) tracedSources.getId = ... closure to give unique id (incremental number)
const tracedSettings = settings;
return app(tracedSources, tracedSettings)
}
// trace the input of the drivers i.e. the sinks
// basically advice on input of driver
const tracedDrivers = driverNames.reduce((acc, driverName) => {
acc[driverName] = function (sink){
return apply(drivers[driverName], traceSpecs[driverName](sink))
}
return acc;
}, {});
return run(tracedApp, tracedDrivers)
}
}
- settings : {_trace : {combinatorName: ..., componentName: ..., id: ...}}
- !! NOTE :
m
should have pass a unique id to disambiguate possible conflicts, samem
running, same id, differntm
different id - id accumulate
- combinatorName accumulate combinators
- we have a combinator path from the graph we want to display
- this combined with the id mechanism gives us unicity of the path, so we can do the graph reliably
- this is necessary would it be that we have component or combinator with same name causing disambiguation issues
- componentName accumulate components so we have a component path (use with id to disambiguate)
- [App, MainPanel] in one branch
- [App, SidePanel] in another branch
- last one in array is current
- or could use other structure (zipper? [[before], latest])
For example :
const App =
// NOTE : manually add component name for now
InjectSourcesAndSettings({
_trace: {componentName : 'App'},
sourceFactory : sources => ({url$: ..., project$: ...})
}, [Div('.app'), [
SidePanel,
MainPanel
]])
// NOTE : Always put a named function for leaves components so it is picked up by the trace mechanism
// NOTE : that will force me to not use vLift for Div... mmmm. then write helper `NameFn(fn, name)`
function SidePanel(sources, settings) {
const { url$ } = sources;
const events = {
// NOTE : we avoid having to isolate by using the link which MUST be unique over the whole
// application (unicity of a route)
click: sources.DOM.select(`.navigation-section__link.${linkSanitized}`).events('click')
};
const state$ = url$
.map(url => ({isLinkActive : url.indexOf(link) > -1}))
.shareReplay(1);
const actions = {
domUpdate: state$.map(isLinkActive => {
const isLinkActiveClass = isLinkActive ? '.navigation-section__link--active' : '';
return a(
`${isLinkActiveClass}.navigation-item.navigation-section__link.${linkSanitized}`,
{ attrs: { href: link }, slot: 'navigation-item' },
title)
}),
router: events.click
.map(always('/' + link + '/'))
}
};
InjectSourcesAndSettings = m (specs, {_trace: {combinatorName : 'InjectSourcesAndSettings'}})
Given :
- INIT event (i.e. subscription of sinks in drivers, for instance DOM display)
- this is where behaviours may emit if on the path of a sink subscription
- here url$ emits its initial value ''
- componentName:[App-1], combinatorName : [Inject..], emits : {type: source, value: ''}, when:..., id : 1
- componentName:[App-1, SidePanel-3], combinatorName : 'Inject..', emits : {type: source, value: ''}, when:..., id : 3
- componentName:[App-1, SidePanel-3], combinatorName : 'Inject..', emits : {type: sink, value: ..}, when:..., id : 3
(id has to be with the stuff it disambiguates..., so App-1, SidePanel-3, instead of App,Sidepanel, 3)
combinatorName will be set by the combinator creator componentName will be set by the programmer
- m will traceInput sources with settings.id (if none then 0 - for root)
- then call children components with settings :
- id : old ID + push getID
- set componentName to component.name if no (outer)settings._trace.componentName (case leave
compponent)
- as a matter tree should be only combinator till leaves (to put in contract)