mirror of https://gitee.com/bigwinds/arangodb
259 lines
8.1 KiB
C++
259 lines
8.1 KiB
C++
////////////////////////////////////////////////////////////////////////////////
|
|
/// 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 Jan Steemann
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "MMFilesRestWalHandler.h"
|
|
|
|
#include "Basics/StringUtils.h"
|
|
#include "Basics/VelocyPackHelper.h"
|
|
#include "Cluster/ClusterFeature.h"
|
|
#include "Cluster/ClusterMethods.h"
|
|
#include "Cluster/ServerState.h"
|
|
#include "MMFiles/MMFilesLogfileManager.h"
|
|
|
|
using namespace arangodb;
|
|
using namespace arangodb::rest;
|
|
|
|
MMFilesRestWalHandler::MMFilesRestWalHandler(application_features::ApplicationServer& server,
|
|
GeneralRequest* request,
|
|
GeneralResponse* response)
|
|
: RestVocbaseBaseHandler(server, request, response) {}
|
|
|
|
RestStatus MMFilesRestWalHandler::execute() {
|
|
std::vector<std::string> const& suffixes = _request->suffixes();
|
|
|
|
if (suffixes.size() != 1) {
|
|
generateError(rest::ResponseCode::BAD, TRI_ERROR_HTTP_BAD_PARAMETER,
|
|
"expecting /_admin/wal/<operation>");
|
|
return RestStatus::DONE;
|
|
}
|
|
|
|
std::string const& operation = suffixes[0];
|
|
|
|
// extract the sub-request type
|
|
auto const type = _request->requestType();
|
|
|
|
if (operation == "transactions") {
|
|
if (type == rest::RequestType::GET) {
|
|
transactions();
|
|
return RestStatus::DONE;
|
|
}
|
|
} else if (operation == "flush") {
|
|
if (type == rest::RequestType::PUT) {
|
|
flush();
|
|
return RestStatus::DONE;
|
|
}
|
|
} else if (operation == "properties") {
|
|
if (type == rest::RequestType::GET || type == rest::RequestType::PUT) {
|
|
properties();
|
|
return RestStatus::DONE;
|
|
}
|
|
} else {
|
|
generateError(rest::ResponseCode::BAD, TRI_ERROR_HTTP_BAD_PARAMETER,
|
|
"expecting /_admin/wal/<operation>");
|
|
return RestStatus::DONE;
|
|
}
|
|
|
|
generateError(rest::ResponseCode::METHOD_NOT_ALLOWED, TRI_ERROR_HTTP_METHOD_NOT_ALLOWED);
|
|
return RestStatus::DONE;
|
|
}
|
|
|
|
void MMFilesRestWalHandler::properties() {
|
|
auto l = MMFilesLogfileManager::instance();
|
|
|
|
if (_request->requestType() == rest::RequestType::PUT) {
|
|
if (!ExecContext::current().isAdminUser()) {
|
|
generateError(rest::ResponseCode::FORBIDDEN, TRI_ERROR_HTTP_FORBIDDEN,
|
|
"you need admin rights to modify WAL properties");
|
|
return;
|
|
}
|
|
|
|
std::shared_ptr<VPackBuilder> parsedRequest;
|
|
VPackSlice slice;
|
|
try {
|
|
slice = _request->payload();
|
|
} catch (...) {
|
|
generateError(rest::ResponseCode::BAD, TRI_ERROR_HTTP_BAD_PARAMETER,
|
|
"invalid body value. expecting object");
|
|
return;
|
|
}
|
|
if (!slice.isObject()) {
|
|
generateError(rest::ResponseCode::BAD, TRI_ERROR_HTTP_BAD_PARAMETER,
|
|
"invalid body value. expecting object");
|
|
return;
|
|
}
|
|
|
|
if (slice.hasKey("allowOversizeEntries")) {
|
|
bool value = slice.get("allowOversizeEntries").getBoolean();
|
|
l->allowOversizeEntries(value);
|
|
}
|
|
|
|
if (slice.hasKey("logfileSize")) {
|
|
uint32_t value = slice.get("logfileSize").getNumericValue<uint32_t>();
|
|
l->filesize(value);
|
|
}
|
|
|
|
if (slice.hasKey("historicLogfiles")) {
|
|
uint32_t value = slice.get("historicLogfiles").getNumericValue<uint32_t>();
|
|
l->historicLogfiles(value);
|
|
}
|
|
|
|
if (slice.hasKey("reserveLogfiles")) {
|
|
uint32_t value = slice.get("reserveLogfiles").getNumericValue<uint32_t>();
|
|
l->reserveLogfiles(value);
|
|
}
|
|
|
|
if (slice.hasKey("throttleWait")) {
|
|
uint64_t value = slice.get("throttleWait").getNumericValue<uint64_t>();
|
|
l->maxThrottleWait(value);
|
|
}
|
|
|
|
if (slice.hasKey("throttleWhenPending")) {
|
|
uint64_t value = slice.get("throttleWhenPending").getNumericValue<uint64_t>();
|
|
l->throttleWhenPending(value);
|
|
}
|
|
}
|
|
|
|
VPackBuilder builder;
|
|
builder.openObject();
|
|
builder.add("allowOversizeEntries", VPackValue(l->allowOversizeEntries()));
|
|
builder.add("logfileSize", VPackValue(l->filesize()));
|
|
builder.add("historicLogfiles", VPackValue(l->historicLogfiles()));
|
|
builder.add("reserveLogfiles", VPackValue(l->reserveLogfiles()));
|
|
builder.add("syncInterval", VPackValue(l->syncInterval()));
|
|
builder.add("throttleWait", VPackValue(l->maxThrottleWait()));
|
|
builder.add("throttleWhenPending", VPackValue(l->throttleWhenPending()));
|
|
|
|
builder.close();
|
|
generateResult(rest::ResponseCode::OK, builder.slice());
|
|
}
|
|
|
|
void MMFilesRestWalHandler::flush() {
|
|
std::shared_ptr<VPackBuilder> parsedRequest;
|
|
VPackSlice slice;
|
|
try {
|
|
slice = _request->payload();
|
|
} catch (...) {
|
|
generateError(rest::ResponseCode::BAD, TRI_ERROR_HTTP_BAD_PARAMETER,
|
|
"invalid body value. expecting object");
|
|
return;
|
|
}
|
|
if (!slice.isObject() && !slice.isNone()) {
|
|
generateError(rest::ResponseCode::BAD, TRI_ERROR_HTTP_BAD_PARAMETER,
|
|
"invalid body value. expecting object");
|
|
}
|
|
|
|
bool waitForSync = false;
|
|
bool waitForCollector = false;
|
|
double maxWaitTime = 300.0;
|
|
|
|
if (slice.isObject()) {
|
|
// got a request body
|
|
VPackSlice value = slice.get("waitForSync");
|
|
if (value.isString()) {
|
|
waitForSync = basics::StringUtils::boolean(value.copyString());
|
|
} else if (value.isBoolean()) {
|
|
waitForSync = value.getBoolean();
|
|
}
|
|
|
|
value = slice.get("waitForCollector");
|
|
if (value.isString()) {
|
|
waitForCollector = basics::StringUtils::boolean(value.copyString());
|
|
} else if (value.isBoolean()) {
|
|
waitForCollector = value.getBoolean();
|
|
}
|
|
|
|
value = slice.get("maxWaitTime");
|
|
if (value.isNumber()) {
|
|
maxWaitTime = value.getNumericValue<double>();
|
|
}
|
|
} else {
|
|
// no request body
|
|
bool found;
|
|
{
|
|
std::string const& v = _request->value("waitForSync", found);
|
|
if (found) {
|
|
waitForSync = basics::StringUtils::boolean(v);
|
|
}
|
|
}
|
|
{
|
|
std::string const& v = _request->value("waitForCollector", found);
|
|
if (found) {
|
|
waitForCollector = basics::StringUtils::boolean(v);
|
|
}
|
|
}
|
|
{
|
|
std::string const& v = _request->value("maxWaitTime", found);
|
|
if (found) {
|
|
maxWaitTime = basics::StringUtils::doubleDecimal(v);
|
|
}
|
|
}
|
|
}
|
|
|
|
int res;
|
|
if (ServerState::instance()->isCoordinator()) {
|
|
auto& feature = server().getFeature<ClusterFeature>();
|
|
res = flushWalOnAllDBServers(feature, waitForSync, waitForCollector, maxWaitTime);
|
|
} else {
|
|
res = MMFilesLogfileManager::instance()->flush(waitForSync, waitForCollector,
|
|
false, maxWaitTime, true);
|
|
}
|
|
|
|
if (res != TRI_ERROR_NO_ERROR) {
|
|
THROW_ARANGO_EXCEPTION(res);
|
|
}
|
|
|
|
generateResult(rest::ResponseCode::OK, arangodb::velocypack::Slice::emptyObjectSlice());
|
|
}
|
|
|
|
void MMFilesRestWalHandler::transactions() {
|
|
auto const& info = MMFilesLogfileManager::instance()->runningTransactions();
|
|
|
|
VPackBuilder builder;
|
|
builder.openObject();
|
|
builder.add("runningTransactions", VPackValue(static_cast<double>(std::get<0>(info))));
|
|
|
|
// lastCollectedId
|
|
{
|
|
auto value = std::get<1>(info);
|
|
if (value == UINT64_MAX) {
|
|
builder.add("minLastCollected", VPackValue(VPackValueType::Null));
|
|
} else {
|
|
builder.add("minLastCollected", VPackValue(value));
|
|
}
|
|
}
|
|
|
|
// lastSealedId
|
|
{
|
|
auto value = std::get<2>(info);
|
|
if (value == UINT64_MAX) {
|
|
builder.add("minLastSealed", VPackValue(VPackValueType::Null));
|
|
} else {
|
|
builder.add("minLastSealed", VPackValue(value));
|
|
}
|
|
}
|
|
|
|
builder.close();
|
|
|
|
generateResult(rest::ResponseCode::OK, builder.slice());
|
|
}
|