Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/meteor-community-packages/meteor-publish-composite
Meteor.publishComposite provides a flexible way to publish a set of related documents from various collections using a reactive join
https://github.com/meteor-community-packages/meteor-publish-composite
hacktoberfest meteor mongodb
Last synced: 2 days ago
JSON representation
Meteor.publishComposite provides a flexible way to publish a set of related documents from various collections using a reactive join
- Host: GitHub
- URL: https://github.com/meteor-community-packages/meteor-publish-composite
- Owner: Meteor-Community-Packages
- License: mit
- Created: 2014-02-08T19:14:49.000Z (almost 11 years ago)
- Default Branch: master
- Last Pushed: 2024-09-05T14:37:22.000Z (5 months ago)
- Last Synced: 2025-01-25T01:34:10.718Z (2 days ago)
- Topics: hacktoberfest, meteor, mongodb
- Language: JavaScript
- Homepage: https://atmospherejs.com/reywood/publish-composite
- Size: 636 KB
- Stars: 553
- Watchers: 15
- Forks: 58
- Open Issues: 12
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
# meteor-publish-composite
`publishComposite(...)` provides a flexible way to publish a set of related documents from various collections using a reactive join. This makes it easy to publish a whole tree of documents at once. The published collections are reactive and will update when additions/changes/deletions are made.
## Project
[![Project Status: Active – The project has reached a stable, usable state and is being actively developed.](https://www.repostatus.org/badges/latest/active.svg)](https://www.repostatus.org/#active)
![GitHub](https://img.shields.io/github/license/Meteor-Community-Packages/meteor-publish-composite)
[![JavaScript Style Guide](https://img.shields.io/badge/code_style-standard-brightgreen.svg)](https://standardjs.com)
![GitHub tag (latest SemVer)](https://img.shields.io/github/v/tag/Meteor-Community-Packages/meteor-publish-composite?label=latest&sort=semver)
[![](https://img.shields.io/badge/semver-2.0.0-success)](http://semver.org/spec/v2.0.0.html)
[![All Contributors](https://img.shields.io/badge/all_contributors-9-orange.svg?style=flat-square)](#contributors-)This project differs from many other parent/child relationship mappers in its flexibility. The relationship between a parent and its children can be based on almost anything. For example, let's say you have a site that displays news articles. On each article page, you would like to display a list at the end containing a couple of related articles. You could use `publishComposite` to publish the primary article, scan the body for keywords which are then used to search for other articles, and publish these related articles as children. Of course, the keyword extraction and searching are up to you to implement.
*Found a problem with this package? [See below for instructions on reporting](#reporting-issuesbugs).*
## Installation
```sh
$ meteor add reywood:publish-composite
```## Usage
This package exports a function on the server:
#### publishComposite(name, options)
Arguments
* **`name`** -- *string*
The name of the publication
* **`options`** -- *object literal or callback function*
An object literal specifying the configuration of the composite publication **or** a function that returns said object literal. If a function is used, it will receive the arguments passed to `Meteor.subscribe('myPub', arg1, arg2, ...)` (much like the `func` argument of [`Meteor.publish`](http://docs.meteor.com/#meteor_publish)). Basically, if your publication will take **no** arguments, pass an object literal for this argument. If your publication **will** take arguments, use a function that returns an object literal.
The object literal must have a `find` property, and can optionally have `children` and `collectionName` properties.
* **`find`** -- *function (required)*
A function that returns a MongoDB cursor (e.g., `return Meteor.users.find({ active: true });`)
* **`children`** -- *array (optional)* or *function*
- An array containing any number of object literals with this same structure
- A function with top level documents as arguments. It helps dynamically build
the array based on conditions ( like documents fields values)* **`collectionName`** -- *string (optional)*
A string specifying an alternate collection name to publish documents to (see [this blog post][blog-collection-name] for more details)
Example:
```javascript
{
find() {
// Must return a cursor containing top level documents
},
children: [
{
find(topLevelDocument) {
// Called for each top level document. Top level document is passed
// in as an argument.
// Must return a cursor of second tier documents.
},
children: [
{
collectionName: 'alt', // Docs from this find will be published to the 'alt' collection
find(secondTierDocument, topLevelDocument) {
// Called for each second tier document. These find functions
// will receive all parent documents starting with the nearest
// parent and working all the way up to the top level as
// arguments.
// Must return a cursor of third tier documents.
},
children: [
// Repeat as many levels deep as you like
]
}
]
},
{
find(topLevelDocument) {
// Also called for each top level document.
// Must return another cursor of second tier documents.
}
// The children property is optional at every level.
}
]
}
```Example with children as function:
```javascript
{
find() {
return Notifications.find();
},
children(parentNotification) {
// children is a function that returns an array of objects.
// It takes parent documents as arguments and dynamically builds children array.
if (parentNotification.type === 'about_post') {
return [{
find(notification) {
return Posts.find(parentNotification.objectId);
}
}];
}
return [
{
find(notification) {
return Comments.find(parentNotification.objectId);
}
}
]
}
}
```## Examples
### Example 1: A publication that takes **no** arguments.
First, we'll create our publication on the server.
```javascript
// Server
import { publishComposite } from 'meteor/reywood:publish-composite';publishComposite('topTenPosts', {
find() {
// Find top ten highest scoring posts
return Posts.find({}, { sort: { score: -1 }, limit: 10 });
},
children: [
{
find(post) {
// Find post author. Even though we only want to return
// one record here, we use "find" instead of "findOne"
// since this function should return a cursor.
return Meteor.users.find(
{ _id: post.authorId },
{ fields: { profile: 1 } });
}
},
{
find(post) {
// Find top two comments on post
return Comments.find(
{ postId: post._id },
{ sort: { score: -1 }, limit: 2 });
},
children: [
{
find(comment, post) {
// Find user that authored comment.
return Meteor.users.find(
{ _id: comment.authorId },
{ fields: { profile: 1 } });
}
}
]
}
]
});
```Next, we subscribe to our publication on the client.
```javascript
// Client
Meteor.subscribe('topTenPosts');
```Now we can use the published data in one of our templates.
```handlebars
Top Ten Posts
- {{title}} -- {{postAuthor.profile.name}}
{{#each posts}}
{{/each}}
```
```javascript
Template.topTenPosts.helpers({
posts() {
return Posts.find({}, { sort: { score: -1 }, limit: 10 });
},
postAuthor() {
// We use this helper inside the {{#each posts}} loop, so the context
// will be a post object. Thus, we can use this.authorId.
return Meteor.users.findOne(this.authorId);
}
})
```
### Example 2: A publication that **does** take arguments
Note a function is passed for the `options` argument to `publishComposite`.
```javascript
// Server
import { publishComposite } from 'meteor/reywood:publish-composite';
publishComposite('postsByUser', function(userId, limit) {
return {
find() {
// Find posts made by user. Note arguments for callback function
// being used in query.
return Posts.find({ authorId: userId }, { limit: limit });
},
children: [
// This section will be similar to that of the previous example.
]
}
});
```
```javascript
// Client
var userId = 1, limit = 10;
Meteor.subscribe('postsByUser', userId, limit);
```
### Example 3: A publication from async function
Note a function is passed for the `options` argument to `publishComposite`.
```javascript
// Server
import { publishComposite } from 'meteor/reywood:publish-composite';
publishComposite('postsByUser', async function(userId) {
const user = await Users.findOneAsync(userId)
const limit = user.limit
return {
find() {
// Find posts made by user. Note arguments for callback function
// being used in query.
return Posts.find({ authorId: userId }, { limit: limit });
},
children: [
// This section will be similar to that of the previous example.
]
}
});
```
## Known issues
**Avoid publishing very large sets of documents**
This package is great for publishing small sets of related documents. If you use it for large sets of documents with many child publications, you'll probably experience performance problems. Using this package to publish documents for a page with infinite scrolling is probably a bad idea. It's hard to offer exact numbers (i.e. don't publish more than X parent documents with Y child publications) so some experimentation may be necessary on your part to see what works for your application.
**Arrow functions**
You will not be able to access `this.userId` inside your `find` functions if you use [arrow functions](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Functions/Arrow_functions).
## Testing
Run the following:
```shell
meteor test-packages reywood:publish-composite --driver-package meteortesting:mocha
```
The tests are executing a combination of methods and subscriptions. The quickest option was to add a pause after each
operation (see usage of `sleep()` in `./tests/server.js`), to allow for the publications to send down all the
documents. However, this is flaky, so you may want to refresh the browser if you notice tests failing for no
apparent reason.
## Reporting issues/bugs
If you are experiencing an issue with this package, please create a GitHub repo with the simplest possible Meteor app that demonstrates the problem. This will go a long way toward helping me to diagnose the problem.
## More info
For more info on how to use `publishComposite`, check out these blog posts:
* [Publishing Reactive Joins in Meteor][blog-reactive-joins]
* [Publishing to an Alternative Client-side Collection in Meteor][blog-collection-name]
Note that these articles use the old pre-import notation, `Meteor.publishComposite`, which is still available for backward compatibility.
[blog-reactive-joins]: http://braindump.io/meteor/2014/09/12/publishing-reactive-joins-in-meteor.html
[blog-collection-name]: http://braindump.io/meteor/2014/09/20/publishing-to-an-alternative-clientside-collection-in-meteor.html
## Alternatives
While we are happy that you find this package of value, there are limitations, especially on high traffic applications.
There are also other solutions that can solve the problems that publish-composite solves, so here is a list of possible alternatives:
### MongoDB Aggregations
MongoDB itself has a functionality called Aggregations which allows you to combine data from multiple collections into
one document. It also has other useful features that you can utilize. The downside is that unless you use [reactive-aggregate](https://atmospherejs.com/tunguska/reactive-aggregate)
package the aggregations are not reactive and things it is not the easiest to learn or master.
* [Learn more](https://www.mongodb.com/docs/manual/meta/aggregation-quick-reference/)
### GraphQL
GraphQL allows you to specify exactly which data you need and even embed child documents. Apollo GraphQL also has an [official package]((https://atmospherejs.com/meteor/apollo))
and there is the `apollo` starter skeleton in Meteor itself to get you started quickly.
* [Official Meteor Apollo package](https://atmospherejs.com/meteor/apollo)
## Contributors ✨
Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):
Sean Dwyer
💻 📖 🤔
Seba Kerckhof
💻 👀 ⚠️
Richard Lai
🐛 💻
Simon Fridlund
💻
Patrick Lewis
💻
nabiltntn
💻
Krzysztof Czech
💻
Jan Dvorak
💻 📖 🚇 🚧 🔧
Koen [XII]
💻
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!