Вместо того, чтобы пытаться собрать JSON как строки, используйте библиотеку, такую как json-glib , чтобы построить JSON для вас. Это более гибко и будет обрабатывать все виды крайних случаев. Он предоставляет JsonBuilder для построения структур JSON.
Начнем с того, что берем указатель файла, что-то еще должно открыть файл. Затем мы запускаем JsonBuilder и начинаем строить структуру JSON, объявляя объект { "Clients"
и запуская массив.
JsonNode *bash_connected_clients_to_json(FILE *fp) {
JsonBuilder *builder = json_builder_new();
// { "Clients": [ ...
json_builder_begin_object(builder);
json_builder_set_member_name(builder, "Clients");
json_builder_begin_array(builder);
Теперь мы читаем каждую строку и отправляем ее и компоновщик в функцию для обработки строки и добавления ее в открытый массив.
char line[1024];
while (fgets(line, sizeof(line), fp) != NULL) {
bash_connected_clients_line_to_json(line, builder);
}
Наконец закройте массив и объект и верните JsonNode , который мы только что создали.
// ... ] }
json_builder_end_array(builder);
json_builder_end_object(builder);
return json_builder_get_root(builder);
}
Тогда можно распечатать JsonNode.
int main() {
JsonNode *json = bash_connected_clients_to_json(stdin);
printf("%s", json_to_string(json, TRUE));
}
Обработка каждой строки начинается с ее анализа. Это можно сделать по-разному. sscanf
отлично работает.
void bash_connected_clients_line_to_json( const char *line, JsonBuilder *builder ) {
char hostname[1024], ip[1024], macaddr[1024];
if( sscanf(line, "%1023s %1023s %1023s", hostname, ip, macaddr) != 3 ) {
fprintf(stderr, "Could not parse line: '%s'\n", line);
return;
}
Затем мы добавляем объект JSON в массив, который у нас уже открыт, добавляем каждый из наших элементов в объект и закрываем объект.
// { "Hostname": "foo", "IP", "bar", "MacAddr", "baz" }
json_builder_begin_object(builder);
json_builder_set_member_name(builder, "Hostname");
json_builder_add_string_value(builder, hostname);
json_builder_set_member_name(builder, "IP");
json_builder_add_string_value(builder, ip);
json_builder_set_member_name(builder, "MacAddr");
json_builder_add_string_value(builder, macaddr);
json_builder_end_object(builder);
}
$ cat > test.txt
example.com 127.0.0.1 mactonight
foo.biz 0.0.0.0 12:34:56:78
$ ./test < test.txt
{
"Clients" : [
{
"Hostname" : "example.com",
"IP" : "127.0.0.1",
"MacAddr" : "mactonight"
},
{
"Hostname" : "foo.biz",
"IP" : "0.0.0.0",
"MacAddr" : "12:34:56:78"
}
]
}
Или вы можете сделать это в несколько строк Ruby.
require 'json'
clients = []
STDIN.each do |line|
fields = line.split(/\s+/)
clients << {
Hostname: fields[0],
IP: fields[1],
MacAddr: fields[2]
}
end
connections = {}
connections[:Clients] = clients
puts connections.to_json
Для json-c это в основном то же самое. Основное отличие состоит в том, что вместо bash_connected_clients_line_to_json
добавление объекта JSON в конструктор возвращает объект JSON.
json_object* bash_connected_clients_line_to_json( const char *line ) {
char hostname[1024], ip[1024], macaddr[1024];
if( sscanf(line, "%1023s %1023s %1023s", hostname, ip, macaddr) != 3 ) {
fprintf(stderr, "Could not parse line: '%s'\n", line);
return NULL;
}
json_object *json = json_object_new_object();
json_object_object_add(json, "Hostname", json_object_new_string(hostname));
json_object_object_add(json, "IP", json_object_new_string(ip));
json_object_object_add(json, "MacAddr", json_object_new_string(macaddr));
return json;
}
Затем он добавляется в массив клиентов JSON.
json_object *bash_connected_clients_to_json(FILE *fp) {
json_object *clients = json_object_new_array();
char line[1024];
while (fgets(line, sizeof(line), fp) != NULL) {
json_object_array_add(
clients,
bash_connected_clients_line_to_json(line)
);
}
json_object *json = json_object_new_object();
json_object_object_add(json, "Clients", clients);
return json;
}