Как правильно вызывать jsonnet с импортом из Python


person ceridwen    schedule 20.04.2018    source источник


Ответы (2)


Импортер по умолчанию использует пути относительно файла, из которого они импортируются. В случае evaluate_snippet вам нужно передать путь вручную. Таким образом, jsonnet знает, где искать импортированные файлы.

Если вы намерены обрабатывать файлы, вы можете использовать пользовательский импортер. (Отступление: jsonnet пытается избежать предварительной обработки исходных файлов, поэтому, вероятно, есть лучший способ или отсутствующая функция в jsonnet.)

Ниже приведен полный рабочий пример того, как использовать пользовательские импортеры в Python (с поправкой на предоставленную структуру каталогов):

import os
import unittest

import _jsonnet


#  Returns content if worked, None if file not found, or throws an exception
def try_path(dir, rel):
    if not rel:
        raise RuntimeError('Got invalid filename (empty string).')
    if rel[0] == '/':
        full_path = rel
    else:
        full_path = dir + rel
    if full_path[-1] == '/':
        raise RuntimeError('Attempted to import a directory')

    if not os.path.isfile(full_path):
        return full_path, None
    with open(full_path) as f:
        return full_path, f.read()


def import_callback(dir, rel):
    full_path, content = try_path(dir, rel)
    if content:
        return full_path, content
    raise RuntimeError('File not found')


class JsonnetTests(unittest.TestCase):
    def setUp(self):
        self.input_filename = os.path.join(
            "jsonnet",
            "bar.jsonnet",
        )
        self.expected_str = '{\n   "num": 42,\n   "str": "The answer to life ..."\n}\n'
        with open(self.input_filename, "r") as infile:
            self.input_snippet = infile.read()

    def test_evaluate_file(self):
        json_str = _jsonnet.evaluate_file(
            self.input_filename,
            import_callback=import_callback,
        )
        self.assertEqual(json_str, self.expected_str)

    def test_evaluate_snippet(self):
        json_str = _jsonnet.evaluate_snippet(
            "jsonnet/bar.jsonnet",
            self.input_snippet,
            import_callback=import_callback,
        )
        self.assertEqual(json_str, self.expected_str)

if __name__ == '__main__':
    unittest.main()

Примечание. Это модифицированная версия примера из репозитория jsonnet.

person sbarzowski    schedule 21.04.2018

Я не совсем понимаю, почему вы должны использовать evaluate_snippet() (может быть, замаскировать фактические имена файлов, загрузив их из python в строки + evaluate_snippet("blah", str) ?), вместо evaluate_file() - в любом случае эта структура должна работать нормально.

Пример:

jsonnet_test.py:

import json:
import _jsonnet

jsonnet_file = "jsonnet/bar.jsonnet"
data = json.loads(_jsonnet.evaluate_file(jsonnet_file))
print("{str} => {num}".format(**data))

jsonnet/bar.jsonnet:

local baz = import "baz.libsonnet";
{
  str: "The answer to life ...",
  num: baz.mult(6, 7),
}

jsonnet/baz.libsonnet:

{
  mult(a, b):: (
    a * b
  ),
}

Выход:

$ python jsonnet_test.py
The answer to life ... => 42
person jjo    schedule 20.04.2018
comment
Мне нужно изменить код jsonnet, прежде чем передать его в _jsonnet. - person ceridwen; 21.04.2018