diff options
author | Carlo Nucera <carlo.nucera@protonmail.com> | 2020-08-06 15:52:33 -0400 |
---|---|---|
committer | Carlo Nucera <carlo.nucera@protonmail.com> | 2020-08-06 15:53:09 -0400 |
commit | f795f0fabc785cd2eb2ccfec71564c8af819ecf4 (patch) | |
tree | 76b0719449f036ea9312a9fb96e5ea241b455d9d /src/libstore | |
parent | 7302761f6451dafd8834f0db9fdd7dcc27f49eff (diff) | |
parent | 8b175f58d1a25dd904fb0ffdf5b2b9501983dba0 (diff) |
Merge branch 'drv-outputs-map-allow-missing-namespace' of github.com:obsidiansystems/nix into templated-daemon-protocol
Diffstat (limited to 'src/libstore')
-rw-r--r-- | src/libstore/build.cc | 4 | ||||
-rw-r--r-- | src/libstore/daemon.cc | 36 | ||||
-rw-r--r-- | src/libstore/derivations.cc | 4 | ||||
-rw-r--r-- | src/libstore/export-import.cc | 4 | ||||
-rw-r--r-- | src/libstore/legacy-ssh-store.cc | 14 | ||||
-rw-r--r-- | src/libstore/remote-store.cc | 55 | ||||
-rw-r--r-- | src/libstore/worker-protocol.hh | 64 |
7 files changed, 102 insertions, 79 deletions
diff --git a/src/libstore/build.cc b/src/libstore/build.cc index 3fd60c5ab..8340828e7 100644 --- a/src/libstore/build.cc +++ b/src/libstore/build.cc @@ -1864,11 +1864,11 @@ HookReply DerivationGoal::tryBuildHook() /* Tell the hook all the inputs that have to be copied to the remote system. */ - write(worker.store, hook->sink, inputPaths); + nix::worker_proto::write(worker.store, hook->sink, inputPaths); /* Tell the hooks the missing outputs that have to be copied back from the remote system. */ - write(worker.store, hook->sink, missingPaths); + nix::worker_proto::write(worker.store, hook->sink, missingPaths); hook->sink = FdSink(); hook->toHook.writeSide = -1; diff --git a/src/libstore/daemon.cc b/src/libstore/daemon.cc index b90372a71..6d734abdc 100644 --- a/src/libstore/daemon.cc +++ b/src/libstore/daemon.cc @@ -256,11 +256,11 @@ static void performOp(TunnelLogger * logger, ref<Store> store, } case wopQueryValidPaths: { - auto paths = read(*store, from, Proxy<StorePathSet> {}); + auto paths = nix::worker_proto::read(*store, from, Phantom<StorePathSet> {}); logger->startWork(); auto res = store->queryValidPaths(paths); logger->stopWork(); - write(*store, to, res); + nix::worker_proto::write(*store, to, res); break; } @@ -276,11 +276,11 @@ static void performOp(TunnelLogger * logger, ref<Store> store, } case wopQuerySubstitutablePaths: { - auto paths = read(*store, from, Proxy<StorePathSet> {}); + auto paths = nix::worker_proto::read(*store, from, Phantom<StorePathSet> {}); logger->startWork(); auto res = store->querySubstitutablePaths(paths); logger->stopWork(); - write(*store, to, res); + nix::worker_proto::write(*store, to, res); break; } @@ -309,7 +309,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, paths = store->queryValidDerivers(path); else paths = store->queryDerivationOutputs(path); logger->stopWork(); - write(*store, to, paths); + nix::worker_proto::write(*store, to, paths); break; } @@ -327,7 +327,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, logger->startWork(); auto outputs = store->queryDerivationOutputMap(path); logger->stopWork(); - write(*store, to, outputs); + nix::worker_proto::write(*store, to, outputs); break; } @@ -397,7 +397,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, case wopAddTextToStore: { string suffix = readString(from); string s = readString(from); - auto refs = read(*store, from, Proxy<StorePathSet> {}); + auto refs = nix::worker_proto::read(*store, from, Phantom<StorePathSet> {}); logger->startWork(); auto path = store->addTextToStore(suffix, s, refs, NoRepair); logger->stopWork(); @@ -518,7 +518,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, case wopCollectGarbage: { GCOptions options; options.action = (GCOptions::GCAction) readInt(from); - options.pathsToDelete = read(*store, from, Proxy<StorePathSet> {}); + options.pathsToDelete = nix::worker_proto::read(*store, from, Phantom<StorePathSet> {}); from >> options.ignoreLiveness >> options.maxFreed; // obsolete fields readInt(from); @@ -587,7 +587,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, else { to << 1 << (i->second.deriver ? store->printStorePath(*i->second.deriver) : ""); - write(*store, to, i->second.references); + nix::worker_proto::write(*store, to, i->second.references); to << i->second.downloadSize << i->second.narSize; } @@ -598,11 +598,11 @@ static void performOp(TunnelLogger * logger, ref<Store> store, SubstitutablePathInfos infos; StorePathCAMap pathsMap = {}; if (GET_PROTOCOL_MINOR(clientVersion) < 22) { - auto paths = read(*store, from, Proxy<StorePathSet> {}); + auto paths = nix::worker_proto::read(*store, from, Phantom<StorePathSet> {}); for (auto & path : paths) pathsMap.emplace(path, std::nullopt); } else - pathsMap = read(*store, from, Proxy<StorePathCAMap> {}); + pathsMap = nix::worker_proto::read(*store, from, Phantom<StorePathCAMap> {}); logger->startWork(); store->querySubstitutablePathInfos(pathsMap, infos); logger->stopWork(); @@ -610,7 +610,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, for (auto & i : infos) { to << store->printStorePath(i.first) << (i.second.deriver ? store->printStorePath(*i.second.deriver) : ""); - write(*store, to, i.second.references); + nix::worker_proto::write(*store, to, i.second.references); to << i.second.downloadSize << i.second.narSize; } break; @@ -620,7 +620,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, logger->startWork(); auto paths = store->queryAllValidPaths(); logger->stopWork(); - write(*store, to, paths); + nix::worker_proto::write(*store, to, paths); break; } @@ -639,7 +639,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, to << 1; to << (info->deriver ? store->printStorePath(*info->deriver) : "") << info->narHash->to_string(Base16, false); - write(*store, to, info->references); + nix::worker_proto::write(*store, to, info->references); to << info->registrationTime << info->narSize; if (GET_PROTOCOL_MINOR(clientVersion) >= 16) { to << info->ultimate @@ -699,7 +699,7 @@ static void performOp(TunnelLogger * logger, ref<Store> store, if (deriver != "") info.deriver = store->parseStorePath(deriver); info.narHash = Hash::parseAny(readString(from), htSHA256); - info.references = read(*store, from, Proxy<StorePathSet> {}); + info.references = nix::worker_proto::read(*store, from, Phantom<StorePathSet> {}); from >> info.registrationTime >> info.narSize >> info.ultimate; info.sigs = readStrings<StringSet>(from); info.ca = parseContentAddressOpt(readString(from)); @@ -799,9 +799,9 @@ static void performOp(TunnelLogger * logger, ref<Store> store, uint64_t downloadSize, narSize; store->queryMissing(targets, willBuild, willSubstitute, unknown, downloadSize, narSize); logger->stopWork(); - write(*store, to, willBuild); - write(*store, to, willSubstitute); - write(*store, to, unknown); + nix::worker_proto::write(*store, to, willBuild); + nix::worker_proto::write(*store, to, willSubstitute); + nix::worker_proto::write(*store, to, unknown); to << downloadSize << narSize; break; } diff --git a/src/libstore/derivations.cc b/src/libstore/derivations.cc index 23af4f59c..b7aef9d65 100644 --- a/src/libstore/derivations.cc +++ b/src/libstore/derivations.cc @@ -605,7 +605,7 @@ Source & readDerivation(Source & in, const Store & store, BasicDerivation & drv, drv.outputs.emplace(std::move(name), std::move(output)); } - drv.inputSrcs = read(store, in, Proxy<StorePathSet> {}); + drv.inputSrcs = nix::worker_proto::read(store, in, Phantom<StorePathSet> {}); in >> drv.platform >> drv.builder; drv.args = readStrings<Strings>(in); @@ -640,7 +640,7 @@ void writeDerivation(Sink & out, const Store & store, const BasicDerivation & dr }, }, i.second.output); } - write(store, out, drv.inputSrcs); + nix::worker_proto::write(store, out, drv.inputSrcs); out << drv.platform << drv.builder << drv.args; out << drv.env.size(); for (auto & i : drv.env) diff --git a/src/libstore/export-import.cc b/src/libstore/export-import.cc index c20c56156..7551294f7 100644 --- a/src/libstore/export-import.cc +++ b/src/libstore/export-import.cc @@ -45,7 +45,7 @@ void Store::exportPath(const StorePath & path, Sink & sink) teeSink << exportMagic << printStorePath(path); - write(*this, teeSink, info->references); + nix::worker_proto::write(*this, teeSink, info->references); teeSink << (info->deriver ? printStorePath(*info->deriver) : "") << 0; @@ -73,7 +73,7 @@ StorePaths Store::importPaths(Source & source, CheckSigsFlag checkSigs) //Activity act(*logger, lvlInfo, format("importing path '%s'") % info.path); - info.references = read(*this, source, Proxy<StorePathSet> {}); + info.references = nix::worker_proto::read(*this, source, Phantom<StorePathSet> {}); auto deriver = readString(source); if (deriver != "") diff --git a/src/libstore/legacy-ssh-store.cc b/src/libstore/legacy-ssh-store.cc index 52c03e116..0951610d3 100644 --- a/src/libstore/legacy-ssh-store.cc +++ b/src/libstore/legacy-ssh-store.cc @@ -107,7 +107,7 @@ struct LegacySSHStore : public Store auto deriver = readString(conn->from); if (deriver != "") info->deriver = parseStorePath(deriver); - info->references = read(*this, conn->from, Proxy<StorePathSet> {}); + info->references = nix::worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); readLongLong(conn->from); // download size info->narSize = readLongLong(conn->from); @@ -139,7 +139,7 @@ struct LegacySSHStore : public Store << printStorePath(info.path) << (info.deriver ? printStorePath(*info.deriver) : "") << info.narHash->to_string(Base16, false); - write(*this, conn->to, info.references); + nix::worker_proto::write(*this, conn->to, info.references); conn->to << info.registrationTime << info.narSize @@ -168,7 +168,7 @@ struct LegacySSHStore : public Store conn->to << exportMagic << printStorePath(info.path); - write(*this, conn->to, info.references); + nix::worker_proto::write(*this, conn->to, info.references); conn->to << (info.deriver ? printStorePath(*info.deriver) : "") << 0 @@ -251,10 +251,10 @@ struct LegacySSHStore : public Store conn->to << cmdQueryClosure << includeOutputs; - write(*this, conn->to, paths); + nix::worker_proto::write(*this, conn->to, paths); conn->to.flush(); - for (auto & i : read(*this, conn->from, Proxy<StorePathSet> {})) + for (auto & i : nix::worker_proto::read(*this, conn->from, Phantom<StorePathSet> {})) out.insert(i); } @@ -267,10 +267,10 @@ struct LegacySSHStore : public Store << cmdQueryValidPaths << false // lock << maybeSubstitute; - write(*this, conn->to, paths); + nix::worker_proto::write(*this, conn->to, paths); conn->to.flush(); - return read(*this, conn->from, Proxy<StorePathSet> {}); + return nix::worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); } void connect() override diff --git a/src/libstore/remote-store.cc b/src/libstore/remote-store.cc index 3c3577c13..ac776b95a 100644 --- a/src/libstore/remote-store.cc +++ b/src/libstore/remote-store.cc @@ -22,8 +22,9 @@ namespace nix { +namespace worker_proto { -std::string read(const Store & store, Source & from, Proxy<std::string> _) +std::string read(const Store & store, Source & from, Phantom<std::string> _) { return readString(from); } @@ -33,8 +34,7 @@ void write(const Store & store, Sink & out, const std::string & str) out << str; } - -StorePath read(const Store & store, Source & from, Proxy<StorePath> _) +StorePath read(const Store & store, Source & from, Phantom<StorePath> _) { return store.parseStorePath(readString(from)); } @@ -44,8 +44,7 @@ void write(const Store & store, Sink & out, const StorePath & storePath) out << store.printStorePath(storePath); } - -ContentAddress read(const Store & store, Source & from, Proxy<ContentAddress> _) +ContentAddress read(const Store & store, Source & from, Phantom<ContentAddress> _) { return parseContentAddress(readString(from)); } @@ -55,6 +54,8 @@ void write(const Store & store, Sink & out, const ContentAddress & ca) out << renderContentAddress(ca); } +} + /* TODO: Separate these store impls into different files, give them better names */ RemoteStore::RemoteStore(const Params & params) @@ -289,9 +290,9 @@ StorePathSet RemoteStore::queryValidPaths(const StorePathSet & paths, Substitute return res; } else { conn->to << wopQueryValidPaths; - write(*this, conn->to, paths); + nix::worker_proto::write(*this, conn->to, paths); conn.processStderr(); - return read(*this, conn->from, Proxy<StorePathSet> {}); + return worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); } } @@ -301,7 +302,7 @@ StorePathSet RemoteStore::queryAllValidPaths() auto conn(getConnection()); conn->to << wopQueryAllValidPaths; conn.processStderr(); - return read(*this, conn->from, Proxy<StorePathSet> {}); + return nix::worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); } @@ -318,9 +319,9 @@ StorePathSet RemoteStore::querySubstitutablePaths(const StorePathSet & paths) return res; } else { conn->to << wopQuerySubstitutablePaths; - write(*this, conn->to, paths); + nix::worker_proto::write(*this, conn->to, paths); conn.processStderr(); - return read(*this, conn->from, Proxy<StorePathSet> {}); + return worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); } } @@ -342,7 +343,7 @@ void RemoteStore::querySubstitutablePathInfos(const StorePathCAMap & pathsMap, S auto deriver = readString(conn->from); if (deriver != "") info.deriver = parseStorePath(deriver); - info.references = read(*this, conn->from, Proxy<StorePathSet> {}); + info.references = worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); info.downloadSize = readLongLong(conn->from); info.narSize = readLongLong(conn->from); infos.insert_or_assign(i.first, std::move(info)); @@ -355,9 +356,9 @@ void RemoteStore::querySubstitutablePathInfos(const StorePathCAMap & pathsMap, S StorePathSet paths; for (auto & path : pathsMap) paths.insert(path.first); - write(*this, conn->to, paths); + worker_proto::write(*this, conn->to, paths); } else - write(*this, conn->to, pathsMap); + worker_proto::write(*this, conn->to, pathsMap); conn.processStderr(); size_t count = readNum<size_t>(conn->from); for (size_t n = 0; n < count; n++) { @@ -365,7 +366,7 @@ void RemoteStore::querySubstitutablePathInfos(const StorePathCAMap & pathsMap, S auto deriver = readString(conn->from); if (deriver != "") info.deriver = parseStorePath(deriver); - info.references = read(*this, conn->from, Proxy<StorePathSet> {}); + info.references = worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); info.downloadSize = readLongLong(conn->from); info.narSize = readLongLong(conn->from); } @@ -398,7 +399,7 @@ void RemoteStore::queryPathInfoUncached(const StorePath & path, auto deriver = readString(conn->from); if (deriver != "") info->deriver = parseStorePath(deriver); info->narHash = Hash::parseAny(readString(conn->from), htSHA256); - info->references = read(*this, conn->from, Proxy<StorePathSet> {}); + info->references = worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); conn->from >> info->registrationTime >> info->narSize; if (GET_PROTOCOL_MINOR(conn->daemonVersion) >= 16) { conn->from >> info->ultimate; @@ -417,7 +418,7 @@ void RemoteStore::queryReferrers(const StorePath & path, auto conn(getConnection()); conn->to << wopQueryReferrers << printStorePath(path); conn.processStderr(); - for (auto & i : read(*this, conn->from, Proxy<StorePathSet> {})) + for (auto & i : worker_proto::read(*this, conn->from, Phantom<StorePathSet> {})) referrers.insert(i); } @@ -427,7 +428,7 @@ StorePathSet RemoteStore::queryValidDerivers(const StorePath & path) auto conn(getConnection()); conn->to << wopQueryValidDerivers << printStorePath(path); conn.processStderr(); - return read(*this, conn->from, Proxy<StorePathSet> {}); + return worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); } @@ -439,7 +440,7 @@ StorePathSet RemoteStore::queryDerivationOutputs(const StorePath & path) } conn->to << wopQueryDerivationOutputs << printStorePath(path); conn.processStderr(); - return read(*this, conn->from, Proxy<StorePathSet> {}); + return worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); } @@ -448,7 +449,7 @@ std::map<std::string, std::optional<StorePath>> RemoteStore::queryDerivationOutp auto conn(getConnection()); conn->to << wopQueryDerivationOutputMap << printStorePath(path); conn.processStderr(); - return read(*this, conn->from, Proxy<std::map<std::string, std::optional<StorePath>>> {}); + return worker_proto::read(*this, conn->from, Phantom<std::map<std::string, std::optional<StorePath>>> {}); } @@ -478,7 +479,7 @@ void RemoteStore::addToStore(const ValidPathInfo & info, Source & source, sink << exportMagic << printStorePath(info.path); - write(*this, sink, info.references); + worker_proto::write(*this, sink, info.references); sink << (info.deriver ? printStorePath(*info.deriver) : "") << 0 // == no legacy signature @@ -488,7 +489,7 @@ void RemoteStore::addToStore(const ValidPathInfo & info, Source & source, conn.processStderr(0, source2.get()); - auto importedPaths = read(*this, conn->from, Proxy<StorePathSet> {}); + auto importedPaths = worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); assert(importedPaths.size() <= 1); } @@ -497,7 +498,7 @@ void RemoteStore::addToStore(const ValidPathInfo & info, Source & source, << printStorePath(info.path) << (info.deriver ? printStorePath(*info.deriver) : "") << info.narHash->to_string(Base16, false); - write(*this, conn->to, info.references); + worker_proto::write(*this, conn->to, info.references); conn->to << info.registrationTime << info.narSize << info.ultimate << info.sigs << renderContentAddress(info.ca) << repair << !checkSigs; @@ -630,7 +631,7 @@ StorePath RemoteStore::addTextToStore(const string & name, const string & s, auto conn(getConnection()); conn->to << wopAddTextToStore << name << s; - write(*this, conn->to, references); + worker_proto::write(*this, conn->to, references); conn.processStderr(); return parseStorePath(readString(conn->from)); @@ -732,7 +733,7 @@ void RemoteStore::collectGarbage(const GCOptions & options, GCResults & results) conn->to << wopCollectGarbage << options.action; - write(*this, conn->to, options.pathsToDelete); + worker_proto::write(*this, conn->to, options.pathsToDelete); conn->to << options.ignoreLiveness << options.maxFreed /* removed options */ @@ -794,9 +795,9 @@ void RemoteStore::queryMissing(const std::vector<StorePathWithOutputs> & targets ss.push_back(p.to_string(*this)); conn->to << ss; conn.processStderr(); - willBuild = read(*this, conn->from, Proxy<StorePathSet> {}); - willSubstitute = read(*this, conn->from, Proxy<StorePathSet> {}); - unknown = read(*this, conn->from, Proxy<StorePathSet> {}); + willBuild = worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); + willSubstitute = worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); + unknown = worker_proto::read(*this, conn->from, Phantom<StorePathSet> {}); conn->from >> downloadSize >> narSize; return; } diff --git a/src/libstore/worker-protocol.hh b/src/libstore/worker-protocol.hh index 08eec9b48..b4c260e26 100644 --- a/src/libstore/worker-protocol.hh +++ b/src/libstore/worker-protocol.hh @@ -68,15 +68,43 @@ struct Source; /* To guide overloading */ template<typename T> -struct Proxy {}; +struct Phantom {}; + +namespace worker_proto { +/* FIXME maybe move more stuff inside here */ + +std::string read(const Store & store, Source & from, Phantom<std::string> _); +void write(const Store & store, Sink & out, const std::string & str); + +StorePath read(const Store & store, Source & from, Phantom<StorePath> _); +void write(const Store & store, Sink & out, const StorePath & storePath); + +ContentAddress read(const Store & store, Source & from, Phantom<ContentAddress> _); +void write(const Store & store, Sink & out, const ContentAddress & ca); + +template<typename T> +std::set<T> read(const Store & store, Source & from, Phantom<std::set<T>> _); template<typename T> -std::set<T> read(const Store & store, Source & from, Proxy<std::set<T>> _) +void write(const Store & store, Sink & out, const std::set<T> & resSet); + +template<typename K, typename V> +std::map<K, V> read(const Store & store, Source & from, Phantom<std::map<K, V>> _); +template<typename K, typename V> +void write(const Store & store, Sink & out, const std::map<K, V> & resMap); + +template<typename T> +std::optional<T> read(const Store & store, Source & from, Phantom<std::optional<T>> _); +template<typename T> +void write(const Store & store, Sink & out, const std::optional<T> & optVal); + +template<typename T> +std::set<T> read(const Store & store, Source & from, Phantom<std::set<T>> _) { std::set<T> resSet; auto size = readNum<size_t>(from); while (size--) { - resSet.insert(read(store, from, Proxy<T> {})); + resSet.insert(read(store, from, Phantom<T> {})); } return resSet; } @@ -91,14 +119,14 @@ void write(const Store & store, Sink & out, const std::set<T> & resSet) } template<typename K, typename V> -std::map<K, V> read(const Store & store, Source & from, Proxy<std::map<K, V>> _) +std::map<K, V> read(const Store & store, Source & from, Phantom<std::map<K, V>> _) { std::map<K, V> resMap; auto size = readNum<size_t>(from); while (size--) { resMap.insert_or_assign( - read(store, from, Proxy<K> {}), - read(store, from, Proxy<V> {})); + read(store, from, Phantom<K> {}), + read(store, from, Phantom<V> {})); } return resMap; } @@ -107,21 +135,23 @@ template<typename K, typename V> void write(const Store & store, Sink & out, const std::map<K, V> & resMap) { out << resMap.size(); - for (auto & [key, value] : resMap) { - write(store, out, key); - write(store, out, value); + for (auto & i : resMap) { + // out << i.first; + write(store, out, i.first); + write(store, out, i.second); } } template<typename T> -std::optional<T> read(const Store & store, Source & from, Proxy<std::optional<T>> _) +std::optional<T> read(const Store & store, Source & from, Phantom<std::optional<T>> _) { auto tag = readNum<uint8_t>(from); switch (tag) { case 0: return std::nullopt; case 1: - return read(store, from, Proxy<T> {}); + // return nix::worker_proto::read(store, from, Phantom<T> {}); + return read(store, from, Phantom<T> {}); default: throw Error("got an invalid tag bit for std::optional: %#04x", (size_t)tag); } @@ -132,19 +162,11 @@ void write(const Store & store, Sink & out, const std::optional<T> & optVal) { out << (uint64_t) (optVal ? 1 : 0); if (optVal) - write(store, out, *optVal); + nix::worker_proto::write(store, out, *optVal); } -std::string read(const Store & store, Source & from, Proxy<std::string> _); - -void write(const Store & store, Sink & out, const std::string & str); - -StorePath read(const Store & store, Source & from, Proxy<StorePath> _); - -void write(const Store & store, Sink & out, const StorePath & storePath); -ContentAddress read(const Store & store, Source & from, Proxy<ContentAddress> _); +} -void write(const Store & store, Sink & out, const ContentAddress & ca); } |