//////////////////////////////////////////////////////////////////////////////// /// DISCLAIMER /// /// Copyright 2014-2016 ArangoDB GmbH, Cologne, Germany /// Copyright 2004-2014 triAGENS GmbH, Cologne, Germany /// /// 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. /// /// Copyright holder is ArangoDB GmbH, Cologne, Germany /// /// @author Dr. Frank Celler /// @author Achim Brandt //////////////////////////////////////////////////////////////////////////////// #include "HttpServerJob.h" #include "Basics/WorkMonitor.h" #include "Dispatcher/DispatcherQueue.h" #include "HttpServer/AsyncJobManager.h" #include "HttpServer/HttpCommTask.h" #include "HttpServer/HttpHandler.h" #include "HttpServer/HttpServer.h" #include "Logger/Logger.h" #include "Scheduler/Scheduler.h" #include "Scheduler/SchedulerFeature.h" using namespace arangodb; using namespace arangodb::rest; //////////////////////////////////////////////////////////////////////////////// /// @brief constructs a new server job //////////////////////////////////////////////////////////////////////////////// HttpServerJob::HttpServerJob(HttpServer* server, WorkItem::uptr& handler, bool isAsync) : Job("HttpServerJob"), _server(server), _workDesc(nullptr), _isAsync(isAsync) { _handler.swap(handler); } //////////////////////////////////////////////////////////////////////////////// /// @brief destructs a server job //////////////////////////////////////////////////////////////////////////////// HttpServerJob::~HttpServerJob() { if (_workDesc != nullptr) { WorkMonitor::freeWorkDescription(_workDesc); } } size_t HttpServerJob::queue() const { return _handler->queue(); } void HttpServerJob::work() { TRI_ASSERT(_handler.get() != nullptr); RequestStatisticsAgent::transferTo(_handler.get()); LOG(TRACE) << "beginning job " << (void*)this; // the _handler needs to stay intact, so that we can cancel the job // therefore cannot use HandlerWorkStack here. Because we need to // keep the handler until the job is destroyed. Note that destroying // might happen in a different thread in case of shutdown. WorkMonitor::pushHandler(_handler.get()); try { _handler->executeFull(); if (_isAsync) { _handler->RequestStatisticsAgent::release(); _server->jobManager()->finishAsyncJob(_jobId, _handler->stealResponse()); } else { auto data = std::make_unique(); data->_taskId = _handler->taskId(); data->_loop = _handler->eventLoop(); data->_type = TaskData::TASK_DATA_RESPONSE; data->_response.reset(_handler->stealResponse()); _handler->RequestStatisticsAgent::transferTo(data.get()); SchedulerFeature::SCHEDULER->signalTask(data); } LOG(TRACE) << "finished job " << (void*)this; } catch (...) { _handler->requestStatisticsAgentSetExecuteError(); _workDesc = WorkMonitor::popHandler(_handler.release(), false); throw; } _workDesc = WorkMonitor::popHandler(_handler.release(), false); } bool HttpServerJob::cancel() { return _handler->cancel(); } void HttpServerJob::cleanup(DispatcherQueue* queue) { queue->removeJob(this); delete this; } void HttpServerJob::handleError(arangodb::basics::Exception const& ex) { _handler->handleError(ex); }