So how do you do this? Well, first you need to grab the code:
Then edit jpath.js so that, at the bottom, instead of assigning to module.exports it creates an object called jpath and adds the select and filter methods to it.curl -o underscore.js http://underscorejs.org/underscore-min.js curl -o jpath.js https://raw.github.com/stsvilik/node-jpath/master/lib/node-jpath.js
Then you'll need a table to load the modules from, and to put the modules in this table. You'll also need a function to load them. These can be accomplished with the following psql script:
\set underscore `cat underscore-min.js`
\set jpath `cat jpath.js`
create table plv8_modules(modname text primary key, load_on_start boolean, code text);
insert into plv8_modules values ('underscore',true,:'underscore'),
    ('jpath',true,:'jpath');
create or replace function plv8_startup()
returns void
language plv8
as
$$
load_module = function(modname)
{
    var rows = plv8.execute("SELECT code from plv8_modules " +
                            " where modname = $1", [modname]);
    for (var r = 0; r < rows.length; r++)
    {
        var code = rows[r].code;
        eval("(function() { " + code + "})")();
    }
        
};
$$;
Now test it out:
select plv8_startup();
do language plv8 ' load_module("underscore"); load_module("jpath"); ';
-- test the underscore module's extend function
do language plv8 $$ 
   x = {'a':1}; 
   y=_.extend(x,{'a':2,'b':3},{'b':4,'c':5}); 
   plv8.elog(NOTICE,JSON.stringify(y)); 
$$;
-- test jpath module's filter function
do language plv8 $$
  var jsonData = {
    people: [
        {name: "John", age:26, gender:"male"},
        {name: "Steve", age:24, gender:"male"},
        {name: "Susan", age:22, gender:"female"},
        {name: "Linda", age:30, gender:"female"},
        {name: "Adam", age:32, gender:"male"}
    ]
  };
  //We want to get all males younger then 25
  var match = jpath.filter(jsonData, "people[gender=male && age < 25]");
  plv8.elog(NOTICE,JSON.stringify(match));
$$;
NOTICE:  {"a":2,"b":4,"c":5}
DO
NOTICE:  [{"name":"Steve","age":24,"gender":"male"}]
DO
// now load all the modules marked for loading on start
var rows = plv8.execute("SELECT modname, code from plv8_modules where load_on_start");
for (var r = 0; r < rows.length; r++)
{
 var code = rows[r].code;
 eval("(function() { " + code + "})")();
}
to you postgresql.conf file, and restart. You should now be able to use these modules anywhere in your PLv8 code.plv8.start_proc = 'plv8_startup'
 
Using Extension Templates it's going to be even easier shipping those extra JS files in 9.3, and with the Event Triggers capabilities it will even be possible to include the whole fetching and template installation as part of the CREATE EXTENSION command... stay tuned :)
ReplyDeleteThis comment has been removed by the author.
ReplyDeletesorry, not working only
ReplyDeleteplv8.start_proc = 'plv8_startup'
in postgresql.conf!
Thanks for the great idea! :)
Hmm, it looks like you need to mangle jpath some to make it postgres friendly. I must have forgotten that when I wrote this post.
ReplyDeleteThanks for letting me know. Maybe I will fork it for use with PLV8.
Actually, I did mention that in the post, and you seem to have missed it: "Then edit jpath.js so that, at the bottom, instead of assigning to module.exports it creates an object called jpath and adds the select and filter methods to it."
DeleteMy memory must be totally fading.
DeleteUse https://raw.github.com/adunstan/plv8-jpath/master/lib/plv8-jpath.js and it should load cleanly
Thanks, that works!
ReplyDeletealso this: for (var r = 0; r > rows.length; r++)
ReplyDeleteshould be: for (var r = 0; r < rows.length; r++)
Fixed, thanks.
DeleteYou might want to look at this project. It's less painful. It includes unit testing, migrations, logging, bundling ... https://github.com/mgutz/plv8-bedrock
ReplyDeleteHi Andrew,
ReplyDeleteI used some of the code as reference for an example from a book about Underscore.js that I am working on. I hope that is ok - you can browse my code version at https://github.com/popalexandruvasile/underscorejs-examples/blob/master/browser-server-database/postgresql-underscore/insertSeedData.sql
Thank you for you blog post I could not have linked Underscore to PostgreSQL without it.
Based on this article, I've made this extension : https://github.com/slardiere/plv8_modules
ReplyDeleteThis is great. Do you happen to have any suggestions how to load this automatically on startup and have it available on all connections? For some reason, anytime I make a new connection to the database (from the application or the command line), it acts like the load_module function hasn't been run. If I run it manually, it works for that connection, but doesn't for others. Thanks!
ReplyDeleteYou set it in via plv8.start_proc in postgresql.conf, as the post says.
DeleteI use the instance of postgresql provided from Amazon RDS, so I can't edit `postgresql.conf` to add plv8.start_proc = 'plv8_startup'', is there some other way to do this?
ReplyDeleteSetting it on the database via ALTER DATABASE might work. Try it and see.
DeleteJust for the record: yes it does: "ALTER DATABASE dbname SET plv8.start_proc = 'plv8_startup';"
DeleteYeah. I just tried and got a permissions error. I don't know any other way, sorry.
Deletecurrently it is not clear where i should store grabbed js code? in which location?
ReplyDelete