2008-06-01 11:21:21 -07:00
#!/usr/bin/env ruby
commands = %w{pull push}
command = ARGV[0]
if !commands.include?(command)
puts <<-USAGE
2008-06-20 13:36:36 -07:00
Usage: couchview (pull|push) my-database-name
2008-06-01 11:21:21 -07:00
For help on pull and push run script/views (pull|push) without a database name.
USAGE
exit
end
if ARGV.length == 1
case command
when "pull"
puts <<-PULL
2008-06-20 13:36:36 -07:00
couchview pull my-database-name
2008-06-01 11:21:21 -07:00
I will automagically create a "views" directory in your current working directory if none exists.
Then I copy the design documents and views into a directory structure like:
./views/my-design-doc/view-name-map.js
./views/my-design-doc/view-name-reduce.js
If your view names don't end in "map" or "reduce" I'll add those suffixes as a pull. On push I'll put them in new locations corresponding to these new names (overwriting the old design documents). I'm opinionated, but if these conventions don't work for you, the source code is right here.
PULL
when "push"
puts <<-PUSH
2008-06-20 13:36:36 -07:00
couchview push my-database-name
2008-06-01 11:21:21 -07:00
I'll push all the files in your views directory to the specified database. Because CouchDB caches the results of view calculation by function content, there's no performance penalty for duplicating the map function twice, which I'll do if you have a reduce function. This makes it possible to browse the results of just the map, which can be useful for both queries and debugging.
./views/my-design-doc/view-name-map.js
./views/my-design-doc/view-name-reduce.js
Pushed to =>
http://localhost:5984/my-database-name/_design/my-design-doc
{
"views" : {
"view-name-map" : {
"map" : "### contents of view-name-map.js ###"
},
"view-name-reduce" : {
"map" : "### contents of view-name-map.js ###",
"reduce" : "### contents of view-name-reduce.js ###"
},
}
}
PUSH
end
exit
end
dbname = ARGV[1]
2008-06-09 16:01:24 -07:00
dirname = ARGV[2] || "views"
puts "Running #{command} on #{dbname} from directory #{dirname}."
2008-06-01 11:21:21 -07:00
require File.expand_path(File.dirname(__FILE__)) + '/../couchrest'
2008-06-01 12:23:04 -07:00
require 'fileutils'
module Enumerable
def group_by
inject({}) do |groups, element|
(groups[yield(element)] ||= []) << element
groups
end
end if RUBY_VERSION < '1.9'
end
2008-06-01 11:21:21 -07:00
# connect to couchdb
cr = CouchRest.new("http://localhost:5984")
db = cr.database(dbname)
2008-06-01 12:23:04 -07:00
def readjs(file, libs=nil)
st = open(file).read
st.sub!(/\/\/include-lib/,libs) if libs
2008-06-20 13:25:22 -07:00
st
2008-06-01 12:23:04 -07:00
end
2008-06-01 11:21:21 -07:00
2008-06-01 12:23:04 -07:00
case command
when "push" # files to views
views = {}
2008-06-09 16:01:24 -07:00
viewfiles = Dir.glob(File.join(dirname,"**","*.js")) # todo support non-js views
2008-06-01 12:23:04 -07:00
libfiles = viewfiles.select{|f|/lib\.js/.match(f)}
libs = open(libfiles[0]).read if libfiles[0]
all = (viewfiles-libfiles).collect do |file|
fileparts = file.split('/')
filename = /(\w.*)-(\w.*)\.js/.match file.split('/').pop
design = fileparts[1]
view = filename[1]
func = filename[2]
path = file
[design,view,func,path]
end
designs = all.group_by{|f|f[0]}
designs.each do |design,parts|
2008-06-07 08:32:51 -07:00
# puts "replace _design/#{design}? (enter to proceed, 'n' to skip)"
# rep = $stdin.gets.chomp
# next if rep == 'n'
2008-06-01 12:23:04 -07:00
dviews = {}
parts.group_by{|p|p[1]}.each do |view,fs|
fs.each do |f|
dviews["#{view}-reduce"] ||= {}
dviews["#{view}-reduce"][f[2]] = readjs(f.last,libs)
2008-06-01 11:21:21 -07:00
end
2008-06-07 08:32:51 -07:00
dviews["#{view}-map"] = {'map' => dviews["#{view}-reduce"]['map']}
2008-06-01 12:23:04 -07:00
dviews.delete("#{view}-reduce") unless dviews["#{view}-reduce"]["reduce"]
2008-06-01 11:21:21 -07:00
end
2008-06-01 12:23:04 -07:00
# save them to the db
2008-07-04 16:56:09 -07:00
begin
view = db.get("_design/#{design}")
rescue RestClient::Request::RequestFailed
view = nil
end
2008-06-07 08:32:51 -07:00
if (view && view['views'] == dviews)
puts "no change to _design/#{design}. skipping..."
else
puts "replacing _design/#{design}"
db.delete(view) rescue nil
db.save({
"_id" => "_design/#{design}",
:views => dviews
})
2008-06-01 12:23:04 -07:00
end
2008-06-01 11:21:21 -07:00
end
2008-06-01 12:23:04 -07:00
when "pull" # views to files
ds = db.documents(:startkey => '_design/', :endkey => '_design/ZZZZZZZZZ')
ds['rows'].collect{|r|r['id']}.each do |id|
puts directory = id.split('/').last
FileUtils.mkdir_p(File.join("views",directory))
views = db.get(id)['views']
vgroups = views.keys.group_by{|k|k.sub(/\-(map|reduce)$/,'')}
vgroups.each do|g,vs|
mapname = vs.find {|v|views[v]["map"]}
if mapname
# save map
mapfunc = views[mapname]["map"]
2008-06-09 16:01:24 -07:00
mapfile = File.join(dirname,directory,"#{g}-map.js") # todo support non-js views
2008-06-01 12:23:04 -07:00
File.open(mapfile,'w') do |f|
f.write mapfunc
end
end
reducename = vs.find {|v|views[v]["reduce"]}
if reducename
# save reduce
reducefunc = views[reducename]["reduce"]
2008-06-09 16:01:24 -07:00
reducefile = File.join(dirname,directory,"#{g}-reduce.js") # todo support non-js views
2008-06-01 12:23:04 -07:00
File.open(reducefile,'w') do |f|
f.write reducefunc
end
end
end
end
end