diff options
author | Sage Weil <sage@inktank.com> | 2013-01-23 21:31:11 -0800 |
---|---|---|
committer | Dan Mick <dan.mick@inktank.com> | 2013-03-05 15:12:02 -0800 |
commit | e89884da9f76b713372a79b772ba3cc2f3b03048 (patch) | |
tree | b49eb6e05e280ae91809aa3e85c630343074de39 | |
parent | ee943c8bcf36f1e2218d8e25edfa38ec5fe4bec2 (diff) | |
download | ceph-e89884da9f76b713372a79b772ba3cc2f3b03048.tar.gz |
rados: obey op_size for 'get'
Otherwise we try to read the whole object in one go, which doesn't bode
well for large objects (either non-optimal or simply broken).
Signed-off-by: Sage Weil <sage@inktank.com>
Reviewed-by: Dan Mick <dan.mick@inktank.com>
(cherry picked from commit 234becd3447a679a919af458440bc31c8bd6b84f)
-rw-r--r-- | src/rados.cc | 43 |
1 files changed, 31 insertions, 12 deletions
diff --git a/src/rados.cc b/src/rados.cc index 4c3a93734e2..4bbfaed8650 100644 --- a/src/rados.cc +++ b/src/rados.cc @@ -169,22 +169,41 @@ static void usage_exit() exit(1); } -static int do_get(IoCtx& io_ctx, const char *objname, const char *outfile, bool check_stdio) +static int do_get(IoCtx& io_ctx, const char *objname, const char *outfile, unsigned op_size) { string oid(objname); - bufferlist outdata; - int ret = io_ctx.read(oid, outdata, 0, 0); - if (ret < 0) { - return ret; - } + int fd; if (check_stdio && strcmp(outfile, "-") == 0) { - fwrite(outdata.c_str(), outdata.length(), 1, stdout); + fd = 1; } else { - outdata.write_file(outfile); - generic_dout(0) << "wrote " << outdata.length() << " byte payload to " << outfile << dendl; + fd = TEMP_FAILURE_RETRY(::open(outfile, O_WRONLY|O_CREAT|O_TRUNC, 0644)); + if (fd < 0) { + int err = errno; + cerr << "failed to open file: " << cpp_strerror(err) << std::endl; + return -err; + } + } + + uint64_t offset = 0; + while (true) { + bufferlist outdata; + int ret = io_ctx.read(oid, outdata, op_size, offset); + if (ret <= 0) { + return ret; + } + ret = outdata.write_fd(fd); + if (ret < 0) { + cerr << "error writing to file: " << cpp_strerror(ret) << std::endl; + return ret; + } + if (outdata.length() < op_size) + break; + offset += outdata.length(); } + if (!check_stdio) + TEMP_FAILURE_RETRY(::close(fd)); return 0; } @@ -315,7 +334,7 @@ static int do_copy_pool(Rados& rados, const char *src_pool, const char *target_p return 0; } -static int do_put(IoCtx& io_ctx, const char *objname, const char *infile, int op_size, bool check_stdio) +static int do_put(IoCtx& io_ctx, const char *objname, const char *infile, int op_size) { string oid(objname); bufferlist indata; @@ -1384,7 +1403,7 @@ static int rados_tool_common(const std::map < std::string, std::string > &opts, else if (strcmp(nargs[0], "get") == 0) { if (!pool_name || nargs.size() < 3) usage_exit(); - ret = do_get(io_ctx, nargs[1], nargs[2], true); + ret = do_get(io_ctx, nargs[1], nargs[2], op_size); if (ret < 0) { cerr << "error getting " << pool_name << "/" << nargs[1] << ": " << strerror_r(-ret, buf, sizeof(buf)) << std::endl; return 1; @@ -1393,7 +1412,7 @@ static int rados_tool_common(const std::map < std::string, std::string > &opts, else if (strcmp(nargs[0], "put") == 0) { if (!pool_name || nargs.size() < 3) usage_exit(); - ret = do_put(io_ctx, nargs[1], nargs[2], op_size, true); + ret = do_put(io_ctx, nargs[1], nargs[2], op_size); if (ret < 0) { cerr << "error putting " << pool_name << "/" << nargs[1] << ": " << strerror_r(-ret, buf, sizeof(buf)) << std::endl; return 1; |