I want to use transform to make a "virtual field" out of a collection. However, the new field I'm adding (within the transform function) is adding quite a bit of data to the returned document.
This is fine if the transform is taking place inside the client. If it is done on server-side, then there will be bandwidth concerns.
So I'm wondering if the transform is done on the server, or on the client, or it depends on how I find/fetch the document?
In Meteor a publication is a named API on the server that constructs a set of data to send to a client. A client initiates a subscription which connects to a publication, and receives that data.
Collections are Meteor's way of storing persistent data. The special thing about collections in Meteor is that they can be accessed from both the server and the client, making it easy to write view logic without having to write a lot of server code.
UPDATE: It's possible to do a transform on the server.
You can have a transform on the client like this:
return YourCollection.find({}, {transform: function (doc) {
doc.test = true;
return true;
}});
Meteor ignores transform
on queries that are published (from within Meteor.publish
). The client sees the document as if the transform didn't exist.
If you would like to use transforms on the server you can do this:
YourCollection = new Mongo.Collection("collection_name");
Meteor.publish("yourRecordSet", function() {
//Transform function
var transform = function(doc) {
doc.date = new Date();
return doc;
}
var self = this;
var observer = YourCollection.find().observe({
added: function (document) {
self.added('collection_name', document._id, transform(document));
},
changed: function (newDocument, oldDocument) {
self.changed('collection_name', newDocument._id, transform(newDocument));
},
removed: function (oldDocument) {
self.removed('collection_name', oldDocument._id);
}
});
self.onStop(function () {
observer.stop();
});
self.ready();
});
You can add and use the following function:
Meteor.publishWithTransform = function(publicationName, cursorGetter , transform)
{
transform = transform || function(o){return o;};
Meteor.publish(publicationName, function ()
{
var cursor = cursorGetter.apply(this, arguments);
var collectionName = cursor._cursorDescription.collectionName;
var self = this;
var observer = cursor.observe({
added: function (document) {
self.added(collectionName, document._id, transform(document));
},
changed: function (newDocument, oldDocument) {
self.changed(collectionName, newDocument._id, transform(newDocument));
},
removed: function (oldDocument) {
self.removed(collectionName, oldDocument._id);
}
});
self.onStop(function () {
observer.stop();
});
self.ready();
});
}
Usage (server side):
Meteor.publishWithTransform("publication-name", function () {
aCollection.find({})
}, function (item)
{
item.aField = "something";
return item;
});
Drawback: if you save the published document you will save the added changes as well. Here the client has no clue the property "aField" has been added during the publication.
You can use transform on both sides, when you specify a transform option to the collection or findOne,fetch,etc
transform Function
An optional transformation function. Documents will be passed through this function before being returned from fetch or findOne, and before being passed to callbacks of observe, allow, and deny.
If you need to get raw document from a collection with transform option.
myCollection.findOne({},{transform:null})
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With