1
0
mirror of https://github.com/pgbackrest/pgbackrest.git synced 2025-04-21 11:57:01 +02:00

210 lines
6.2 KiB
C
Raw Normal View History

/***********************************************************************************************************************************
Yaml Handler
***********************************************************************************************************************************/
#include "build.auto.h"
#include <yaml.h>
#include "common/debug.h"
#include "common/log.h"
#include "common/memContext.h"
#include "build/common/yaml.h"
/***********************************************************************************************************************************
Object type
***********************************************************************************************************************************/
struct Yaml
{
MemContext *memContext; // Mem context
yaml_parser_t parser; // Parse context
};
/***********************************************************************************************************************************
Free parser context
***********************************************************************************************************************************/
static void
yamlFreeResource(THIS_VOID)
{
THIS(Yaml);
FUNCTION_LOG_BEGIN(logLevelTrace);
FUNCTION_LOG_PARAM(YAML, this);
FUNCTION_LOG_END();
ASSERT(this != NULL);
yaml_parser_delete(&this->parser);
FUNCTION_LOG_RETURN_VOID();
}
/**********************************************************************************************************************************/
Yaml *
yamlNew(const Buffer *const buffer)
{
FUNCTION_TEST_BEGIN();
FUNCTION_TEST_PARAM(BUFFER, buffer);
FUNCTION_TEST_END();
Yaml *this = NULL;
MEM_CONTEXT_NEW_BEGIN("Yaml")
{
// Create object
this = memNew(sizeof(Yaml));
*this = (Yaml)
{
.memContext = MEM_CONTEXT_NEW(),
};
// Initialize parser context
CHECK(yaml_parser_initialize(&this->parser));
memContextCallbackSet(this->memContext, yamlFreeResource, this);
// Set yaml string
yaml_parser_set_input_string(&this->parser, bufPtrConst(buffer), bufUsed(buffer));
// Start document
CHECK(yamlEventNext(this).type == yamlEventTypeStreamBegin);
CHECK(yamlEventNext(this).type == yamlEventTypeDocBegin);
}
MEM_CONTEXT_NEW_END();
FUNCTION_TEST_RETURN(this);
}
/**********************************************************************************************************************************/
// Helper to map event type
static YamlEventType
yamlEventType(yaml_event_type_t type)
{
FUNCTION_TEST_BEGIN();
FUNCTION_TEST_PARAM(ENUM, type);
FUNCTION_TEST_END();
switch (type)
{
case YAML_STREAM_START_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeStreamBegin);
case YAML_STREAM_END_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeStreamEnd);
case YAML_DOCUMENT_START_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeDocBegin);
case YAML_DOCUMENT_END_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeDocEnd);
case YAML_ALIAS_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeAlias);
case YAML_SCALAR_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeScalar);
case YAML_SEQUENCE_START_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeSeqBegin);
case YAML_SEQUENCE_END_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeSeqEnd);
case YAML_MAPPING_START_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeMapBegin);
case YAML_MAPPING_END_EVENT:
FUNCTION_TEST_RETURN(yamlEventTypeMapEnd);
default:
CHECK(type == YAML_NO_EVENT);
FUNCTION_TEST_RETURN(yamlEventTypeNone);
}
}
YamlEvent
yamlEventNext(Yaml *this)
{
FUNCTION_TEST_BEGIN();
FUNCTION_TEST_PARAM(YAML, this);
FUNCTION_TEST_END();
ASSERT(this != NULL);
yaml_event_t event;
if (!yaml_parser_parse(&this->parser, &event))
{
// These should always be set
CHECK(this->parser.problem_mark.line && this->parser.problem_mark.column);
THROW_FMT(
FormatError, "yaml parse error: %s at line: %lu column: %lu", this->parser.problem,
(unsigned long)this->parser.problem_mark.line + 1, (unsigned long)this->parser.problem_mark.column + 1);
}
YamlEvent result =
{
.type = yamlEventType(event.type),
.line = event.start_mark.line + 1,
.column = event.start_mark.column + 1,
};
if (result.type == yamlEventTypeScalar)
result.value = strNewZ((const char *)event.data.scalar.value);
yaml_event_delete(&event);
FUNCTION_TEST_RETURN(result);
}
/**********************************************************************************************************************************/
YamlEvent
yamlEventNextCheck(Yaml *this, YamlEventType type)
{
FUNCTION_TEST_BEGIN();
FUNCTION_TEST_PARAM(YAML, this);
FUNCTION_TEST_PARAM(STRING_ID, type);
FUNCTION_TEST_END();
YamlEvent result = yamlEventNext(this);
yamlEventCheck(result, type);
FUNCTION_TEST_RETURN(result);
}
/**********************************************************************************************************************************/
void
yamlEventCheck(YamlEvent event, YamlEventType type)
{
FUNCTION_TEST_BEGIN();
FUNCTION_TEST_PARAM(YAML_EVENT, event);
FUNCTION_TEST_PARAM(STRING_ID, type);
FUNCTION_TEST_END();
if (event.type != type)
{
THROW_FMT(
FormatError, "expected event type '%s' but got '%s' at line %zu, column %zu", strZ(strIdToStr(type)),
strZ(strIdToStr(event.type)), event.line, event.column);
}
FUNCTION_TEST_RETURN_VOID();
}
/**********************************************************************************************************************************/
bool
yamlBoolParse(YamlEvent event)
{
FUNCTION_TEST_BEGIN();
FUNCTION_TEST_PARAM(YAML_EVENT, event);
FUNCTION_TEST_END();
if (strEq(event.value, FALSE_STR))
FUNCTION_TEST_RETURN(false);
else if (strEq(event.value, TRUE_STR))
FUNCTION_TEST_RETURN(true);
THROW_FMT(FormatError, "invalid boolean '%s'", strZ(event.value));
}