blob: 8121a7db9f965a70c2c39c9ddabb85b4751e40cc [file] [log] [blame] [edit]
/*
* plistutil.c
* Simple tool to convert a plist into different formats
*
* Copyright (c) 2009-2020 Martin Szulecki All Rights Reserved.
* Copyright (c) 2013-2020 Nikias Bassen, All Rights Reserved.
* Copyright (c) 2008 Zach C., All Rights Reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include "plist/plist.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <errno.h>
#include <unistd.h>
#ifdef _MSC_VER
#pragma warning(disable:4996)
#endif
typedef struct _options
{
char *in_file, *out_file;
uint8_t in_fmt, out_fmt; // fmts 0 = undef, 1 = bin, 2 = xml, 3 = json, 4 = openstep
uint8_t flags;
} options_t;
#define OPT_DEBUG (1 << 0)
#define OPT_COMPACT (1 << 1)
#define OPT_SORT (1 << 2)
static void print_usage(int argc, char *argv[])
{
char *name = NULL;
name = strrchr(argv[0], '/');
printf("Usage: %s [OPTIONS] [-i FILE] [-o FILE]\n", (name ? name + 1: argv[0]));
printf("\n");
printf("Convert a plist FILE between binary, XML, JSON, and OpenStep format.\n");
printf("If -f is omitted, XML plist data will be converted to binary and vice-versa.\n");
printf("To convert to/from JSON or OpenStep the output format needs to be specified.\n");
printf("\n");
printf("OPTIONS:\n");
printf(" -i, --infile FILE Optional FILE to convert from or stdin if - or not used\n");
printf(" -o, --outfile FILE Optional FILE to convert to or stdout if - or not used\n");
printf(" -f, --format FORMAT Force output format, regardless of input type\n");
printf(" FORMAT is one of xml, bin, json, or openstep\n");
printf(" If omitted, XML will be converted to binary,\n");
printf(" and binary to XML.\n");
printf(" -p, --print FILE Print the PList in human-readable format.\n");
printf(" -c, --compact JSON and OpenStep only: Print output in compact form.\n");
printf(" By default, the output will be pretty-printed.\n");
printf(" -s, --sort Sort all dictionary nodes lexicographically by key\n");
printf(" before converting to the output format.\n");
printf(" -d, --debug Enable extended debug output\n");
printf(" -v, --version Print version information\n");
printf("\n");
printf("Homepage: <" PACKAGE_URL ">\n");
printf("Bug Reports: <" PACKAGE_BUGREPORT ">\n");
}
static options_t *parse_arguments(int argc, char *argv[])
{
int i = 0;
options_t *options = (options_t*)calloc(1, sizeof(options_t));
options->out_fmt = 0;
for (i = 1; i < argc; i++)
{
if (!strcmp(argv[i], "--infile") || !strcmp(argv[i], "-i"))
{
if ((i + 1) == argc)
{
free(options);
return NULL;
}
options->in_file = argv[i + 1];
i++;
continue;
}
else if (!strcmp(argv[i], "--outfile") || !strcmp(argv[i], "-o"))
{
if ((i + 1) == argc)
{
free(options);
return NULL;
}
options->out_file = argv[i + 1];
i++;
continue;
}
else if (!strcmp(argv[i], "--format") || !strcmp(argv[i], "-f"))
{
if ((i + 1) == argc)
{
free(options);
return NULL;
}
if (!strncmp(argv[i+1], "bin", 3)) {
options->out_fmt = PLIST_FORMAT_BINARY;
} else if (!strncmp(argv[i+1], "xml", 3)) {
options->out_fmt = PLIST_FORMAT_XML;
} else if (!strncmp(argv[i+1], "json", 4)) {
options->out_fmt = PLIST_FORMAT_JSON;
} else if (!strncmp(argv[i+1], "openstep", 8) || !strncmp(argv[i+1], "ostep", 5)) {
options->out_fmt = PLIST_FORMAT_OSTEP;
} else {
fprintf(stderr, "ERROR: Unsupported output format\n");
free(options);
return NULL;
}
i++;
continue;
}
else if (!strcmp(argv[i], "--compact") || !strcmp(argv[i], "-c"))
{
options->flags |= OPT_COMPACT;
}
else if (!strcmp(argv[i], "--sort") || !strcmp(argv[i], "-s"))
{
options->flags |= OPT_SORT;
}
else if (!strcmp(argv[i], "--print") || !strcmp(argv[i], "-p"))
{
if ((i + 1) == argc)
{
free(options);
return NULL;
}
options->in_file = argv[i + 1];
options->out_fmt = PLIST_FORMAT_PRINT;
char *env_fmt = getenv("PLIST_OUTPUT_FORMAT");
if (env_fmt) {
if (!strcmp(env_fmt, "plutil")) {
options->out_fmt = PLIST_FORMAT_PLUTIL;
} else if (!strcmp(env_fmt, "limd")) {
options->out_fmt = PLIST_FORMAT_LIMD;
}
}
i++;
continue;
}
else if (!strcmp(argv[i], "--debug") || !strcmp(argv[i], "-d"))
{
options->flags |= OPT_DEBUG;
}
else if (!strcmp(argv[i], "--help") || !strcmp(argv[i], "-h"))
{
free(options);
return NULL;
}
else if (!strcmp(argv[i], "--version") || !strcmp(argv[i], "-v"))
{
printf("plistutil %s\n", PACKAGE_VERSION);
exit(EXIT_SUCCESS);
}
else
{
fprintf(stderr, "ERROR: Invalid option '%s'\n", argv[i]);
free(options);
return NULL;
}
}
return options;
}
int main(int argc, char *argv[])
{
int ret = 0;
int input_res = PLIST_ERR_UNKNOWN;
int output_res = PLIST_ERR_UNKNOWN;
FILE *iplist = NULL;
plist_t root_node = NULL;
char *plist_out = NULL;
uint32_t size = 0;
int read_size = 0;
int read_capacity = 4096;
char *plist_entire = NULL;
struct stat filestats;
options_t *options = parse_arguments(argc, argv);
if (!options)
{
print_usage(argc, argv);
return 0;
}
if (options->flags & OPT_DEBUG)
{
plist_set_debug(1);
}
if (!options->in_file || !strcmp(options->in_file, "-"))
{
read_size = 0;
plist_entire = malloc(sizeof(char) * read_capacity);
if(plist_entire == NULL)
{
fprintf(stderr, "ERROR: Failed to allocate buffer to read from stdin");
free(options);
return 1;
}
plist_entire[read_size] = '\0';
char ch;
while(read(STDIN_FILENO, &ch, 1) > 0)
{
if (read_size >= read_capacity) {
char *old = plist_entire;
read_capacity += 4096;
plist_entire = realloc(plist_entire, sizeof(char) * read_capacity);
if (plist_entire == NULL)
{
fprintf(stderr, "ERROR: Failed to reallocate stdin buffer\n");
free(old);
free(options);
return 1;
}
}
plist_entire[read_size] = ch;
read_size++;
}
if (read_size >= read_capacity) {
char *old = plist_entire;
plist_entire = realloc(plist_entire, sizeof(char) * (read_capacity+1));
if (plist_entire == NULL)
{
fprintf(stderr, "ERROR: Failed to reallocate stdin buffer\n");
free(old);
free(options);
return 1;
}
}
plist_entire[read_size] = '\0';
// Not positive we need this, but it doesnt seem to hurt lol
if(ferror(stdin))
{
fprintf(stderr, "ERROR: reading from stdin.\n");
free(plist_entire);
free(options);
return 1;
}
}
else
{
// read input file
iplist = fopen(options->in_file, "rb");
if (!iplist) {
fprintf(stderr, "ERROR: Could not open input file '%s': %s\n", options->in_file, strerror(errno));
free(options);
return 1;
}
memset(&filestats, '\0', sizeof(struct stat));
fstat(fileno(iplist), &filestats);
plist_entire = (char *) malloc(sizeof(char) * (filestats.st_size + 1));
read_size = fread(plist_entire, sizeof(char), filestats.st_size, iplist);
plist_entire[read_size] = '\0';
fclose(iplist);
}
if (options->out_fmt == 0) {
// convert from binary to xml or vice-versa
if (plist_is_binary(plist_entire, read_size))
{
input_res = plist_from_bin(plist_entire, read_size, &root_node);
if (input_res == PLIST_ERR_SUCCESS) {
if (options->flags & OPT_SORT) {
plist_sort(root_node);
}
output_res = plist_to_xml(root_node, &plist_out, &size);
}
}
else
{
input_res = plist_from_xml(plist_entire, read_size, &root_node);
if (input_res == PLIST_ERR_SUCCESS) {
if (options->flags & OPT_SORT) {
plist_sort(root_node);
}
output_res = plist_to_bin(root_node, &plist_out, &size);
}
}
}
else
{
input_res = plist_from_memory(plist_entire, read_size, &root_node, NULL);
if (input_res == PLIST_ERR_SUCCESS) {
if (options->flags & OPT_SORT) {
plist_sort(root_node);
}
if (options->out_fmt == PLIST_FORMAT_BINARY) {
output_res = plist_to_bin(root_node, &plist_out, &size);
} else if (options->out_fmt == PLIST_FORMAT_XML) {
output_res = plist_to_xml(root_node, &plist_out, &size);
} else if (options->out_fmt == PLIST_FORMAT_JSON) {
output_res = plist_to_json(root_node, &plist_out, &size, !(options->flags & OPT_COMPACT));
} else if (options->out_fmt == PLIST_FORMAT_OSTEP) {
output_res = plist_to_openstep(root_node, &plist_out, &size, !(options->flags & OPT_COMPACT));
} else {
plist_write_to_stream(root_node, stdout, options->out_fmt, PLIST_OPT_PARTIAL_DATA);
plist_free(root_node);
free(plist_entire);
free(options);
return 0;
}
}
}
plist_free(root_node);
free(plist_entire);
if (plist_out)
{
if (options->out_file != NULL && strcmp(options->out_file, "-") != 0)
{
FILE *oplist = fopen(options->out_file, "wb");
if (!oplist) {
fprintf(stderr, "ERROR: Could not open output file '%s': %s\n", options->out_file, strerror(errno));
free(options);
return 1;
}
fwrite(plist_out, size, sizeof(char), oplist);
fclose(oplist);
}
// if no output file specified, write to stdout
else
fwrite(plist_out, size, sizeof(char), stdout);
free(plist_out);
}
if (input_res == PLIST_ERR_SUCCESS) {
switch (output_res) {
case PLIST_ERR_SUCCESS:
break;
case PLIST_ERR_FORMAT:
fprintf(stderr, "ERROR: Input plist data is not compatible with output format.\n");
ret = 2;
break;
default:
fprintf(stderr, "ERROR: Failed to convert plist data (%d)\n", output_res);
ret = 1;
}
} else {
switch (input_res) {
case PLIST_ERR_PARSE:
if (options->out_fmt == 0) {
fprintf(stderr, "ERROR: Could not parse plist data, expected XML or binary plist\n");
} else {
fprintf(stderr, "ERROR: Could not parse plist data (%d)\n", input_res);
}
ret = 3;
break;
default:
fprintf(stderr, "ERROR: Could not parse plist data (%d)\n", input_res);
ret = 1;
break;
}
}
free(options);
return ret;
}