summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--test/query_server_spec.rb307
1 files changed, 307 insertions, 0 deletions
diff --git a/test/query_server_spec.rb b/test/query_server_spec.rb
new file mode 100644
index 00000000..ac519cca
--- /dev/null
+++ b/test/query_server_spec.rb
@@ -0,0 +1,307 @@
+# Licensed under the Apache License, Version 2.0 (the "License"); you may not
+# use this file except in compliance with the License. You may obtain a copy
+# of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+# License for the specific language governing permissions and limitations under
+# the License.
+
+
+# to run (requires ruby and rspec):
+# spec test/query_server_spec.rb -f specdoc --color
+
+COUCH_ROOT = "#{File.dirname(__FILE__)}/.." unless defined?(COUCH_ROOT)
+
+
+require 'open3'
+require 'spec'
+require 'json'
+
+
+JSON_REQ = {
+ "body"=>"undefined",
+ "verb"=>"GET",
+ "info"=>{
+ "disk_format_version"=>2,
+ "purge_seq"=>0,
+ "doc_count"=>9082,
+ "instance_start_time"=>"1243713611467271",
+ "update_seq"=>9512,
+ "disk_size"=>27541604,
+ "compact_running"=>false,
+ "db_name"=>"toast",
+ "doc_del_count"=>1
+ },
+ "cookie"=>{},
+ "form"=>{},
+ "query"=>{"q"=>"stuff"},
+ "path"=>["toast", "_ext"],
+ "headers"=>{
+ "User-Agent"=>"curl/7.18.1 (i386-apple-darwin9.2.2) libcurl/7.18.1 zlib/1.2.3",
+ "Host"=>"localhost:5984",
+ "Accept"=>"*/*"
+ }
+}
+
+class OSProcessRunner
+ def self.run
+ trace = false
+ puts "launching #{run_command}" if trace
+ if block_given?
+ Open3.popen3(run_command) do |jsin, jsout, jserr|
+ js = QueryServerRunner.new(jsin, jsout, jserr, trace)
+ yield js
+ end
+ else
+ jsin, jsout, jserr = Open3.popen3(run_command)
+ QueryServerRunner.new(jsin, jsout, jserr, trace)
+ end
+ end
+ def initialize jsin, jsout, jserr, trace = false
+ @qsin = jsin
+ @qsout = jsout
+ @qserr = jserr
+ @trace = trace
+ end
+ def close
+ @qsin.close
+ @qsout.close
+ @qserr.close
+ end
+ def reset!
+ run(["reset"])
+ end
+ def add_fun(fun)
+ run(["add_fun", fun])
+ end
+ def get_chunk
+ resp = jsgets
+ raise "not a chunk" unless resp.first == "chunk"
+ return resp[1]
+ end
+ def run json
+ rrun json
+ jsgets
+ end
+ def rrun json
+ line = json.to_json
+ puts "run: #{line}" if @trace
+ @qsin.puts line
+ end
+ def rgets
+ resp = @qsout.gets
+ puts "got: #{resp}" if @trace
+ resp
+ end
+ def jsgets
+ resp = rgets
+ # err = @qserr.gets
+ # puts "err: #{err}" if err
+ if resp
+ rj = JSON.parse("[#{resp.chomp}]")[0]
+ if rj.respond_to?(:[]) && !rj.is_a?(Array) &&
+ if rj["log"]
+ log = rj["log"]
+ puts "log: #{log}" #if @trace
+ rj = jsgets
+ end
+ end
+ rj
+ else
+ raise "no response"
+ end
+ end
+end
+
+class QueryServerRunner < OSProcessRunner
+ def self.run_command
+ "#{COUCH_ROOT}/src/couchdb/couchjs #{COUCH_ROOT}/share/server/main.js"
+ end
+end
+
+class ExternalRunner < OSProcessRunner
+ def self.run_command
+ "#{COUCH_ROOT}/src/couchdb/couchjs #{COUCH_ROOT}/share/server/echo.js"
+ end
+end
+
+describe "query server normal case" do
+ before(:all) do
+ `cd #{COUCH_ROOT} && make`
+ @qs = QueryServerRunner.run
+ end
+ after(:all) do
+ @qs.close
+ end
+ it "should reset" do
+ @qs.run(["reset"]).should == true
+ end
+ it "should run map funs" do
+ @qs.reset!
+ @qs.run(["add_fun", %{function(doc){emit("foo",doc.a); emit("bar",doc.a)}}]).should == true
+ @qs.run(["add_fun", %{function(doc){emit("baz",doc.a)}}]).should == true
+ rows = @qs.run(["map_doc", {:a => "b"}])
+ rows[0][0].should == ["foo", "b"]
+ rows[0][1].should == ["bar", "b"]
+ rows[1][0].should == ["baz", "b"]
+ end
+ describe "reduce" do
+ before(:all) do
+ @fun = <<-JS
+ function(keys, values, rereduce) {
+ return values.length;
+ }
+ JS
+ @qs.reset!
+ end
+ it "should reduce" do
+ kvs = (0...10).collect{|i|[i,i*2]}
+ @qs.run(["reduce", [@fun], kvs]).should == [true, [10]]
+ end
+ end
+ describe "rereduce" do
+ before(:all) do
+ @fun = <<-JS
+ function(keys, values, rereduce) {
+ return sum(values);
+ }
+ JS
+ @qs.reset!
+ end
+ it "should rereduce" do
+ vs = (0...10).collect{|i|i}
+ @qs.run(["rereduce", [@fun], vs]).should == [true, [45]]
+ end
+ end
+ # it "should validate"
+
+ describe "show" do
+ before(:all) do
+ @fun = <<-JS
+ function(doc, req) {
+ return [doc.title, doc.body].join(' - ')
+ }
+ JS
+ @qs.reset!
+ end
+ it "should show" do
+ @qs.rrun(["show_doc", @fun,
+ {:title => "Best ever", :body => "Doc body"}, JSON_REQ])
+ @qs.jsgets.should == {"body"=>"Best ever - Doc body"}
+ end
+ end
+
+ describe "show with headers" do
+ before(:all) do
+ @fun = <<-JS
+ function(doc, req) {
+ return {
+ headers : {"X-Plankton":"Rusty"},
+ body : [doc.title, doc.body].join(' - ')
+ }
+
+ }
+ JS
+ @qs.reset!
+ end
+ it "should show" do
+ @qs.rrun(["show_doc", @fun,
+ {:title => "Best ever", :body => "Doc body"}])
+ @qs.jsgets.should == {"headers"=>{"X-Plankton"=>"Rusty"}, "body"=>"Best ever - Doc body"}
+ end
+ end
+
+ describe "list with headers" do
+ before(:each) do
+ @fun = <<-JS
+ function(head, row, req) {
+ if (head) return {headers : {"Content-Type" : "text/plain"}, code : 200, "body" : "foo"};
+ if (row) return 'some "text" here';
+ return "tail";
+ };
+ JS
+ @qs.reset!
+ @qs.add_fun(@fun).should == true
+ end
+ it "should send head, row, and tail" do
+ @qs.rrun(["list_begin", {"total_rows"=>1000}, {"q" => "ok"}])
+ @qs.jsgets.should == {"headers"=>{"Content-Type"=>"text/plain"}, "code"=>200, "body"=>"foo"}
+ @qs.run(["list_row", {"foo"=>"bar"}, {"q" => "ok"}]).should == {"body"=>"some \"text\" here"}
+ @qs.run(["list_tail", {"foo"=>"bar"}, {"q" => "ok"}]).should == {"body"=>"tail"}
+ end
+ end
+
+ describe "list with headers and rows" do
+ before(:each) do
+ @fun = <<-JS
+ function(head, row, req) {
+ if (head) return {headers : {"Content-Type" : "text/plain"}, code : 200, "body" : "foo"};
+ if (row) return 'row value '+row.value;
+ return "tail "+req.q;
+ };
+ JS
+ @qs.reset!
+ @qs.add_fun(@fun).should == true
+ end
+ it "should render rows" do
+ @qs.rrun(["list_begin", {"total_rows"=>1000}, {"q" => "ok"}])
+ @qs.jsgets.should == {"headers"=>{"Content-Type"=>"text/plain"}, "code"=>200, "body"=>"foo"}
+ @qs.run(["list_row", {"value"=>"bar"}, {"q" => "ok"}]).should == {"body"=>"row value bar"}
+ @qs.run(["list_row", {"value"=>"baz"}, {"q" => "ok"}]).should == {"body"=>"row value baz"}
+ @qs.run(["list_row", {"value"=>"bam"}, {"q" => "ok"}]).should == {"body"=>"row value bam"}
+ @qs.run(["list_tail", {"q" => "ok"}]).should == {"body"=>"tail ok"}
+ end
+ end
+ end # query server normal case
+
+ describe "query server errors" do
+ before(:each) do
+ @qs = QueryServerRunner.run
+ end
+ after(:each) do
+ @qs.close
+ end
+
+ describe "list" do
+ before(:each) do
+ @fun = <<-JS
+ function(head, row, req) {
+ if (head) return {headers : {"Content-Type" : "text/plain"}, code : 200, "body" : "foo"};
+ if (row) return 'row value '+row.value;
+ return "tail "+req.q;
+ };
+ JS
+ @qs.run(["reset"]).should == true
+ @qs.add_fun(@fun).should == true
+ end
+ it "should reset in the middle" do
+ @qs.rrun(["list_begin", {"total_rows"=>1000}, {"q" => "ok"}])
+ @qs.jsgets.should == {"headers"=>{"Content-Type"=>"text/plain"}, "code"=>200, "body"=>"foo"}
+ @qs.run(["list_row", {"value"=>"bar"}, {"q" => "ok"}]).should == {"body"=>"row value bar"}
+ @qs.run(["reset"]).should == true
+ end
+ end
+ end #query server that errors
+
+## tests for the generic "echo" external
+
+# describe "running an external" do
+# before(:all) do
+# @ext = ExternalRunner.run
+#
+# end
+# it "should respond to 'info'" do
+# @ext.rrun(['info'])
+# @ext.jsgets.should == ["info", "echo", "external server that prints its arguments as JSON"]
+# end
+# it "should echo the request" do
+
+# @ext.rrun(['req', req_obj])
+# @ext.jsgets.should == ["x"]
+# end
+# end
+#