diff options
-rw-r--r-- | share/www/script/test/replication.js | 17 | ||||
-rw-r--r-- | src/couchdb/couch_rep.erl | 83 |
2 files changed, 91 insertions, 9 deletions
diff --git a/share/www/script/test/replication.js b/share/www/script/test/replication.js index f069dc89..56698c59 100644 --- a/share/www/script/test/replication.js +++ b/share/www/script/test/replication.js @@ -150,6 +150,16 @@ couchTests.replication = function(debug) { } } }); + // make sure on design docs as well + dbA.save({ + _id:"_design/with_bin", + _attachments:{ + "foo.txt": { + "type":"base64", + "data": "VGhpcyBpcyBhIGJhc2U2NCBlbmNvZGVkIHRleHQ=" + } + } + }); }; this.afterAB1 = function(dbA, dbB) { @@ -158,6 +168,13 @@ couchTests.replication = function(debug) { xhr = CouchDB.request("GET", "/test_suite_db_b/bin_doc/foo.txt"); T(xhr.responseText == "This is a base64 encoded text") + + // and the design-doc + xhr = CouchDB.request("GET", "/test_suite_db_a/_design/with_bin/foo.txt"); + T(xhr.responseText == "This is a base64 encoded text") + + xhr = CouchDB.request("GET", "/test_suite_db_b/_design/with_bin/foo.txt"); + T(xhr.responseText == "This is a base64 encoded text") }; }, diff --git a/src/couchdb/couch_rep.erl b/src/couchdb/couch_rep.erl index 3647f6db..aa107cdf 100644 --- a/src/couchdb/couch_rep.erl +++ b/src/couchdb/couch_rep.erl @@ -327,9 +327,30 @@ attachment_loop(ReqId) -> couch_util:should_flush(), receive {From, {set_req_id, NewId}} -> + %% we learn the ReqId to listen for From ! {self(), {ok, NewId}}, attachment_loop(NewId); - {ibrowse_async_headers, ReqId, _Status, _Headers} -> + {ibrowse_async_headers, ReqId, Status, Headers} -> + %% we got header, give the controlling process a chance to react + receive + {From, gimme_status} -> + %% send status/headers to controller + From ! {self(), {status, Status, Headers}}, + receive + {From, continue} -> + %% normal case + attachment_loop(ReqId); + {From, fail} -> + %% error, failure code + ?LOG_ERROR( + "streaming attachment failed with status ~p", + [Status]), + exit(attachment_request_failed); + {From, stop_ok} -> + %% stop looping, controller will start a new loop + stop_ok + end + end, attachment_loop(ReqId); {ibrowse_async_response, ReqId, {chunk_start,_}} -> attachment_loop(ReqId); @@ -349,7 +370,19 @@ attachment_stub_converter(DbS, Id, {Name, {stub, Type, Length}}) -> % TODO worry about revisions Url = DbUrl ++ url_encode(Id) ++ "/" ++ ?b2l(Name), ?LOG_DEBUG("Attachment URL ~p", [Url]), + {ok, RcvFun} = make_attachment_stub_receiver(Url, Headers, Name, + Type, Length), + {Name, {Type, {RcvFun, Length}}}. + +make_attachment_stub_receiver(Url, Headers, Name, Type, Length) -> + make_attachment_stub_receiver(Url, Headers, Name, Type, Length, 10). + +make_attachment_stub_receiver(Url, _Headers, _Name, _Type, _Length, 0) -> + ?LOG_ERROR("streaming attachment request failed after 10 retries: ~s", + [Url]), + exit(attachment_request_failed); +make_attachment_stub_receiver(Url, Headers, Name, Type, Length, Retries) -> %% start the process that receives attachment data from ibrowse Pid = spawn_link(fun() -> attachment_loop(nil) end), @@ -364,14 +397,46 @@ attachment_stub_converter(DbS, Id, {Name, {stub, Type, Length}}) -> Pid ! {self(), {set_req_id, ReqId}}, receive {Pid, {ok, ReqId}} -> ok end, - %% this is the function that goes into the streaming attachment code. - %% It gets executed by the replication gen_server, so it can't - %% be the one to actually receive the ibrowse data. - RcvFun = fun() -> - Pid ! {self(), gimme_data}, - receive {Pid, Data} -> Data end - end, - {Name, {Type, {RcvFun, Length}}}. + %% wait for headers to ensure that we have a 200 status code + %% this is where we follow redirects etc + Pid ! {self(), gimme_status}, + receive {Pid, {status, StreamStatus, StreamHeaders}} -> + ?LOG_DEBUG("streaming attachment Status ~p Headers ~p", + [StreamStatus, StreamHeaders]), + + ResponseCode = list_to_integer(StreamStatus), + if + ResponseCode >= 200, ResponseCode < 300 -> + % the normal case + Pid ! {self(), continue}, + %% this function goes into the streaming attachment code. + %% It gets executed by the replication gen_server, so it can't + %% be the one to actually receive the ibrowse data. + {ok, fun() -> + Pid ! {self(), gimme_data}, + receive {Pid, Data} -> Data end + end}; + ResponseCode >= 300, ResponseCode < 400 -> + % follow the redirect + Pid ! {self(), stop_ok}, + RedirectUrl = mochiweb_headers:get_value("Location", + mochiweb_headers:make(StreamHeaders)), + make_attachment_stub_receiver(RedirectUrl, Headers, Name, Type, Length, + Retries - 1); + ResponseCode >= 400, ResponseCode < 500 -> + % an error... log and fail + ?LOG_ERROR("streaming attachment failed with code ~p: ~s", + [ResponseCode, Url]), + Pid ! {self(), fail}, + exit(attachment_request_failed); + ResponseCode == 500 -> + % an error... log and retry + ?LOG_INFO("retrying streaming attachment request due to 500 error: ~s", [Url]), + Pid ! {self(), fail}, + make_attachment_stub_receiver(Url, Headers, Name, Type, Length, + Retries - 1) + end + end. open_db({remote, Url, Headers})-> |