Reactive API extensions for Feathers
feathers-reactive adds a watch() method to services. The returned object implements all service methods as RxJS observables that automatically update on real-time events.
The following options are supported:
idField(mandatory): The id property field of your services. Depends on your service/database. Usually 'id' (SQL, Rethinkdb, …) or '_id' (MongoDB, NeDB, … ).dataField(default:data): The data property field in paginated responseslistStrategy(default:smart): The strategy to use for streaming the data. Can besmart,alwaysorneversorter(function(query, options) {}): A function that returns a sorting function for the given query and option including pagination and limiting. Does not need to be customized unless there is a sorting mechanism other than Feathers standard in place.matcher(function(query)): A function that returns a function which returns whether an item matches the original query or not.pipe(operator | operator[]) One or multiple rxjs operators of the formfunction(observable) => observablelike you would pass them to an Observable's .pipe method. The supplied operators are applied to any Observable created byfeathers-reactive.options.pipe: tap(data => console.log(data))would log every emitted value to the console.
const feathers = require('feathers');
const reactive = require('feathers-reactive');
const app = feathers().configure(reactive(options));With feathers-reactive configured on the application individual options can be set at the service level with service.rx:
// Set a different id field and always re-fetch data
app.service('todos').rx({
idField: '_id'
});Each method call can also pass its own options via params.rx:
// Always fetch fresh data for this method call
app.service('todos').watch({ listStrategy: 'always' }).find();List strategies are used to determine how a data stream behaves. Currently there are three strategies:
never- Returns a stream from the service promise that only emits the method call data and never updates after thatsmart(default) - Returns a stream that smartly emits updated list data based on the services real-time events. It does not re-query any new data (but does not cover some cases in which thealwaysstrategy can be used).always- Re-runs the original query to always get fresh data from the server on any matching real-time event.
const feathers = require('feathers');
const memory = require('feathers-memory');
const rx = require('feathers-reactive');
const app = feathers()
.configure(rx({idField: "id"}))
.use('/messages', memory());
const messages = app.service('messages');
messages.create({
text: 'A test message'
}).then(() => {
// Get and watch a specific message with id 0. Emit the message data once it resolves
// and every time it changes e.g. through an updated or patched event
messages.watch().get(0).subscribe(message => console.log('My message', message));
// Find and watch all messages and emit a new list every time anything changes
messages.watch().find().subscribe(messages => console.log('Message list', messages));
/* Find and watch all messages with querying functionality.
NOTE: make sure "value" is in the correct variable type (string, boolean, int, float) otherwise create event will not trigger the console log.
*/
messages.watch({ listStrategy: 'always' }).find({query: {text: 'A test message'}})
.subscribe(messages => console.log('Message list with query', messages));
messages.create({ text: 'Another message' }).then(() =>
setTimeout(() => messages.patch(0, { text: 'Updated message' }), 1000)
);
});Will output:
My message { text: 'A test message', id: 0 }
Message list [ { text: 'A test message', id: 0 } ]
Message list [ { text: 'A test message', id: 0 },
{ text: 'Another message', id: 1 } ],
Message list [ { text: 'A test message', id: 0 } ]
Message list with query { text: 'A test message', id: 0 }
Message list [ { text: 'Updated message', id: 0 },
{ text: 'Another message', id: 1 } ]
Let's assume a simple server in app.js like this:
npm install feathers feathers-socketio feathers-memory feathers-errors/handler
const feathers = require('feathers');
const socketio = require('feathers-socketio');
const memory = require('feathers-memory');
const handler = require('feathers-errors/handler');
const app = feathers()
.configure(socketio())
.use('/todos', memory())
.use('/', feathers.static(__dirname))
.use(handler());
app.listen(3030);A real-time ReactJS Todo application (with Bootstrap styles) can look like this (see the examples/react-todos folder for a working example);
npm install react react-dom rxjs
import React from 'react';
import ReactDOM from 'react-dom';
import feathers from 'feathers/client';
import socketio from 'feathers-socketio/client';
import rx from 'feathers-reactive';
import io from 'socket.io-client';
const socket = io();
const app = feathers()
.configure(socketio(socket))
.configure(rx());
const todos = app.service('todos');
const TodoApp = React.createClass({
getInitialState() {
return {
todos: [],
text: ''
};
},
componentDidMount() {
this.todos = todos.watch().find().subscribe(todos => this.setState({ todos }));
},
componentWillUnmount() {
this.todos.unsubscribe();
},
updateText(ev) {
this.setState({ text: ev.target.value });
},
createTodo(ev) {
todos.create({
text: this.state.text,
complete: false
});
this.setState({ text: '' });
ev.preventDefault();
},
updateTodo(todo, ev) {
todo.complete = ev.target.checked;
todos.patch(todo.id, todo);
},
deleteTodo(todo) {
todos.remove(todo.id);
},
render() {
const renderTodo = todo =>
<li className={`page-header checkbox ${todo.complete ? 'done' : ''}`}>
<label>
<input type="checkbox" onChange={this.updateTodo.bind(this, todo)}
checked={todo.complete} />
{todo.text}
</label>
<a href="javascript://" className="pull-right delete"
onClick={this.deleteTodo.bind(this, todo)}>
<span className="glyphicon glyphicon-remove"></span>
</a>
</li>;
return <div className="container" id="todos">
<h1>Feathers real-time Todos</h1>
<ul className="todos list-unstyled">{this.state.todos.map(renderTodo)}</ul>
<form role="form" className="create-todo" onSubmit={this.createTodo}>
<div className="form-group">
<input type="text" className="form-control" name="description"
placeholder="Add a new Todo" onChange={this.updateText}
value={this.state.text} />
</div>
<button type="submit" className="btn btn-info col-md-12">
Add Todo
</button>
</form>
</div>;
}
});
ReactDOM.render(<TodoApp />, document.getElementById('app'));Copyright (c) 2018
Licensed under the MIT license.
