#include "etcd/v3/AsyncWatchAction.hpp" #include "etcd/v3/action_constants.hpp" using etcdserverpb::RangeRequest; using etcdserverpb::RangeResponse; using etcdserverpb::WatchCreateRequest; etcdv3::AsyncWatchAction::AsyncWatchAction( etcdv3::ActionParameters const ¶m) : etcdv3::Action(param) { isCancelled.store(false); stream = parameters.watch_stub->AsyncWatch(&context,&cq_,(void*)etcdv3::WATCH_CREATE); WatchRequest watch_req; WatchCreateRequest watch_create_req; watch_create_req.set_key(parameters.key); watch_create_req.set_prev_kv(true); watch_create_req.set_start_revision(parameters.revision); if(parameters.withPrefix) { if (parameters.key.empty()) { watch_create_req.set_range_end(detail::string_plus_one(etcdv3::NUL)); } else { watch_create_req.set_range_end(detail::string_plus_one(parameters.key)); } } if(!parameters.range_end.empty()) { watch_create_req.set_range_end(parameters.range_end); } watch_req.mutable_create_request()->CopyFrom(watch_create_req); // wait "create" success (the stream becomes ready) void *got_tag; bool ok = false; if (cq_.Next(&got_tag, &ok) && ok && got_tag == (void *)etcdv3::WATCH_CREATE) { stream->Write(watch_req, (void *)etcdv3::WATCH_WRITE); } else { throw std::runtime_error("failed to create a watch connection"); } // wait "write" (WatchCreateRequest) success, and start to read the first reply if (cq_.Next(&got_tag, &ok) && ok && got_tag == (void *)etcdv3::WATCH_WRITE) { stream->Read(&reply, (void*)this); } else { throw std::runtime_error("failed to write WatchCreateRequest to server"); } } void etcdv3::AsyncWatchAction::waitForResponse() { void* got_tag; bool ok = false; while(cq_.Next(&got_tag, &ok)) { if(ok == false) { break; } if(isCancelled.load()) { break; } if(got_tag == (void*)etcdv3::WATCH_WRITES_DONE) { isCancelled.store(true); cq_.Shutdown(); break; } if(got_tag == (void*)this) // read tag { if (reply.canceled()) { isCancelled.store(true); cq_.Shutdown(); break; } else if ((reply.created() && reply.header().revision() < parameters.revision) || reply.events_size() > 0) { // we stop watch under two conditions: // // 1. watch for a future revision, return immediately with empty events set // 2. receive any effective events. isCancelled.store(true); stream->WritesDone((void*)etcdv3::WATCH_WRITES_DONE); grpc::Status status; stream->Finish(&status, (void *)this); cq_.Shutdown(); // leave a warning if the response is too large and been fragmented if (reply.fragment()) { std::cerr << "WARN: The response hasn't been fully received and parsed" << std::endl; } break; } else { // otherwise, start next round read-reply stream->Read(&reply, (void*)this); } } } } void etcdv3::AsyncWatchAction::CancelWatch() { if (!isCancelled.exchange(true)) { stream->WritesDone((void*)etcdv3::WATCH_WRITES_DONE); grpc::Status status; stream->Finish(&status, (void *)this); cq_.Shutdown(); } } bool etcdv3::AsyncWatchAction::Cancelled() const { return isCancelled.load(); } void etcdv3::AsyncWatchAction::waitForResponse(std::function callback) { void* got_tag; bool ok = false; while(cq_.Next(&got_tag, &ok)) { if(ok == false) { break; } if(isCancelled.load()) { break; } if(got_tag == (void*)etcdv3::WATCH_WRITES_DONE) { isCancelled.store(true); cq_.Shutdown(); break; } else if(got_tag == (void*)this) // read tag { if (reply.canceled()) { isCancelled.store(true); cq_.Shutdown(); if (reply.compact_revision() != 0) { auto resp = etcd::Response(grpc::StatusCode::OUT_OF_RANGE /* error code */, "required revision has been compacted"); resp._compact_revision = reply.compact_revision(); callback(resp); } break; } if(reply.events_size()) { // for the callback case, we don't stop immediately if watching for a future revison, // we wait until there are some expected events. auto resp = ParseResponse(); auto duration = std::chrono::duration_cast( std::chrono::high_resolution_clock::now() - start_timepoint); callback(etcd::Response(resp, duration)); start_timepoint = std::chrono::high_resolution_clock::now(); } stream->Read(&reply, (void*)this); } } } etcdv3::AsyncWatchResponse etcdv3::AsyncWatchAction::ParseResponse() { AsyncWatchResponse watch_resp; watch_resp.set_action(etcdv3::WATCH_ACTION); if(!status.ok()) { watch_resp.set_error_code(status.error_code()); watch_resp.set_error_message(status.error_message()); } else { watch_resp.ParseResponse(reply); } return watch_resp; }