
Research
Two Malicious Rust Crates Impersonate Popular Logger to Steal Wallet Keys
Socket uncovers malicious Rust crates impersonating fast_log to steal Solana and Ethereum wallet keys from source code.
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
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.
Research
Socket uncovers malicious Rust crates impersonating fast_log to steal Solana and Ethereum wallet keys from source code.
Research
A malicious package uses a QR code as steganography in an innovative technique.
Research
/Security News
Socket identified 80 fake candidates targeting engineering roles, including suspected North Korean operators, exposing the new reality of hiring as a security function.