2010-03-26 23:29:11 +00:00
|
|
|
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
|
|
|
|
// vim: ts=8 sw=2 smarttab
|
|
|
|
/*
|
|
|
|
* Ceph - scalable distributed file system
|
|
|
|
*
|
|
|
|
* Copyright (C) 2004-2009 Sage Weil <sage@newdream.net>
|
|
|
|
*
|
|
|
|
* This is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License version 2.1, as published by the Free Software
|
|
|
|
* Foundation. See file COPYING.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2010-06-12 13:04:11 +00:00
|
|
|
#ifndef CEPH_RGW_COMMON_H
|
|
|
|
#define CEPH_RGW_COMMON_H
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-03-08 21:49:56 +00:00
|
|
|
#include "common/ceph_crypto.h"
|
2011-05-10 21:08:42 +00:00
|
|
|
#include "common/debug.h"
|
2011-10-11 21:00:42 +00:00
|
|
|
#include "common/perf_counters.h"
|
2011-10-09 22:27:10 +00:00
|
|
|
|
|
|
|
#include "acconfig.h"
|
|
|
|
#ifdef FASTCGI_INCLUDE_DIR
|
|
|
|
# include "fastcgi/fcgiapp.h"
|
|
|
|
#else
|
|
|
|
# include "fcgiapp.h"
|
|
|
|
#endif
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-06-07 21:13:59 +00:00
|
|
|
#include <errno.h>
|
2010-12-03 22:45:59 +00:00
|
|
|
#include <string.h>
|
2009-07-18 00:49:59 +00:00
|
|
|
#include <string>
|
|
|
|
#include <map>
|
|
|
|
#include "include/types.h"
|
2011-03-09 00:49:49 +00:00
|
|
|
#include "include/utime.h"
|
2009-07-18 00:49:59 +00:00
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
2011-08-04 21:43:55 +00:00
|
|
|
namespace ceph {
|
|
|
|
class Formatter;
|
|
|
|
}
|
|
|
|
|
2011-03-16 17:57:48 +00:00
|
|
|
using ceph::crypto::MD5;
|
|
|
|
|
2011-10-04 22:36:15 +00:00
|
|
|
|
2011-04-13 17:38:11 +00:00
|
|
|
#define RGW_ROOT_BUCKET ".rgw"
|
|
|
|
|
2011-07-01 00:13:42 +00:00
|
|
|
#define RGW_CONTROL_BUCKET ".rgw.control"
|
|
|
|
|
2009-08-07 17:46:29 +00:00
|
|
|
#define RGW_ATTR_PREFIX "user.rgw."
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2009-08-07 17:46:29 +00:00
|
|
|
#define RGW_ATTR_ACL RGW_ATTR_PREFIX "acl"
|
|
|
|
#define RGW_ATTR_ETAG RGW_ATTR_PREFIX "etag"
|
2010-03-11 22:49:27 +00:00
|
|
|
#define RGW_ATTR_BUCKETS RGW_ATTR_PREFIX "buckets"
|
2009-08-07 17:46:29 +00:00
|
|
|
#define RGW_ATTR_META_PREFIX RGW_ATTR_PREFIX "x-amz-meta-"
|
|
|
|
#define RGW_ATTR_CONTENT_TYPE RGW_ATTR_PREFIX "content_type"
|
2011-08-02 23:29:16 +00:00
|
|
|
#define RGW_ATTR_ID_TAG RGW_ATTR_PREFIX "idtag"
|
2011-08-03 23:59:33 +00:00
|
|
|
#define RGW_ATTR_SHADOW_OBJ RGW_ATTR_PREFIX "shadow_name"
|
2012-03-01 22:03:57 +00:00
|
|
|
#define RGW_ATTR_MANIFEST RGW_ATTR_PREFIX "manifest"
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-02-28 23:32:05 +00:00
|
|
|
#define RGW_BUCKETS_OBJ_PREFIX ".buckets"
|
|
|
|
|
2011-10-11 17:49:27 +00:00
|
|
|
#define USER_INFO_VER 8
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-06-21 00:09:27 +00:00
|
|
|
#define RGW_MAX_CHUNK_SIZE (512*1024)
|
|
|
|
#define RGW_MAX_PENDING_CHUNKS 16
|
2012-01-13 20:19:35 +00:00
|
|
|
#define RGW_MAX_PUT_SIZE (5ULL*1024*1024*1024)
|
2010-07-19 23:50:43 +00:00
|
|
|
|
2011-02-16 01:30:07 +00:00
|
|
|
#define RGW_FORMAT_XML 1
|
|
|
|
#define RGW_FORMAT_JSON 2
|
|
|
|
|
2011-09-02 21:36:54 +00:00
|
|
|
#define RGW_REST_SWIFT 0x1
|
|
|
|
#define RGW_REST_SWIFT_AUTH 0x2
|
2011-03-23 23:50:21 +00:00
|
|
|
|
2011-06-17 16:26:32 +00:00
|
|
|
#define RGW_SUSPENDED_USER_AUID (uint64_t)-2
|
2011-06-16 23:53:38 +00:00
|
|
|
|
2011-03-09 00:49:49 +00:00
|
|
|
#define CGI_PRINTF(state, format, ...) do { \
|
|
|
|
int __ret = FCGX_FPrintF(state->fcgx->out, format, __VA_ARGS__); \
|
|
|
|
if (state->header_ended) \
|
2011-04-29 23:14:22 +00:00
|
|
|
state->bytes_sent += __ret; \
|
2011-07-26 21:27:36 +00:00
|
|
|
int l = 32, n; \
|
|
|
|
while (1) { \
|
|
|
|
char __buf[l]; \
|
|
|
|
n = snprintf(__buf, sizeof(__buf), format, __VA_ARGS__); \
|
|
|
|
if (n != l) \
|
2012-01-09 19:40:44 +00:00
|
|
|
dout(10) << "--> " << __buf << dendl; \
|
2011-07-26 21:27:36 +00:00
|
|
|
break; \
|
|
|
|
l *= 2; \
|
|
|
|
} \
|
2011-02-16 01:30:07 +00:00
|
|
|
} while (0)
|
|
|
|
|
2011-04-29 23:14:22 +00:00
|
|
|
#define CGI_PutStr(state, buf, len) do { \
|
|
|
|
FCGX_PutStr(buf, len, state->fcgx->out); \
|
|
|
|
if (state->header_ended) \
|
|
|
|
state->bytes_sent += len; \
|
|
|
|
} while (0)
|
2011-02-16 01:30:07 +00:00
|
|
|
|
2011-04-30 00:17:23 +00:00
|
|
|
#define CGI_GetStr(state, buf, buf_len, olen) do { \
|
|
|
|
olen = FCGX_GetStr(buf, buf_len, state->fcgx->in); \
|
|
|
|
state->bytes_received += olen; \
|
|
|
|
} while (0)
|
|
|
|
|
2011-10-27 00:20:51 +00:00
|
|
|
#define STATUS_CREATED 1900
|
|
|
|
#define STATUS_ACCEPTED 1901
|
|
|
|
#define STATUS_NO_CONTENT 1902
|
|
|
|
#define STATUS_PARTIAL_CONTENT 1903
|
|
|
|
|
|
|
|
#define ERR_INVALID_BUCKET_NAME 2000
|
|
|
|
#define ERR_INVALID_OBJECT_NAME 2001
|
|
|
|
#define ERR_NO_SUCH_BUCKET 2002
|
|
|
|
#define ERR_METHOD_NOT_ALLOWED 2003
|
|
|
|
#define ERR_INVALID_DIGEST 2004
|
|
|
|
#define ERR_BAD_DIGEST 2005
|
|
|
|
#define ERR_UNRESOLVABLE_EMAIL 2006
|
|
|
|
#define ERR_INVALID_PART 2007
|
|
|
|
#define ERR_INVALID_PART_ORDER 2008
|
|
|
|
#define ERR_NO_SUCH_UPLOAD 2009
|
|
|
|
#define ERR_REQUEST_TIMEOUT 2010
|
|
|
|
#define ERR_LENGTH_REQUIRED 2011
|
|
|
|
#define ERR_REQUEST_TIME_SKEWED 2012
|
|
|
|
#define ERR_BUCKET_EXISTS 2013
|
|
|
|
#define ERR_BAD_URL 2014
|
|
|
|
#define ERR_PRECONDITION_FAILED 2015
|
|
|
|
#define ERR_NOT_MODIFIED 2016
|
|
|
|
#define ERR_INVALID_UTF8 2017
|
|
|
|
#define ERR_UNPROCESSABLE_ENTITY 2018
|
2012-01-13 20:19:35 +00:00
|
|
|
#define ERR_TOO_LARGE 2019
|
2011-10-27 00:20:51 +00:00
|
|
|
#define ERR_USER_SUSPENDED 2100
|
|
|
|
#define ERR_INTERNAL_ERROR 2200
|
2011-06-16 23:53:38 +00:00
|
|
|
|
2009-08-07 17:46:29 +00:00
|
|
|
typedef void *RGWAccessHandle;
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-10-11 21:00:42 +00:00
|
|
|
|
|
|
|
/* perf counter */
|
|
|
|
|
|
|
|
extern PerfCounters *perfcounter;
|
|
|
|
|
|
|
|
extern int rgw_perf_start(CephContext *cct);
|
|
|
|
extern void rgw_perf_stop(CephContext *cct);
|
|
|
|
|
|
|
|
enum {
|
|
|
|
l_rgw_first = 15000,
|
|
|
|
l_rgw_req,
|
2011-11-08 17:49:22 +00:00
|
|
|
l_rgw_failed_req,
|
2011-10-11 21:00:42 +00:00
|
|
|
|
|
|
|
l_rgw_get,
|
|
|
|
l_rgw_get_b,
|
|
|
|
l_rgw_get_lat,
|
|
|
|
|
|
|
|
l_rgw_put,
|
|
|
|
l_rgw_put_b,
|
|
|
|
l_rgw_put_lat,
|
|
|
|
|
|
|
|
l_rgw_qlen,
|
|
|
|
l_rgw_qactive,
|
|
|
|
|
2011-11-08 17:49:22 +00:00
|
|
|
l_rgw_cache_hit,
|
|
|
|
l_rgw_cache_miss,
|
|
|
|
|
2011-10-11 21:00:42 +00:00
|
|
|
l_rgw_last,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2011-05-27 21:22:55 +00:00
|
|
|
/* size should be the required string size + 1 */
|
|
|
|
extern int gen_rand_base64(char *dest, int size);
|
|
|
|
extern int gen_rand_alphanumeric(char *dest, int size);
|
|
|
|
extern int gen_rand_alphanumeric_upper(char *dest, int size);
|
|
|
|
|
2011-08-08 23:44:13 +00:00
|
|
|
enum RGWIntentEvent {
|
2012-01-31 01:00:37 +00:00
|
|
|
DEL_OBJ = 0,
|
|
|
|
DEL_DIR = 1,
|
2011-08-08 23:44:13 +00:00
|
|
|
};
|
|
|
|
|
2011-09-24 00:11:49 +00:00
|
|
|
enum RGWObjCategory {
|
|
|
|
RGW_OBJ_CATEGORY_NONE = 0,
|
|
|
|
RGW_OBJ_CATEGORY_MAIN = 1,
|
|
|
|
RGW_OBJ_CATEGORY_SHADOW = 2,
|
|
|
|
RGW_OBJ_CATEGORY_MULTIMETA = 3,
|
|
|
|
};
|
|
|
|
|
2010-03-11 22:49:27 +00:00
|
|
|
/** Store error returns for output at a different point in the program */
|
2009-08-07 17:46:29 +00:00
|
|
|
struct rgw_err {
|
2011-04-14 23:14:48 +00:00
|
|
|
rgw_err();
|
2011-04-15 17:52:14 +00:00
|
|
|
rgw_err(int http, const std::string &s3);
|
2011-04-14 23:14:48 +00:00
|
|
|
void clear();
|
|
|
|
bool is_clear() const;
|
2011-04-15 18:15:11 +00:00
|
|
|
bool is_err() const;
|
2011-04-14 23:14:48 +00:00
|
|
|
friend std::ostream& operator<<(std::ostream& oss, const rgw_err &err);
|
|
|
|
|
2011-04-15 17:52:14 +00:00
|
|
|
int http_ret;
|
2011-05-17 11:15:10 +00:00
|
|
|
int ret;
|
2011-04-15 17:52:14 +00:00
|
|
|
std::string s3_code;
|
2011-04-14 23:14:48 +00:00
|
|
|
std::string message;
|
2009-07-18 00:49:59 +00:00
|
|
|
};
|
|
|
|
|
2010-03-11 22:49:27 +00:00
|
|
|
/* Helper class used for XMLArgs parsing */
|
2009-07-18 00:49:59 +00:00
|
|
|
class NameVal
|
|
|
|
{
|
|
|
|
string str;
|
|
|
|
string name;
|
|
|
|
string val;
|
|
|
|
public:
|
|
|
|
NameVal(string nv) : str(nv) {}
|
|
|
|
|
|
|
|
int parse();
|
|
|
|
|
|
|
|
string& get_name() { return name; }
|
|
|
|
string& get_val() { return val; }
|
|
|
|
};
|
|
|
|
|
2010-03-11 22:49:27 +00:00
|
|
|
/** Stores the XML arguments associated with the HTTP request in req_state*/
|
2009-07-18 00:49:59 +00:00
|
|
|
class XMLArgs
|
|
|
|
{
|
|
|
|
string str, empty_str;
|
|
|
|
map<string, string> val_map;
|
2011-05-27 20:35:47 +00:00
|
|
|
map<string, string> sub_resources;
|
2009-07-18 00:49:59 +00:00
|
|
|
public:
|
2011-09-30 06:17:12 +00:00
|
|
|
XMLArgs() {}
|
|
|
|
XMLArgs(string s) : str(s) {}
|
|
|
|
/** Set the arguments; as received */
|
|
|
|
void set(string s) { val_map.clear(); sub_resources.clear(); str = s; }
|
|
|
|
/** parse the received arguments */
|
|
|
|
int parse();
|
|
|
|
/** Get the value for a specific argument parameter */
|
|
|
|
string& get(string& name);
|
|
|
|
string& get(const char *name);
|
|
|
|
/** see if a parameter is contained in this XMLArgs */
|
|
|
|
bool exists(const char *name) {
|
|
|
|
map<string, string>::iterator iter = val_map.find(name);
|
|
|
|
return (iter != val_map.end());
|
|
|
|
}
|
|
|
|
bool sub_resource_exists(const char *name) {
|
|
|
|
map<string, string>::iterator iter = sub_resources.find(name);
|
|
|
|
return (iter != sub_resources.end());
|
|
|
|
}
|
|
|
|
map<string, string>& get_sub_resources() { return sub_resources; }
|
2009-07-18 00:49:59 +00:00
|
|
|
};
|
|
|
|
|
2011-07-11 20:41:40 +00:00
|
|
|
class RGWConf;
|
|
|
|
|
|
|
|
class RGWEnv {
|
|
|
|
std::map<string, string> env_map;
|
|
|
|
public:
|
|
|
|
RGWConf *conf;
|
|
|
|
|
2011-07-11 21:26:16 +00:00
|
|
|
RGWEnv();
|
2011-07-11 20:41:40 +00:00
|
|
|
~RGWEnv();
|
2011-07-19 21:02:11 +00:00
|
|
|
void init(char **envp);
|
2011-07-11 20:41:40 +00:00
|
|
|
const char *get(const char *name, const char *def_val = NULL);
|
|
|
|
int get_int(const char *name, int def_val = 0);
|
2011-07-14 22:55:05 +00:00
|
|
|
bool get_bool(const char *name, bool def_val = 0);
|
2011-07-11 20:41:40 +00:00
|
|
|
size_t get_size(const char *name, size_t def_val = 0);
|
|
|
|
};
|
|
|
|
|
|
|
|
class RGWConf {
|
|
|
|
friend class RGWEnv;
|
|
|
|
protected:
|
|
|
|
void init(RGWEnv * env);
|
|
|
|
public:
|
|
|
|
RGWConf() :
|
2011-07-11 23:55:05 +00:00
|
|
|
should_log(1) {}
|
2011-07-11 20:41:40 +00:00
|
|
|
|
|
|
|
int should_log;
|
|
|
|
};
|
|
|
|
|
2009-07-18 00:49:59 +00:00
|
|
|
enum http_op {
|
|
|
|
OP_GET,
|
|
|
|
OP_PUT,
|
|
|
|
OP_DELETE,
|
|
|
|
OP_HEAD,
|
2011-05-26 22:18:48 +00:00
|
|
|
OP_POST,
|
2011-11-10 22:56:10 +00:00
|
|
|
OP_COPY,
|
2009-07-18 00:49:59 +00:00
|
|
|
OP_UNKNOWN,
|
|
|
|
};
|
|
|
|
|
2009-08-07 17:46:29 +00:00
|
|
|
class RGWAccessControlPolicy;
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-05-20 22:15:48 +00:00
|
|
|
struct RGWAccessKey {
|
|
|
|
string id;
|
|
|
|
string key;
|
|
|
|
string subuser;
|
|
|
|
|
|
|
|
RGWAccessKey() {}
|
|
|
|
void encode(bufferlist& bl) const {
|
|
|
|
__u32 ver = 1;
|
|
|
|
::encode(ver, bl);
|
|
|
|
::encode(id, bl);
|
|
|
|
::encode(key, bl);
|
|
|
|
::encode(subuser, bl);
|
|
|
|
}
|
|
|
|
|
|
|
|
void decode(bufferlist::iterator& bl) {
|
|
|
|
__u32 ver;
|
|
|
|
::decode(ver, bl);
|
|
|
|
::decode(id, bl);
|
|
|
|
::decode(key, bl);
|
|
|
|
::decode(subuser, bl);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
WRITE_CLASS_ENCODER(RGWAccessKey);
|
|
|
|
|
|
|
|
struct RGWSubUser {
|
|
|
|
string name;
|
2011-05-23 22:12:48 +00:00
|
|
|
uint32_t perm_mask;
|
2011-05-20 22:15:48 +00:00
|
|
|
|
2011-05-23 22:12:48 +00:00
|
|
|
RGWSubUser() : perm_mask(0) {}
|
2011-05-20 22:15:48 +00:00
|
|
|
void encode(bufferlist& bl) const {
|
|
|
|
__u32 ver = 1;
|
|
|
|
::encode(ver, bl);
|
|
|
|
::encode(name, bl);
|
2011-05-23 22:12:48 +00:00
|
|
|
::encode(perm_mask, bl);
|
2011-05-20 22:15:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void decode(bufferlist::iterator& bl) {
|
|
|
|
__u32 ver;
|
|
|
|
::decode(ver, bl);
|
|
|
|
::decode(name, bl);
|
2011-05-23 22:12:48 +00:00
|
|
|
::decode(perm_mask, bl);
|
2011-05-20 22:15:48 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
WRITE_CLASS_ENCODER(RGWSubUser);
|
|
|
|
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2009-08-07 17:46:29 +00:00
|
|
|
struct RGWUserInfo
|
2009-07-18 00:49:59 +00:00
|
|
|
{
|
2010-05-07 21:33:42 +00:00
|
|
|
uint64_t auid;
|
2009-07-18 00:49:59 +00:00
|
|
|
string user_id;
|
|
|
|
string display_name;
|
|
|
|
string user_email;
|
2011-05-20 22:15:48 +00:00
|
|
|
map<string, RGWAccessKey> access_keys;
|
2011-10-11 17:49:27 +00:00
|
|
|
map<string, RGWAccessKey> swift_keys;
|
2011-05-20 22:15:48 +00:00
|
|
|
map<string, RGWSubUser> subusers;
|
2011-06-16 23:53:38 +00:00
|
|
|
__u8 suspended;
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-06-24 19:55:25 +00:00
|
|
|
RGWUserInfo() : auid(0), suspended(0) {}
|
2010-03-26 23:29:11 +00:00
|
|
|
|
2009-07-18 00:49:59 +00:00
|
|
|
void encode(bufferlist& bl) const {
|
|
|
|
__u32 ver = USER_INFO_VER;
|
|
|
|
::encode(ver, bl);
|
2010-03-26 23:29:11 +00:00
|
|
|
::encode(auid, bl);
|
2011-05-20 22:15:48 +00:00
|
|
|
string access_key;
|
|
|
|
string secret_key;
|
|
|
|
if (!access_keys.empty()) {
|
|
|
|
map<string, RGWAccessKey>::const_iterator iter = access_keys.begin();
|
|
|
|
const RGWAccessKey& k = iter->second;
|
|
|
|
access_key = k.id;
|
|
|
|
secret_key = k.key;
|
|
|
|
}
|
2011-04-16 00:20:44 +00:00
|
|
|
::encode(access_key, bl);
|
2009-07-18 00:49:59 +00:00
|
|
|
::encode(secret_key, bl);
|
|
|
|
::encode(display_name, bl);
|
|
|
|
::encode(user_email, bl);
|
2011-10-11 17:49:27 +00:00
|
|
|
string swift_name;
|
|
|
|
string swift_key;
|
|
|
|
if (!swift_keys.empty()) {
|
|
|
|
map<string, RGWAccessKey>::const_iterator iter = swift_keys.begin();
|
|
|
|
const RGWAccessKey& k = iter->second;
|
|
|
|
swift_name = k.id;
|
|
|
|
swift_key = k.key;
|
|
|
|
}
|
2011-09-02 21:56:50 +00:00
|
|
|
::encode(swift_name, bl);
|
|
|
|
::encode(swift_key, bl);
|
2011-04-16 00:20:44 +00:00
|
|
|
::encode(user_id, bl);
|
2011-05-20 22:15:48 +00:00
|
|
|
::encode(access_keys, bl);
|
|
|
|
::encode(subusers, bl);
|
2011-06-16 23:53:38 +00:00
|
|
|
::encode(suspended, bl);
|
2011-10-11 17:49:27 +00:00
|
|
|
::encode(swift_keys, bl);
|
2009-07-18 00:49:59 +00:00
|
|
|
}
|
|
|
|
void decode(bufferlist::iterator& bl) {
|
|
|
|
__u32 ver;
|
|
|
|
::decode(ver, bl);
|
2010-03-26 23:29:11 +00:00
|
|
|
if (ver >= 2) ::decode(auid, bl);
|
|
|
|
else auid = CEPH_AUTH_UID_DEFAULT;
|
2011-05-20 22:15:48 +00:00
|
|
|
string access_key;
|
|
|
|
string secret_key;
|
2011-04-16 00:20:44 +00:00
|
|
|
::decode(access_key, bl);
|
2009-07-18 00:49:59 +00:00
|
|
|
::decode(secret_key, bl);
|
2011-05-20 22:15:48 +00:00
|
|
|
if (ver < 6) {
|
|
|
|
RGWAccessKey k;
|
|
|
|
k.id = access_key;
|
|
|
|
k.key = secret_key;
|
|
|
|
access_keys[access_key] = k;
|
|
|
|
}
|
2009-07-18 00:49:59 +00:00
|
|
|
::decode(display_name, bl);
|
|
|
|
::decode(user_email, bl);
|
2011-10-11 17:49:27 +00:00
|
|
|
string swift_name;
|
|
|
|
string swift_key;
|
2011-09-02 21:56:50 +00:00
|
|
|
if (ver >= 3) ::decode(swift_name, bl);
|
|
|
|
if (ver >= 4) ::decode(swift_key, bl);
|
2011-04-16 00:20:44 +00:00
|
|
|
if (ver >= 5)
|
|
|
|
::decode(user_id, bl);
|
|
|
|
else
|
|
|
|
user_id = access_key;
|
2011-05-20 22:15:48 +00:00
|
|
|
if (ver >= 6) {
|
|
|
|
::decode(access_keys, bl);
|
|
|
|
::decode(subusers, bl);
|
|
|
|
}
|
2011-06-16 23:53:38 +00:00
|
|
|
suspended = 0;
|
|
|
|
if (ver >= 7) {
|
|
|
|
::decode(suspended, bl);
|
|
|
|
}
|
2011-10-11 17:49:27 +00:00
|
|
|
if (ver >= 8) {
|
|
|
|
::decode(swift_keys, bl);
|
|
|
|
}
|
2009-07-18 00:49:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void clear() {
|
|
|
|
user_id.clear();
|
|
|
|
display_name.clear();
|
|
|
|
user_email.clear();
|
2010-03-26 23:29:11 +00:00
|
|
|
auid = CEPH_AUTH_UID_DEFAULT;
|
2011-05-20 22:15:48 +00:00
|
|
|
access_keys.clear();
|
2011-06-16 23:53:38 +00:00
|
|
|
suspended = 0;
|
2009-07-18 00:49:59 +00:00
|
|
|
}
|
|
|
|
};
|
2009-08-07 17:46:29 +00:00
|
|
|
WRITE_CLASS_ENCODER(RGWUserInfo)
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-08-18 00:11:55 +00:00
|
|
|
struct rgw_bucket {
|
|
|
|
std::string name;
|
|
|
|
std::string pool;
|
2011-09-19 23:41:44 +00:00
|
|
|
std::string marker;
|
2011-09-28 17:50:48 +00:00
|
|
|
uint64_t bucket_id;
|
2011-08-18 00:11:55 +00:00
|
|
|
|
2012-02-25 01:00:35 +00:00
|
|
|
rgw_bucket() { bucket_id = 0; }
|
2011-08-18 00:11:55 +00:00
|
|
|
rgw_bucket(const char *n) : name(n) {
|
|
|
|
assert(*n == '.'); // only rgw private buckets should be initialized without pool
|
|
|
|
pool = n;
|
2011-09-19 23:41:44 +00:00
|
|
|
marker = "";
|
2011-09-28 17:50:48 +00:00
|
|
|
bucket_id = 0;
|
2011-08-18 00:11:55 +00:00
|
|
|
}
|
2011-09-28 17:50:48 +00:00
|
|
|
rgw_bucket(const char *n, const char *p, const char *m, uint64_t id) :
|
|
|
|
name(n), pool(p), marker(m), bucket_id(id) {}
|
2011-08-18 00:11:55 +00:00
|
|
|
|
|
|
|
void clear() {
|
|
|
|
name = "";
|
|
|
|
pool = "";
|
2011-09-19 23:41:44 +00:00
|
|
|
marker = "";
|
2011-09-28 17:50:48 +00:00
|
|
|
bucket_id = 0;
|
2011-08-18 00:11:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void encode(bufferlist& bl) const {
|
2011-09-19 23:41:44 +00:00
|
|
|
__u8 struct_v = 2;
|
2011-08-18 00:11:55 +00:00
|
|
|
::encode(struct_v, bl);
|
|
|
|
::encode(name, bl);
|
|
|
|
::encode(pool, bl);
|
2011-09-19 23:41:44 +00:00
|
|
|
::encode(marker, bl);
|
2011-09-28 17:50:48 +00:00
|
|
|
::encode(bucket_id, bl);
|
2011-08-18 00:11:55 +00:00
|
|
|
}
|
|
|
|
void decode(bufferlist::iterator& bl) {
|
|
|
|
__u8 struct_v;
|
|
|
|
::decode(struct_v, bl);
|
|
|
|
::decode(name, bl);
|
|
|
|
::decode(pool, bl);
|
2011-09-28 17:50:48 +00:00
|
|
|
if (struct_v >= 2) {
|
2011-09-19 23:41:44 +00:00
|
|
|
::decode(marker, bl);
|
2011-09-28 17:50:48 +00:00
|
|
|
::decode(bucket_id, bl);
|
|
|
|
}
|
2011-08-18 00:11:55 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
WRITE_CLASS_ENCODER(rgw_bucket)
|
|
|
|
|
|
|
|
inline ostream& operator<<(ostream& out, const rgw_bucket b) {
|
|
|
|
out << b.name;
|
|
|
|
if (b.name.compare(b.pool))
|
2011-09-19 23:41:44 +00:00
|
|
|
out << "(@" << b.pool << "[" << b.marker << "])";
|
2011-08-18 00:11:55 +00:00
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern rgw_bucket rgw_root_bucket;
|
|
|
|
|
2011-10-26 21:30:26 +00:00
|
|
|
enum RGWBucketFlags {
|
|
|
|
BUCKET_SUSPENDED = 0x1,
|
|
|
|
};
|
|
|
|
|
2011-08-18 20:21:31 +00:00
|
|
|
struct RGWBucketInfo
|
|
|
|
{
|
|
|
|
rgw_bucket bucket;
|
2011-09-28 20:51:20 +00:00
|
|
|
string owner;
|
2011-10-26 21:30:26 +00:00
|
|
|
uint32_t flags;
|
2011-08-18 20:21:31 +00:00
|
|
|
|
|
|
|
void encode(bufferlist& bl) const {
|
2011-10-26 21:30:26 +00:00
|
|
|
__u32 ver = 3;
|
2011-08-18 20:21:31 +00:00
|
|
|
::encode(ver, bl);
|
|
|
|
::encode(bucket, bl);
|
2011-09-29 23:32:14 +00:00
|
|
|
::encode(owner, bl);
|
2011-10-26 21:30:26 +00:00
|
|
|
::encode(flags, bl);
|
2011-08-18 20:21:31 +00:00
|
|
|
}
|
|
|
|
void decode(bufferlist::iterator& bl) {
|
|
|
|
__u32 ver;
|
|
|
|
::decode(ver, bl);
|
|
|
|
::decode(bucket, bl);
|
2011-10-26 21:30:26 +00:00
|
|
|
if (ver >= 2)
|
2011-09-29 23:32:14 +00:00
|
|
|
::decode(owner, bl);
|
2011-10-26 21:30:26 +00:00
|
|
|
if (ver >= 3)
|
|
|
|
::decode(flags, bl);
|
2011-08-18 20:21:31 +00:00
|
|
|
}
|
2011-10-26 21:30:26 +00:00
|
|
|
|
|
|
|
RGWBucketInfo() : flags(0) {}
|
2011-08-18 20:21:31 +00:00
|
|
|
};
|
|
|
|
WRITE_CLASS_ENCODER(RGWBucketInfo)
|
|
|
|
|
2011-08-24 23:41:10 +00:00
|
|
|
struct RGWBucketStats
|
|
|
|
{
|
2011-09-24 00:11:49 +00:00
|
|
|
RGWObjCategory category;
|
2011-08-24 23:41:10 +00:00
|
|
|
uint64_t num_kb;
|
2011-11-08 21:42:55 +00:00
|
|
|
uint64_t num_kb_rounded;
|
2011-08-24 23:41:10 +00:00
|
|
|
uint64_t num_objects;
|
|
|
|
};
|
|
|
|
|
2011-02-16 01:30:07 +00:00
|
|
|
struct req_state;
|
|
|
|
|
2011-07-19 21:02:11 +00:00
|
|
|
struct RGWEnv;
|
|
|
|
|
2010-03-11 22:49:27 +00:00
|
|
|
/** Store all the state necessary to complete and respond to an HTTP request*/
|
2009-07-18 00:49:59 +00:00
|
|
|
struct req_state {
|
2011-07-15 19:54:32 +00:00
|
|
|
FCGX_Request *fcgx;
|
2009-07-18 00:49:59 +00:00
|
|
|
http_op op;
|
|
|
|
bool content_started;
|
2011-02-16 01:30:07 +00:00
|
|
|
int format;
|
2011-08-04 21:43:55 +00:00
|
|
|
ceph::Formatter *formatter;
|
2012-02-12 06:43:35 +00:00
|
|
|
string decoded_uri;
|
2011-10-28 21:03:38 +00:00
|
|
|
string request_uri;
|
2012-02-12 06:43:35 +00:00
|
|
|
string request_params;
|
2009-07-18 00:49:59 +00:00
|
|
|
const char *host;
|
|
|
|
const char *method;
|
|
|
|
const char *length;
|
2011-07-26 21:27:36 +00:00
|
|
|
uint64_t content_length;
|
2009-07-18 00:49:59 +00:00
|
|
|
const char *content_type;
|
2009-08-07 17:46:29 +00:00
|
|
|
struct rgw_err err;
|
2010-09-27 16:37:07 +00:00
|
|
|
bool expect_cont;
|
2011-03-09 00:49:49 +00:00
|
|
|
bool header_ended;
|
|
|
|
uint64_t bytes_sent; // bytes sent as a response, excluding header
|
2011-04-30 00:17:23 +00:00
|
|
|
uint64_t bytes_received; // data received
|
2011-04-29 23:14:22 +00:00
|
|
|
uint64_t obj_size;
|
2011-03-10 18:01:24 +00:00
|
|
|
bool should_log;
|
2011-05-23 22:12:48 +00:00
|
|
|
uint32_t perm_mask;
|
2011-07-28 21:03:20 +00:00
|
|
|
utime_t header_time;
|
2009-07-18 00:49:59 +00:00
|
|
|
|
|
|
|
XMLArgs args;
|
|
|
|
|
2011-08-18 00:11:55 +00:00
|
|
|
const char *bucket_name;
|
2009-07-18 00:49:59 +00:00
|
|
|
const char *object;
|
|
|
|
|
|
|
|
const char *host_bucket;
|
|
|
|
|
2011-08-18 00:11:55 +00:00
|
|
|
rgw_bucket bucket;
|
|
|
|
string bucket_name_str;
|
2009-07-18 00:49:59 +00:00
|
|
|
string object_str;
|
2011-09-30 16:10:56 +00:00
|
|
|
string bucket_owner;
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-09-02 22:58:49 +00:00
|
|
|
map<string, string> x_meta_map;
|
2011-10-27 21:31:16 +00:00
|
|
|
bool has_bad_meta;
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2009-08-07 17:46:29 +00:00
|
|
|
RGWUserInfo user;
|
2012-02-21 22:39:20 +00:00
|
|
|
RGWAccessControlPolicy *bucket_acl;
|
|
|
|
RGWAccessControlPolicy *object_acl;
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2009-07-20 23:41:25 +00:00
|
|
|
string canned_acl;
|
|
|
|
const char *copy_source;
|
|
|
|
const char *http_auth;
|
|
|
|
|
2011-02-12 01:18:24 +00:00
|
|
|
int prot_flags;
|
|
|
|
|
2011-07-11 20:41:40 +00:00
|
|
|
const char *os_auth_token;
|
2011-02-14 21:23:27 +00:00
|
|
|
char *os_user;
|
|
|
|
char *os_groups;
|
2011-02-12 01:18:24 +00:00
|
|
|
|
2011-03-09 00:49:49 +00:00
|
|
|
utime_t time;
|
|
|
|
|
2011-07-19 21:02:11 +00:00
|
|
|
struct RGWEnv *env;
|
|
|
|
|
2011-08-02 23:29:16 +00:00
|
|
|
void *obj_ctx;
|
|
|
|
|
2012-01-17 01:03:19 +00:00
|
|
|
string dialect;
|
|
|
|
|
2011-07-19 21:02:11 +00:00
|
|
|
req_state(struct RGWEnv *e);
|
2011-07-14 22:29:36 +00:00
|
|
|
~req_state();
|
2009-07-18 00:49:59 +00:00
|
|
|
};
|
|
|
|
|
2011-08-02 23:04:14 +00:00
|
|
|
extern void flush_formatter_to_req_state(struct req_state *s,
|
2011-08-04 21:43:55 +00:00
|
|
|
ceph::Formatter *formatter);
|
2011-08-02 23:04:14 +00:00
|
|
|
|
2010-03-11 22:49:27 +00:00
|
|
|
/** Store basic data on an object */
|
2009-08-07 17:46:29 +00:00
|
|
|
struct RGWObjEnt {
|
2009-07-18 00:49:59 +00:00
|
|
|
std::string name;
|
2011-06-10 22:01:04 +00:00
|
|
|
std::string owner;
|
|
|
|
std::string owner_display_name;
|
2011-10-24 22:22:35 +00:00
|
|
|
uint64_t size;
|
2009-07-18 00:49:59 +00:00
|
|
|
time_t mtime;
|
2011-09-22 00:14:48 +00:00
|
|
|
string etag;
|
2011-03-23 23:50:21 +00:00
|
|
|
string content_type;
|
2011-06-16 20:26:19 +00:00
|
|
|
|
|
|
|
void clear() { // not clearing etag
|
|
|
|
name="";
|
|
|
|
size = 0;
|
|
|
|
mtime = 0;
|
|
|
|
content_type="";
|
|
|
|
}
|
2009-07-18 00:49:59 +00:00
|
|
|
};
|
|
|
|
|
2011-10-26 21:30:26 +00:00
|
|
|
/** Store basic data on bucket */
|
2011-03-03 22:14:19 +00:00
|
|
|
struct RGWBucketEnt {
|
2011-08-18 00:11:55 +00:00
|
|
|
rgw_bucket bucket;
|
2011-03-03 22:14:19 +00:00
|
|
|
size_t size;
|
2011-11-08 21:42:55 +00:00
|
|
|
size_t size_rounded;
|
2011-03-03 22:14:19 +00:00
|
|
|
time_t mtime;
|
|
|
|
uint64_t count;
|
|
|
|
|
2011-12-30 22:18:40 +00:00
|
|
|
RGWBucketEnt() : size(0), size_rounded(0), mtime(0), count(0) {}
|
|
|
|
|
2011-03-03 22:14:19 +00:00
|
|
|
void encode(bufferlist& bl) const {
|
2011-11-08 21:42:55 +00:00
|
|
|
__u8 struct_v = 4;
|
2011-03-03 22:14:19 +00:00
|
|
|
::encode(struct_v, bl);
|
|
|
|
uint64_t s = size;
|
|
|
|
__u32 mt = mtime;
|
2012-02-13 20:07:17 +00:00
|
|
|
string empty_str; // originally had the bucket name here, but we encode bucket later
|
|
|
|
::encode(empty_str, bl);
|
2011-03-03 22:14:19 +00:00
|
|
|
::encode(s, bl);
|
|
|
|
::encode(mt, bl);
|
|
|
|
::encode(count, bl);
|
2011-08-18 00:11:55 +00:00
|
|
|
::encode(bucket, bl);
|
2011-11-08 21:42:55 +00:00
|
|
|
s = size_rounded;
|
|
|
|
::encode(s, bl);
|
2011-03-03 22:14:19 +00:00
|
|
|
}
|
|
|
|
void decode(bufferlist::iterator& bl) {
|
|
|
|
__u8 struct_v;
|
|
|
|
::decode(struct_v, bl);
|
|
|
|
__u32 mt;
|
|
|
|
uint64_t s;
|
2012-02-13 20:07:17 +00:00
|
|
|
string empty_str; // backward compatibility
|
|
|
|
::decode(empty_str, bl);
|
2011-03-03 22:14:19 +00:00
|
|
|
::decode(s, bl);
|
|
|
|
::decode(mt, bl);
|
|
|
|
size = s;
|
|
|
|
mtime = mt;
|
|
|
|
if (struct_v >= 2)
|
|
|
|
::decode(count, bl);
|
2011-08-18 00:11:55 +00:00
|
|
|
if (struct_v >= 3)
|
|
|
|
::decode(bucket, bl);
|
2011-11-08 21:42:55 +00:00
|
|
|
if (struct_v >= 4)
|
|
|
|
::decode(s, bl);
|
|
|
|
size_rounded = s;
|
2011-03-03 22:14:19 +00:00
|
|
|
}
|
2011-03-04 21:57:03 +00:00
|
|
|
void clear() {
|
2011-08-18 00:11:55 +00:00
|
|
|
bucket.clear();
|
2011-03-04 21:57:03 +00:00
|
|
|
size = 0;
|
2011-11-08 21:42:55 +00:00
|
|
|
size_rounded = 0;
|
2011-03-04 21:57:03 +00:00
|
|
|
mtime = 0;
|
|
|
|
count = 0;
|
|
|
|
}
|
2011-03-03 22:14:19 +00:00
|
|
|
};
|
|
|
|
WRITE_CLASS_ENCODER(RGWBucketEnt)
|
|
|
|
|
2011-06-02 17:28:24 +00:00
|
|
|
struct RGWUploadPartInfo {
|
|
|
|
uint32_t num;
|
|
|
|
uint64_t size;
|
|
|
|
string etag;
|
2011-06-02 20:45:20 +00:00
|
|
|
utime_t modified;
|
2011-06-02 17:28:24 +00:00
|
|
|
|
|
|
|
void encode(bufferlist& bl) const {
|
|
|
|
__u8 struct_v = 1;
|
|
|
|
::encode(struct_v, bl);
|
|
|
|
::encode(num, bl);
|
|
|
|
::encode(size, bl);
|
|
|
|
::encode(etag, bl);
|
2011-06-02 20:45:20 +00:00
|
|
|
::encode(modified, bl);
|
2011-06-02 17:28:24 +00:00
|
|
|
}
|
|
|
|
void decode(bufferlist::iterator& bl) {
|
|
|
|
__u8 struct_v;
|
|
|
|
::decode(struct_v, bl);
|
|
|
|
::decode(num, bl);
|
|
|
|
::decode(size, bl);
|
|
|
|
::decode(etag, bl);
|
2011-06-02 20:45:20 +00:00
|
|
|
::decode(modified, bl);
|
2011-06-02 17:28:24 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
WRITE_CLASS_ENCODER(RGWUploadPartInfo)
|
|
|
|
|
2011-06-08 19:32:50 +00:00
|
|
|
class rgw_obj {
|
2011-06-09 20:25:46 +00:00
|
|
|
std::string orig_obj;
|
|
|
|
std::string orig_key;
|
2011-06-08 19:32:50 +00:00
|
|
|
public:
|
2011-08-18 00:11:55 +00:00
|
|
|
rgw_bucket bucket;
|
2011-06-08 20:10:01 +00:00
|
|
|
std::string key;
|
2011-06-09 20:25:46 +00:00
|
|
|
std::string ns;
|
|
|
|
std::string object;
|
2011-06-08 19:32:50 +00:00
|
|
|
|
|
|
|
rgw_obj() {}
|
2011-08-18 00:11:55 +00:00
|
|
|
rgw_obj(rgw_bucket& b, std::string& o) {
|
2011-06-08 20:10:01 +00:00
|
|
|
init(b, o);
|
|
|
|
}
|
2011-08-18 00:11:55 +00:00
|
|
|
rgw_obj(rgw_bucket& b, std::string& o, std::string& k) {
|
2011-06-08 20:10:01 +00:00
|
|
|
init(b, o, k);
|
|
|
|
}
|
2011-08-18 00:11:55 +00:00
|
|
|
rgw_obj(rgw_bucket& b, std::string& o, std::string& k, std::string& n) {
|
2011-06-09 20:25:46 +00:00
|
|
|
init(b, o, k, n);
|
|
|
|
}
|
2011-08-18 00:11:55 +00:00
|
|
|
void init(rgw_bucket& b, std::string& o, std::string& k, std::string& n) {
|
2011-06-09 20:25:46 +00:00
|
|
|
bucket = b;
|
|
|
|
set_ns(n);
|
|
|
|
set_obj(o);
|
|
|
|
set_key(k);
|
|
|
|
}
|
2011-08-18 00:11:55 +00:00
|
|
|
void init(rgw_bucket& b, std::string& o, std::string& k) {
|
2011-06-08 20:10:01 +00:00
|
|
|
bucket = b;
|
2011-06-09 20:25:46 +00:00
|
|
|
set_obj(o);
|
|
|
|
set_key(k);
|
2011-06-08 20:10:01 +00:00
|
|
|
}
|
2011-08-18 00:11:55 +00:00
|
|
|
void init(rgw_bucket& b, std::string& o) {
|
2011-06-08 19:32:50 +00:00
|
|
|
bucket = b;
|
2011-06-09 20:25:46 +00:00
|
|
|
set_obj(o);
|
2011-09-19 23:41:44 +00:00
|
|
|
orig_key = key = o;
|
2011-06-09 20:25:46 +00:00
|
|
|
}
|
|
|
|
int set_ns(const char *n) {
|
|
|
|
if (!n)
|
|
|
|
return -EINVAL;
|
|
|
|
string ns_str(n);
|
|
|
|
return set_ns(ns_str);
|
|
|
|
}
|
|
|
|
int set_ns(string& n) {
|
|
|
|
if (n[0] == '_')
|
|
|
|
return -EINVAL;
|
|
|
|
ns = n;
|
|
|
|
set_obj(orig_obj);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_key(string& k) {
|
|
|
|
orig_key = k;
|
2011-09-19 23:41:44 +00:00
|
|
|
key = k;
|
2011-06-09 20:25:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void set_obj(string& o) {
|
|
|
|
orig_obj = o;
|
|
|
|
if (ns.empty()) {
|
|
|
|
if (o.empty())
|
|
|
|
return;
|
|
|
|
if (o.size() < 1 || o[0] != '_') {
|
|
|
|
object = o;
|
|
|
|
return;
|
|
|
|
}
|
2012-02-06 21:12:58 +00:00
|
|
|
object = "_";
|
2011-06-09 20:25:46 +00:00
|
|
|
object.append(o);
|
|
|
|
} else {
|
|
|
|
object = "_";
|
|
|
|
object.append(ns);
|
|
|
|
object.append("_");
|
|
|
|
object.append(o);
|
|
|
|
}
|
2011-09-19 23:41:44 +00:00
|
|
|
if (orig_key.size())
|
|
|
|
set_key(orig_key);
|
|
|
|
else
|
|
|
|
set_key(orig_obj);
|
2011-06-09 20:25:46 +00:00
|
|
|
}
|
|
|
|
|
2011-08-18 23:33:18 +00:00
|
|
|
string loc() {
|
|
|
|
if (orig_key.empty())
|
|
|
|
return orig_obj;
|
|
|
|
else
|
|
|
|
return orig_key;
|
|
|
|
}
|
|
|
|
|
2011-10-20 20:49:45 +00:00
|
|
|
/**
|
|
|
|
* Translate a namespace-mangled object name to the user-facing name
|
|
|
|
* existing in the given namespace.
|
|
|
|
*
|
|
|
|
* If the object is part of the given namespace, it returns true
|
|
|
|
* and cuts down the name to the unmangled version. If it is not
|
|
|
|
* part of the given namespace, it returns false.
|
|
|
|
*/
|
|
|
|
static bool translate_raw_obj_to_obj_in_ns(string& obj, string& ns) {
|
2011-06-09 20:25:46 +00:00
|
|
|
if (ns.empty()) {
|
|
|
|
if (obj[0] != '_')
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (obj.size() >= 2 && obj[1] == '_') {
|
|
|
|
obj = obj.substr(1);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj[0] != '_' || obj.size() < 3) // for namespace, min size would be 3: _x_
|
|
|
|
return false;
|
|
|
|
|
|
|
|
int pos = obj.find('_', 1);
|
|
|
|
if (pos <= 1) // if it starts with __, it's not in our namespace
|
|
|
|
return false;
|
|
|
|
|
|
|
|
string obj_ns = obj.substr(1, pos - 1);
|
|
|
|
if (obj_ns.compare(ns) != 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
obj = obj.substr(pos + 1);
|
|
|
|
return true;
|
2011-06-08 19:32:50 +00:00
|
|
|
}
|
2011-07-01 00:13:42 +00:00
|
|
|
|
2011-10-20 23:26:15 +00:00
|
|
|
/**
|
|
|
|
* Given a mangled object name and an empty namespace string, this
|
|
|
|
* function extracts the namespace into the string and sets the object
|
|
|
|
* name to be the unmangled version.
|
|
|
|
*
|
|
|
|
* It returns true after successfully doing so, or
|
|
|
|
* false if it fails.
|
|
|
|
*/
|
|
|
|
static bool strip_namespace_from_object(string& obj, string& ns) {
|
|
|
|
ns.clear();
|
|
|
|
if (obj[0] != '_') {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t pos = obj.find('_', 1);
|
|
|
|
if (pos == string::npos) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t period_pos = obj.find('.');
|
|
|
|
if (period_pos < pos) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ns = obj.substr(1, pos-1);
|
|
|
|
obj = obj.substr(pos+1, string::npos);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-07-01 00:13:42 +00:00
|
|
|
void encode(bufferlist& bl) const {
|
2011-08-18 00:11:55 +00:00
|
|
|
__u8 struct_v = 2;
|
2011-07-01 00:13:42 +00:00
|
|
|
::encode(struct_v, bl);
|
2011-08-18 00:11:55 +00:00
|
|
|
::encode(bucket.name, bl);
|
2011-07-01 00:13:42 +00:00
|
|
|
::encode(key, bl);
|
|
|
|
::encode(ns, bl);
|
|
|
|
::encode(object, bl);
|
2011-08-18 00:11:55 +00:00
|
|
|
::encode(bucket, bl);
|
2011-07-01 00:13:42 +00:00
|
|
|
}
|
|
|
|
void decode(bufferlist::iterator& bl) {
|
|
|
|
__u8 struct_v;
|
|
|
|
::decode(struct_v, bl);
|
2011-08-18 00:11:55 +00:00
|
|
|
::decode(bucket.name, bl);
|
2011-07-01 00:13:42 +00:00
|
|
|
::decode(key, bl);
|
|
|
|
::decode(ns, bl);
|
|
|
|
::decode(object, bl);
|
2011-08-18 00:11:55 +00:00
|
|
|
if (struct_v >= 2)
|
|
|
|
::decode(bucket, bl);
|
2011-07-01 00:13:42 +00:00
|
|
|
}
|
2011-08-02 23:29:16 +00:00
|
|
|
|
2012-03-01 22:03:57 +00:00
|
|
|
bool operator==(const rgw_obj& o) const {
|
|
|
|
return (object.compare(o.object) == 0) &&
|
|
|
|
(bucket.name.compare(o.bucket.name) == 0) &&
|
|
|
|
(ns.compare(o.ns) == 0);
|
|
|
|
}
|
2011-08-02 23:29:16 +00:00
|
|
|
bool operator<(const rgw_obj& o) const {
|
2011-10-24 23:43:14 +00:00
|
|
|
int r = bucket.name.compare(o.bucket.name);
|
|
|
|
if (r == 0) {
|
|
|
|
r = object.compare(o.object);
|
|
|
|
if (r == 0) {
|
|
|
|
r = ns.compare(o.ns);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (r < 0);
|
2011-08-02 23:29:16 +00:00
|
|
|
}
|
2011-06-08 19:32:50 +00:00
|
|
|
};
|
2011-07-01 00:13:42 +00:00
|
|
|
WRITE_CLASS_ENCODER(rgw_obj)
|
2011-06-08 19:32:50 +00:00
|
|
|
|
2011-06-24 19:55:25 +00:00
|
|
|
inline ostream& operator<<(ostream& out, const rgw_obj o) {
|
2011-08-18 00:11:55 +00:00
|
|
|
return out << o.bucket.name << ":" << o.object;
|
2011-06-24 19:55:25 +00:00
|
|
|
}
|
|
|
|
|
2009-07-20 23:41:25 +00:00
|
|
|
static inline void buf_to_hex(const unsigned char *buf, int len, char *str)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
str[0] = '\0';
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
sprintf(&str[i*2], "%02x", (int)buf[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-07 21:13:59 +00:00
|
|
|
static inline int hexdigit(char c)
|
|
|
|
{
|
|
|
|
if (c >= '0' && c <= '9')
|
|
|
|
return (c - '0');
|
|
|
|
c = toupper(c);
|
|
|
|
if (c >= 'A' && c <= 'F')
|
|
|
|
return c - 'A' + 0xa;
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int hex_to_buf(const char *hex, char *buf, int len)
|
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
const char *p = hex;
|
|
|
|
while (*p) {
|
|
|
|
if (i >= len)
|
|
|
|
return -EINVAL;
|
|
|
|
buf[i] = 0;
|
|
|
|
int d = hexdigit(*p);
|
|
|
|
if (d < 0)
|
|
|
|
return d;
|
|
|
|
buf[i] = d << 4;
|
|
|
|
p++;
|
|
|
|
if (!*p)
|
|
|
|
return -EINVAL;
|
|
|
|
d = hexdigit(*p);
|
|
|
|
if (d < 0)
|
|
|
|
return -d;
|
|
|
|
buf[i] += d;
|
|
|
|
i++;
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2011-03-10 18:01:24 +00:00
|
|
|
static inline int rgw_str_to_bool(const char *s, int def_val)
|
|
|
|
{
|
|
|
|
if (!s)
|
|
|
|
return def_val;
|
|
|
|
|
|
|
|
return (strcasecmp(s, "on") == 0 ||
|
|
|
|
strcasecmp(s, "yes") == 0 ||
|
2011-03-24 21:24:11 +00:00
|
|
|
strcasecmp(s, "1") == 0);
|
2011-03-10 18:01:24 +00:00
|
|
|
}
|
|
|
|
|
2011-06-24 21:50:01 +00:00
|
|
|
static inline void append_rand_alpha(string& src, string& dest, int len)
|
|
|
|
{
|
|
|
|
dest = src;
|
|
|
|
char buf[len + 1];
|
|
|
|
gen_rand_alphanumeric(buf, len);
|
|
|
|
dest.append("_");
|
|
|
|
dest.append(buf);
|
|
|
|
}
|
|
|
|
|
2011-09-24 00:11:49 +00:00
|
|
|
static inline const char *rgw_obj_category_name(RGWObjCategory category)
|
|
|
|
{
|
|
|
|
switch (category) {
|
|
|
|
case RGW_OBJ_CATEGORY_NONE:
|
|
|
|
return "rgw.none";
|
|
|
|
case RGW_OBJ_CATEGORY_MAIN:
|
|
|
|
return "rgw.main";
|
|
|
|
case RGW_OBJ_CATEGORY_SHADOW:
|
|
|
|
return "rgw.shadow";
|
|
|
|
case RGW_OBJ_CATEGORY_MULTIMETA:
|
|
|
|
return "rgw.multimeta";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "unknown";
|
|
|
|
}
|
|
|
|
|
2011-10-14 20:20:20 +00:00
|
|
|
/** time parsing */
|
2009-07-18 00:49:59 +00:00
|
|
|
extern int parse_time(const char *time_str, time_t *time);
|
2011-10-14 20:20:20 +00:00
|
|
|
extern bool parse_rfc2616(const char *s, struct tm *t);
|
|
|
|
|
2010-03-11 22:49:27 +00:00
|
|
|
/** Check if the req_state's user has the necessary permissions
|
|
|
|
* to do the requested action */
|
2012-02-21 22:39:20 +00:00
|
|
|
extern bool verify_bucket_permission(struct req_state *s, int perm);
|
|
|
|
extern bool verify_object_permission(struct req_state *s, int perm);
|
2010-03-11 22:49:27 +00:00
|
|
|
/** Convert an input URL into a sane object name
|
|
|
|
* by converting %-escaped strings into characters, etc*/
|
2009-07-20 23:41:25 +00:00
|
|
|
extern bool url_decode(string& src_str, string& dest_str);
|
2009-07-18 00:49:59 +00:00
|
|
|
|
2011-03-24 05:42:02 +00:00
|
|
|
extern void calc_hmac_sha1(const char *key, int key_len,
|
|
|
|
const char *msg, int msg_len, char *dest);
|
|
|
|
/* destination should be CEPH_CRYPTO_HMACSHA1_DIGESTSIZE bytes long */
|
|
|
|
|
2009-07-18 00:49:59 +00:00
|
|
|
#endif
|