Files
Flipper-ARF/lib/mlib/example/ex-multi03.c
Andrea Santaniello d6c2757f21 First push, clean slate
2026-03-08 18:48:37 +01:00

195 lines
4.9 KiB
C

#include <time.h>
#include <stdio.h>
/* Register a dynamic seed to M*LIB hash function.
This shall be done before any M*LIB header inclusion. */
unsigned long long rand_seed;
#define M_USE_HASH_SEED rand_seed
#include "m-array.h"
#include "m-dict.h"
#include "m-string.h"
#include "m-variant.h"
/* This example will show how to create JSON-like files.
This is a JSON **like** syntax, not exactly JSON format.
For an example with JSON format, see https://github.com/P-p-H-d/mjsonlib
*/
/* First let's handle the OUT/IN functions of boolean */
static inline void boolean_out_str(FILE *f, bool b)
{
fprintf(f, b ? "true" : "false");
}
static inline bool boolean_in_str(bool *b, FILE *f)
{
char c = fgetc(f);
if (c == 't') {
*b = true;
c = fgetc(f);
if (c != 'r') return false;
c = fgetc(f);
if (c != 'u') return false;
c = fgetc(f);
return c == 'e';
} else if (c == 'f') {
*b = false;
c = fgetc(f);
if (c != 'a') return false;
c = fgetc(f);
if (c != 'l') return false;
c = fgetc(f);
if (c != 's') return false;
c = fgetc(f);
return c == 'e';
} else {
return false;
}
}
/* Then the OUT/IN functions of float */
static inline void real_out_str(FILE *f, float r)
{
fprintf(f, "%f", (double) r);
}
static inline bool real_in_str(float *r, FILE *f)
{
return fscanf(f, "%f", r) == 1;
}
/* Let's define the JSON structure & its interface.
The JSON structure is incomplete as it can only be complete
after the definition of all its content.
*/
typedef struct json_node_s *json_t;
extern void json_init (json_t *);
extern void json_clear (json_t);
extern void json_init_set (json_t *, json_t);
extern void json_set (json_t *, json_t);
extern void json_out_str(FILE *, json_t);
extern bool json_in_str(json_t *, FILE *);
#define JSON_OPLIST \
(INIT(json_init M_IPTR), CLEAR(json_clear), \
INIT_SET(json_init_set M_IPTR), SET(json_set M_IPTR), \
OUT_STR(json_out_str), IN_STR(json_in_str M_IPTR) )
DICT_DEF2(dict_json, string_t, STRING_OPLIST, json_t, JSON_OPLIST)
ARRAY_DEF(array_json, json_t, JSON_OPLIST)
VARIANT_DEF2(variant_json,
(boolean, bool, M_OPEXTEND(M_DEFAULT_OPLIST, OUT_STR(boolean_out_str), IN_STR(boolean_in_str M_IPTR)) ),
(real, float, M_OPEXTEND (M_DEFAULT_OPLIST, OUT_STR(real_out_str), IN_STR(real_in_str M_IPTR))),
(array, array_json_t, ARRAY_OPLIST(array_json, JSON_OPLIST)),
(dict, dict_json_t, DICT_OPLIST(dict_json, STRING_OPLIST, JSON_OPLIST)))
/* Let's complete the structure */
struct json_node_s {
variant_json_t json;
};
/* Body definition */
void json_init(json_t *p)
{
assert (p != NULL);
*p = malloc (sizeof (struct json_node_s));
if (*p == NULL) abort();
variant_json_init((*p)->json);
}
void json_clear (json_t p)
{
assert (p != NULL);
variant_json_clear (p->json);
free(p);
}
void json_init_set(json_t *p, json_t o)
{
assert (p != NULL && o != NULL);
*p = malloc (sizeof (struct json_node_s));
if (*p == NULL) abort();
variant_json_init_set((*p)->json, o->json);
}
void json_set(json_t *p, json_t o)
{
assert (p != NULL && *p != NULL && o != NULL);
variant_json_set((*p)->json, o->json);
}
void json_out_str(FILE *f, json_t o)
{
variant_json_out_str(f, o->json);
}
bool json_in_str(json_t *p, FILE *f)
{
assert (p != NULL);
return variant_json_in_str((*p)->json, f);
}
/* test */
static json_t generate(void)
{
json_t p, k;
dict_json_t d;
array_json_t a;
string_t s;
/* This is a low level generation for example */
json_init(&p);
json_init(&k);
array_json_init(a);
dict_json_init(d);
string_init (s);
/* Let's generate the following structure :
{ channel: TRUE, filter: 2.3, tab: [2., 3.] } */
variant_json_set_boolean(k->json, true);
dict_json_set_at(d, STRING_CTE("channel"), k);
variant_json_set_real(k->json, 2.3);
dict_json_set_at(d, STRING_CTE("filter"), k);
variant_json_set_real(k->json, 2.);
array_json_push_back (a, k);
variant_json_set_real(k->json, 3.);
array_json_push_back (a, k);
variant_json_set_array (k->json, a);
dict_json_set_at(d, STRING_CTE("tab"), k);
variant_json_set_dict(p->json, d);
json_clear(k);
string_clear(s);
dict_json_clear(d);
array_json_clear(a);
return p;
}
int main(void)
{
/* Compute a new random seed for the hash computation */
srand(time(NULL));
rand_seed = ((rand() * RAND_MAX + rand()) * RAND_MAX + rand()) * RAND_MAX + rand();
/* Generate data */
json_t p = generate();
json_out_str(stdout, p);
/* Typical Output:
@dict@{"filter":@real@2.300000@,"tab":@array@[@real@2.000000@,@real@3.000000@]@,"channel":@boolean@true@}@
If you want a true JSON format, you'll have to overload the variant IN/OUT function so that they
can detect the type of argument and, as such, don't need to output it in the format */
printf("\n");
json_clear(p);
return 0;
}