Blame json_visit.c

Packit ea8578
/*
Packit ea8578
 * Copyright (c) 2016 Eric Haszlakiewicz
Packit ea8578
 *
Packit ea8578
 * This is free software; you can redistribute it and/or modify
Packit ea8578
 * it under the terms of the MIT license. See COPYING for details.
Packit ea8578
 */
Packit ea8578
Packit ea8578
#include <stdio.h>
Packit ea8578
Packit ea8578
#include "config.h"
Packit ea8578
#include "json_inttypes.h"
Packit ea8578
#include "json_object.h"
Packit ea8578
#include "json_visit.h"
Packit ea8578
#include "linkhash.h"
Packit ea8578
Packit ea8578
static int _json_c_visit(json_object *jso, json_object *parent_jso,
Packit ea8578
                         const char *jso_key, size_t *jso_index,
Packit ea8578
                         json_c_visit_userfunc *userfunc, void *userarg);
Packit ea8578
Packit ea8578
int json_c_visit(json_object *jso, int future_flags,
Packit ea8578
                 json_c_visit_userfunc *userfunc, void *userarg)
Packit ea8578
{
Packit ea8578
	int ret = _json_c_visit(jso, NULL, NULL, NULL, userfunc, userarg);
Packit ea8578
	switch(ret)
Packit ea8578
	{
Packit ea8578
	case JSON_C_VISIT_RETURN_CONTINUE:
Packit ea8578
	case JSON_C_VISIT_RETURN_SKIP:
Packit ea8578
	case JSON_C_VISIT_RETURN_POP:
Packit ea8578
	case JSON_C_VISIT_RETURN_STOP:
Packit ea8578
		return 0;
Packit ea8578
	default:
Packit ea8578
		return JSON_C_VISIT_RETURN_ERROR;
Packit ea8578
	}
Packit ea8578
}
Packit ea8578
static int _json_c_visit(json_object *jso, json_object *parent_jso,
Packit ea8578
                         const char *jso_key, size_t *jso_index,
Packit ea8578
                         json_c_visit_userfunc *userfunc, void *userarg)
