123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899 |
- /* vim: set expandtab ts=4 sw=4: */
- /*
- * You may redistribute this program and/or modify it under the terms of
- * the GNU General Public License as published by the Free Software Foundation,
- * either version 3 of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program. If not, see <https://www.gnu.org/licenses/>.
- */
- #include "admin/Admin.h"
- #include "benc/List.h"
- #include "crypto/Key.h"
- #include "subnode/ReachabilityCollector.h"
- #include "subnode/ReachabilityCollector_admin.h"
- #include "util/AddrTools.h"
- #include "util/Identity.h"
- struct Context {
- struct Admin* admin;
- struct Allocator* alloc;
- struct ReachabilityCollector* rc;
- Identity
- };
- static List* numList(void* buf, uint32_t numberSz, struct Allocator* alloc)
- {
- List* l = List_new(alloc);
- for (int i = 0; i < LinkState_SLOTS; i++) {
- int64_t num;
- if (numberSz == 2) {
- num = ((uint16_t*)buf)[i];
- } else if (numberSz == 4) {
- num = ((uint32_t*)buf)[i];
- } else {
- Assert_failure("unexpected number size");
- }
- List_addInt(l, num, alloc);
- }
- return l;
- }
- #define NODES_PER_PAGE 8
- static void getPeerInfo(Dict* args, void* vcontext, String* txid, struct Allocator* requestAlloc)
- {
- struct Context* ctx = Identity_check((struct Context*) vcontext);
- int page = 0;
- int64_t* pageP = Dict_getIntC(args, "page");
- if (pageP && *pageP > 0) { page = *pageP; }
- List* peerList = List_new(requestAlloc);
- for (int i = page * NODES_PER_PAGE, j = 0; j < NODES_PER_PAGE; i++, j++) {
- struct ReachabilityCollector_PeerInfo* pi = ReachabilityCollector_getPeerInfo(ctx->rc, i);
- if (!pi) { break; }
- Dict* pid = Dict_new(requestAlloc);
- Dict_putStringC(pid, "addr", Address_toString(&pi->addr, requestAlloc), requestAlloc);
- uint8_t rpath[20];
- AddrTools_printPath(rpath, pi->pathThemToUs);
- Dict_putStringC(
- pid, "pathThemToUs", String_newBinary(rpath, 19, requestAlloc), requestAlloc);
- Dict_putIntC(pid, "querying", pi->querying, requestAlloc);
- Dict_putListC(
- pid, "lagSlots", numList(pi->linkState.lagSlots, 2, requestAlloc), requestAlloc);
- Dict_putListC(
- pid, "dropSlots", numList(pi->linkState.dropSlots, 2, requestAlloc), requestAlloc);
- Dict_putListC(
- pid, "kbRecvSlots", numList(pi->linkState.kbRecvSlots, 4, requestAlloc), requestAlloc);
- Dict_putIntC(
- pid, "samples", pi->linkState.samples, requestAlloc);
- List_addDict(peerList, pid, requestAlloc);
- }
- Dict* out = Dict_new(requestAlloc);
- Dict_putListC(out, "peers", peerList, requestAlloc);
- Dict_putStringCC(out, "error", "none", requestAlloc);
- Admin_sendMessage(out, txid, ctx->admin);
- }
- void ReachabilityCollector_admin_register(struct ReachabilityCollector* rc,
- struct Admin* admin,
- struct Allocator* alloc)
- {
- struct Context* ctx = Allocator_clone(alloc, (&(struct Context) {
- .admin = admin,
- .alloc = alloc,
- .rc = rc
- }));
- Identity_set(ctx);
- Admin_registerFunction("ReachabilityCollector_getPeerInfo", getPeerInfo, ctx, true,
- ((struct Admin_FunctionArg[]) {
- { .name = "page", .required = true, .type = "Int" }
- }), admin);
- }
|