mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-11 09:18:25 +03:00
05e87b6b65
effectively emulating a future "pvscan --lvmetad" command.
128 lines
3.4 KiB
C
128 lines
3.4 KiB
C
#include "lvmetad-client.h"
|
|
#include "label.h"
|
|
#include "lvmcache.h"
|
|
#include "metadata.h"
|
|
|
|
const char *uuid1 = "abcd-efgh";
|
|
const char *uuid2 = "bbcd-efgh";
|
|
const char *vgid = "yada-yada";
|
|
const char *uuid3 = "cbcd-efgh";
|
|
|
|
const char *metadata2 = "{\n"
|
|
"id = \"yada-yada\"\n"
|
|
"seqno = 15\n"
|
|
"status = [\"READ\", \"WRITE\"]\n"
|
|
"flags = []\n"
|
|
"extent_size = 8192\n"
|
|
"physical_volumes {\n"
|
|
" pv0 {\n"
|
|
" id = \"abcd-efgh\"\n"
|
|
" }\n"
|
|
" pv1 {\n"
|
|
" id = \"bbcd-efgh\"\n"
|
|
" }\n"
|
|
" pv2 {\n"
|
|
" id = \"cbcd-efgh\"\n"
|
|
" }\n"
|
|
"}\n"
|
|
"}\n";
|
|
|
|
void _handle_reply(daemon_reply reply) {
|
|
const char *repl = daemon_reply_str(reply, "response", NULL);
|
|
const char *status = daemon_reply_str(reply, "status", NULL);
|
|
const char *vgid = daemon_reply_str(reply, "vgid", NULL);
|
|
|
|
fprintf(stderr, "[C] REPLY: %s\n", repl);
|
|
if (!strcmp(repl, "failed"))
|
|
fprintf(stderr, "[C] REASON: %s\n", daemon_reply_str(reply, "reason", "unknown"));
|
|
if (vgid)
|
|
fprintf(stderr, "[C] VGID: %s\n", vgid);
|
|
if (status)
|
|
fprintf(stderr, "[C] STATUS: %s\n", status);
|
|
daemon_reply_destroy(reply);
|
|
}
|
|
|
|
void _pv_add(daemon_handle h, const char *uuid, const char *metadata)
|
|
{
|
|
daemon_reply reply = daemon_send_simple(h, "pv_add", "uuid = %s", uuid,
|
|
"metadata = %b", metadata,
|
|
NULL);
|
|
_handle_reply(reply);
|
|
}
|
|
|
|
int scan(daemon_handle h, char *fn) {
|
|
struct device *dev = dev_cache_get(fn, NULL);
|
|
|
|
struct label *label;
|
|
if (!label_read(dev, &label, 0)) {
|
|
fprintf(stderr, "[C] no label found on %s\n", fn);
|
|
return;
|
|
}
|
|
|
|
char uuid[64];
|
|
id_write_format(dev->pvid, uuid, 64);
|
|
fprintf(stderr, "[C] found PV: %s\n", uuid);
|
|
struct lvmcache_info *info = (struct lvmcache_info *) label->info;
|
|
struct physical_volume pv = { 0, };
|
|
|
|
if (!(info->fmt->ops->pv_read(info->fmt, dev_name(dev), &pv, 0))) {
|
|
fprintf(stderr, "[C] Failed to read PV %s", dev_name(dev));
|
|
return;
|
|
}
|
|
|
|
struct format_instance_ctx fic;
|
|
struct format_instance *fid = info->fmt->ops->create_instance(info->fmt, &fic);
|
|
struct metadata_area *mda;
|
|
struct volume_group *vg = NULL;
|
|
dm_list_iterate_items(mda, &info->mdas) {
|
|
struct volume_group *this = mda->ops->vg_read(fid, "", mda);
|
|
if (this && !vg || this->seqno > vg->seqno)
|
|
vg = this;
|
|
}
|
|
if (vg) {
|
|
char *buf = NULL;
|
|
/* TODO. This is not entirely correct, since export_vg_to_buffer
|
|
* adds trailing garbage to the buffer. We may need to use
|
|
* export_vg_to_config_tree and format the buffer ourselves. It
|
|
* does, however, work for now, since the garbage is well
|
|
* formatted and has no conflicting keys with the rest of the
|
|
* request. */
|
|
export_vg_to_buffer(vg, &buf);
|
|
daemon_reply reply =
|
|
daemon_send_simple(h, "pv_add", "uuid = %s", uuid,
|
|
"metadata = %b", strchr(buf, '{'),
|
|
NULL);
|
|
_handle_reply(reply);
|
|
}
|
|
}
|
|
|
|
void _dump_vg(daemon_handle h, const char *uuid)
|
|
{
|
|
daemon_reply reply = daemon_send_simple(h, "vg_by_uuid", "uuid = %s", uuid, NULL);
|
|
fprintf(stderr, "[C] reply buffer: %s\n", reply.buffer);
|
|
daemon_reply_destroy(reply);
|
|
}
|
|
|
|
int main(int argc, char **argv) {
|
|
daemon_handle h = lvmetad_open();
|
|
|
|
if (argc > 1) {
|
|
int i;
|
|
struct cmd_context *cmd = create_toolcontext(0, NULL, 0);
|
|
for (i = 1; i < argc; ++i) {
|
|
const char *uuid = NULL;
|
|
scan(h, argv[i]);
|
|
}
|
|
destroy_toolcontext(cmd);
|
|
return 0;
|
|
}
|
|
|
|
_pv_add(h, uuid1, NULL);
|
|
_pv_add(h, uuid2, metadata2);
|
|
_dump_vg(h, vgid);
|
|
_pv_add(h, uuid3, NULL);
|
|
|
|
daemon_close(h);
|
|
return 0;
|
|
}
|