Это ваш пример неверен является только, потому что вы выбрали зарезервированный символ, чтобы начать свой скаляры с. Если заменить *
с каким-либо другим незарезервированный характера (я предпочитаю использовать не-ASCII символы для этого, поскольку они редко используются как часть некоторой спецификации), вы в конечном итоге с совершенно законного YAML:
paths:
root: /path/to/root/
patha: ♦root♦ + a
pathb: ♦root♦ + b
pathc: ♦root♦ + c
Это будет загружаться в стандартное представление для сопоставлений на языке, который использует ваш парсер, и не волшебным образом расширяет что-либо.
Для этого используется тип локально по умолчанию объекта как в следующей программе Python:
# coding: utf-8
from __future__ import print_function
import ruamel.yaml as yaml
class Paths:
def __init__(self):
self.d = {}
def __repr__(self):
return repr(self.d).replace('ordereddict', 'Paths')
@staticmethod
def __yaml_in__(loader, data):
result = Paths()
loader.construct_mapping(data, result.d)
return result
@staticmethod
def __yaml_out__(dumper, self):
return dumper.represent_mapping('!Paths', self.d)
def __getitem__(self, key):
res = self.d[key]
return self.expand(res)
def expand(self, res):
try:
before, rest = res.split(u'♦', 1)
kw, rest = rest.split(u'♦ +', 1)
rest = rest.lstrip() # strip any spaces after "+"
# the lookup will throw the correct keyerror if kw is not found
# recursive call expand() on the tail if there are multiple
# parts to replace
return before + self.d[kw] + self.expand(rest)
except ValueError:
return res
yaml_str = """\
paths: !Paths
root: /path/to/root/
patha: ♦root♦ + a
pathb: ♦root♦ + b
pathc: ♦root♦ + c
"""
loader = yaml.RoundTripLoader
loader.add_constructor('!Paths', Paths.__yaml_in__)
paths = yaml.load(yaml_str, Loader=yaml.RoundTripLoader)['paths']
for k in ['root', 'pathc']:
print(u'{} -> {}'.format(k, paths[k]))
, который будет печатать:
root -> /path/to/root/
pathc -> /path/to/root/c
расширяющейся делается на лету и обрабатывает вложенные определения, но должны быть осторожны, чтобы не вызвать бесконечную рекурсию.
Указав самосвал, вы можете сбросить первоначальную YAML из данных, загруженных в, из-за расширения на лету:
dumper = yaml.RoundTripDumper
dumper.add_representer(Paths, Paths.__yaml_out__)
print(yaml.dump(paths, Dumper=dumper, allow_unicode=True))
это изменит порядок ключей отображения. Если это проблема, вы должны сделать self.d
CommentedMap
(импорт из ruamel.yaml.comments.py
)
** См. Также: ** https://stackoverflow.com/a/41620747/42223 – dreftymac