Packit ea8578
{
Packit ea8578
	int userret = userfunc(jso, 0, parent_jso, jso_key, jso_index, userarg);
Packit ea8578
	switch(userret)
Packit ea8578
	{
Packit ea8578
	case JSON_C_VISIT_RETURN_CONTINUE:
Packit ea8578
		break;
Packit ea8578
	case JSON_C_VISIT_RETURN_SKIP:
Packit ea8578
	case JSON_C_VISIT_RETURN_POP:
Packit ea8578
	case JSON_C_VISIT_RETURN_STOP:
Packit ea8578
	case JSON_C_VISIT_RETURN_ERROR:
Packit ea8578
		return userret;
Packit ea8578
	default:
Packit ea8578
		fprintf(stderr, "ERROR: invalid return value from json_c_visit userfunc: %d\n", userret);
Packit ea8578
		return JSON_C_VISIT_RETURN_ERROR;
Packit ea8578
	}
Packit ea8578
Packit ea8578
	switch(json_object_get_type(jso))
Packit ea8578
	{
Packit ea8578
	case json_type_null:
Packit ea8578
	case json_type_boolean:
Packit ea8578
	case json_type_double:
Packit ea8578
	case json_type_int:
Packit ea8578
	case json_type_string:
Packit ea8578
		// we already called userfunc above, move on to the next object
Packit ea8578
		return JSON_C_VISIT_RETURN_CONTINUE;
Packit ea8578
Packit ea8578
	case json_type_object:
Packit ea8578
	{
Packit ea8578
		json_object_object_foreach(jso, key, child)
Packit ea8578
		{
Packit ea8578
			userret = _json_c_visit(child, jso, key, NULL, userfunc, userarg);
Packit ea8578
			if (userret == JSON_C_VISIT_RETURN_POP)
Packit ea8578
				break;
Packit ea8578
			if (userret == JSON_C_VISIT_RETURN_STOP ||
Packit ea8578
				userret == JSON_C_VISIT_RETURN_ERROR)
Packit ea8578
				return userret;
Packit ea8578
			if (userret != JSON_C_VISIT_RETURN_CONTINUE &&
Packit ea8578
				userret != JSON_C_VISIT_RETURN_SKIP)
Packit ea8578
			{
Packit ea8578
				fprintf(stderr, "INTERNAL ERROR: _json_c_visit returned %d\n", userret);
Packit ea8578
				return JSON_C_VISIT_RETURN_ERROR;
Packit ea8578
			}
Packit ea8578
		}
Packit ea8578
		break;
Packit ea8578
	}
Packit ea8578
	case json_type_array:
Packit ea8578
	{
Packit ea8578
		size_t array_len = json_object_array_length(jso);
Packit ea8578
		size_t ii;
Packit ea8578
		for (ii = 0; ii < array_len; ii++)
Packit ea8578
		{
Packit ea8578
			json_object *child = json_object_array_get_idx(jso, ii);
Packit ea8578
			userret = _json_c_visit(child, jso, NULL, &ii, userfunc, userarg);
Packit ea8578
			if (userret == JSON_C_VISIT_RETURN_POP)
Packit ea8578
				break;
Packit ea8578
			if (userret == JSON_C_VISIT_RETURN_STOP ||
Packit ea8578
				userret == JSON_C_VISIT_RETURN_ERROR)
Packit ea8578
				return userret;
Packit ea8578
			if (userret != JSON_C_VISIT_RETURN_CONTINUE &&
Packit ea8578
				userret != JSON_C_VISIT_RETURN_SKIP)
Packit ea8578
			{
Packit ea8578
				fprintf(stderr, "INTERNAL ERROR: _json_c_visit returned %d\n", userret);
Packit ea8578
				return JSON_C_VISIT_RETURN_ERROR;
Packit ea8578
			}
Packit ea8578
		}
Packit ea8578
		break;
Packit ea8578
	}
Packit ea8578
	default:
Packit ea8578
		fprintf(stderr, "INTERNAL ERROR: _json_c_visit found object of unknown type: %d\n", json_object_get_type(jso));
Packit ea8578
		return JSON_C_VISIT_RETURN_ERROR;
Packit ea8578
	}
Packit ea8578
Packit ea8578
	// Call userfunc for the second type on container types, after all
Packit ea8578
	//  members of the container have been visited.
Packit ea8578
	// Non-container types will have already returned before this point.
Packit ea8578
Packit ea8578
	userret = userfunc(jso, JSON_C_VISIT_SECOND, parent_jso, jso_key, jso_index, userarg);
Packit ea8578
	switch(userret)
Packit ea8578
	{
Packit ea8578
	case JSON_C_VISIT_RETURN_SKIP:
Packit ea8578
	case JSON_C_VISIT_RETURN_POP:
Packit ea8578
		// These are not really sensible during JSON_C_VISIT_SECOND, 
Packit ea8578
		// but map them to JSON_C_VISIT_CONTINUE anyway.
Packit ea8578
		// FALLTHROUGH
Packit ea8578
	case JSON_C_VISIT_RETURN_CONTINUE:
Packit ea8578
		return JSON_C_VISIT_RETURN_CONTINUE;
Packit ea8578
	case JSON_C_VISIT_RETURN_STOP:
Packit ea8578
	case JSON_C_VISIT_RETURN_ERROR:
Packit ea8578
		return userret;
Packit ea8578
	default:
Packit ea8578
		fprintf(stderr, "ERROR: invalid return value from json_c_visit userfunc: %d\n", userret);
Packit ea8578
		return JSON_C_VISIT_RETURN_ERROR;
Packit ea8578
	}
Packit ea8578
	// NOTREACHED
Packit ea8578
}
Packit ea8578