
Security News
Security Community Slams MIT-linked Report Claiming AI Powers 80% of Ransomware
Experts push back on new claims about AI-driven ransomware, warning that hype and sponsored research are distorting how the threat is understood.
cerebral-router
Advanced tools
An opinionated URL change handler for Cerebral
npm install cerebral-router
When you build your Cerebral application you do not have to think about routing at all. You only trigger signals that brings your application into the correct state. Let us imagine an application that can open a messages page, and also open specific messages.
import controller from './controller.js';
import homeOpened from './signals/homeOpened';
import messagesOpened from './signals/messagesOpened';
import messageOpened from './signals/messageOpened';
controller.signal('messagesOpened', messagesOpened);
controller.signal('messageOpened', messageOpened);
When we want to open the messages we call the signal:
onMessagesClick() {
  this.props.signals.messagesOpened();
}
When we want to open a single message we call the signal and pass a payload:
onMessageClick(id) {
  this.props.signals.messageOpened({
    id: id
  });
}
The signature of a state change is the signal and the payload passed. We can bind this signature to a route. Lets imagine we have implemented our whole application and it works great, we just need to update the addressbar with a url representing the current state of the application.
So let us also add a homeOpened signal so that we handle the root url as well.
import Router from 'cerebral-router';
import controller from './controller.js';
import homeOpened from './signals/homeOpened';
import messagesOpened from './signals/messagesOpened';
import messageOpened from './signals/messageOpened';
controller.signal('homeOpened', homeOpened);
controller.signal('messagesOpened', messagesOpened);
controller.signal('messageOpened', messageOpened);
Router(controller, {
  '/': 'homeOpened',
  '/messages': 'messagesOpened',
  '/messages/:id': 'messageOpened'
}, {
  mapper: { query: true } // Read about this below
});
Initial url would be handled automatically during application bootstrap if you are using cerebral-react (in container's componentDidMount method) or cerebral-angular (module's run section) packages.
Otherwise you should call trigger method to ensure that initial url is handled.
The router checks the url and fires the signal related to the url. The url will be parsed and any payload will be passed on the signal. That means if you go to example.com/messages/123 it will trigger the messageOpened signal with the payload {id: '123'}.
But if you click a message in the list it will also trigger the messageOpened signal with the payload {id: '456'} and now the url will also update to example.com/messages/456.
So it works both ways!
The important thing to understand here is that your application does not trigger urls to change its state. It triggers signals. Then you bind a route to a signal to allow a url to trigger the signal as well.
That means:
// Going to url
"example.com/messages/456"
// Is exactly the same as
this.props.signals.messageOpened({
  id: '456'
});
In the example above, when navigating in the app, you have to go to /messages before you can go to /messages/456.
But when you expose urls you could go directly to /messages/456. So how do you handle that?
...
controller.signal('messageOpened', [...messagesOpened, ...messageOpened]);
Router(controller, {
  '/': 'homeOpened',
  '/messages': 'messagesOpened',
  '/messages/:id': 'messageOpened'
});
With Cerebral you are already used to composing chains and actions together and this is also effective when creating routes. Now you might say, "I do not want to load my messages every time I open a message!". There are multiple ways to handle this. It depends on when you want to load the messages.
But lets say you want to load them whenever you actually go to /messages.
Inside your messagesOpened signal you can just check if there is an ID on the input.
If there is an ID it means you are about to open a message, if not it means you are just opening the messages.
With Cerebral you get a very powerful way to use queries.
But first we have to make a statement: "Queries are produced by your application, not by users".
With this perspective we can do some wonderful things. Lets get back to opening our message.
Inside the component opening the message we want to pass more than the ID of the message.
We want to pass: {withComments: true}. So that when we load the message, we load it with comments.
onMessageClick(id) {
  this.props.signals.messageOpened({
    id: id,
    withComments: true
  });
}
Since this signal is bound to a url Cerebral router will automatically make this part of the query, turning your url into example.com/messages/123?withComments:true.
That means if you refresh or pass the url to somebody else it will pass {id: '123', withComments: true} as the payload to the signal, opening the message in the exact same way, with the comments.
Notice here that we have withComments:true, not withComment=true.
This is because Cerebral router uses the URLON project to create serializable queries.
As you can see it is very powerful.
cerebral-router relies on url-mapper default behavior:
path-to-regexp library is used to define routes and payload parameters.
It tweaked to preserve payload parameters with Number and Boolean types, prepending it with colon.URLON notation if mapper: { query: true } option was passed to router.Routable part of url is extracted based on onlyHash and baseUrl options provided to router.
Router(controller, {
  '/': 'homeOpened',
  '/messages': 'messagesOpened',
  '/messages/:id': 'messageOpened'
}, {
  onlyHash: true,
  baseUrl: '/path',
  mapper: { query: true }
});
With given config and https://example.com/path#/messages/123?withComments:true url /messages/123?withComments:true is routable part.
Then url-mapper matches it to route /messages/:id and extracts payload {id: '123', withComments: true}.
Feel free to implement compileFn which return your own parse and stringify methods for given route.
The only recommendation is: parse method for previously stringified payload should result the same payload. Just like JSON.parse(JSON.stringify(object)).
Create an issue if you need this now since cerebral-router had to be patched to support custom mapper.
FAQs
An opinionated URL change handler for Cerebral
The npm package cerebral-router receives a total of 456 weekly downloads. As such, cerebral-router popularity was classified as not popular.
We found that cerebral-router demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 2 open source maintainers collaborating on the project.
Did you know?

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Security News
Experts push back on new claims about AI-driven ransomware, warning that hype and sponsored research are distorting how the threat is understood.

Security News
Ruby's creator Matz assumes control of RubyGems and Bundler repositories while former maintainers agree to step back and transfer all rights to end the dispute.

Research
/Security News
Socket researchers found 10 typosquatted npm packages that auto-run on install, show fake CAPTCHAs, fingerprint by IP, and deploy a credential stealer.