Yson
- Примеры
- Yson::Parse...
- Yson::From
- Yson::WithAttributes
- Yson::Equals
- Yson::GetHash
- Yson::Is...
- Yson::GetLength
- Yson::ConvertTo...
- Yson::Contains
- Yson::Lookup...
- Yson::YPath
- Yson::Attributes
- Yson::Serialize...
- Yson::SerializeJson
- Yson::Options
- Yson::Iterate
- Yson::As... и Yson::TryAs...
- In-place изменения Yson узлов
- Yson::MutCreate
- Yson::Mutate
- Yson::MutFreeze
- Yson::MutUpsert
- Yson::MutInsert
- Yson::MutUpdate
- Yson::MutRemove
- Yson::MutRewind
- Yson::MutUp
- Yson::MutDown, Yson::MutDownOrCreate, Yson::MutTryDown
- Yson::MutExists
- Yson::MutView
- Смотрите также
YSON — разработанный в Яндексе формат данных, похожий на JSON.
-
Сходства с JSON:
- не имеет строгой схемы;
- помимо простых типов данных поддерживает словари и списки в произвольных комбинациях.
-
Некоторые отличия от JSON:
- Помимо текстового представления имеет и бинарное;
- В текстовом представлении вместо запятых — точки с запятой, а вместо двоеточий — равно;
-
Поддерживается концепция «атрибутов», то есть именованных свойств, которые могут быть присвоены узлу в дереве.
Особенности реализации и функциональность модуля:
-
Наравне с YSON данный модуль поддерживает и стандартный JSON, что несколько расширяет область его применения.
-
Работает с DOM представлением YSON в памяти, которое в терминах YQL передается между функциями как «ресурс» (см. описание специальных типов данных). Большинство функций модуля имеют семантику запроса на выполнение указанной операции с ресурсом и возвращают пустой optional, если операция не удалась из-за несоответствия фактического типа данных ожидаемому.
-
Предоставляет несколько основных классов функций (полный список и подробное описание функций см. ниже):
Yson::Parse***— получение ресурса с DOM-объектом из сериализованных данных, все дальнейшие операции выполняются уже над полученным ресурсом;Yson::From— получение ресурса с DOM-объектом из простых типов данных YQL или контейнеров (списков или словарей);Yson::ConvertTo***— преобразовать ресурс к простым типам данных или контейнерам;Yson::Lookup***— получение одного элемента списка или словаря с опциональным преобразованием в нужный тип данных;Yson::YPath***— получение одного элемента дерева документа по указанному относительному пути с опциональным преобразованием в нужный тип данных;Yson::Serialize***— получить из ресурса копию его данных, сериализованную в одном из форматов;
-
Для удобства при передаче сериализованного Yson и Json в функции, ожидающие на входе ресурс с DOM-объектом, неявное преобразование через
Yson::ParseилиYson::ParseJsonпроисходит автоматически. Также в SQL синтаксисе оператор точки или квадратных скобок автоматически добавляет вызовYson::Lookup. Для сериализации ресурса по-прежнему нужно вызыватьYson::ConvertTo***илиYson::Serialize***. Таким образом, например, получение элемента "foo" словаря из колонки mycolumn типа Yson в виде строки может выглядеть так:SELECT Yson::ConvertToString(mycolumn["foo"]) FROM mytable;илиSELECT Yson::ConvertToString(mycolumn.foo) FROM mytable;. В варианте с точкой можно экранировать спецсимволы по общим правилам для индентификаторов.
Функции модуля стоит рассматривать как «кубики», из которых можно собирать разные конструкции, например:
Yson::Parse*** -> Yson::Serialize***— конвертация из одного формата в другой;Yson::Parse*** -> Yson::Lookup -> Yson::Serialize***— извлечение значения указанного поддерева в исходном дереве YSON;Yson::Parse*** -> Yson::ConvertToList -> ListMap -> Yson::Lookup***— извлечение элементов по ключу из YSON списка.
См. также примеры комбинирования YSON-функций в tutorial.
Примеры
$node = Json(@@
{"abc": {"def": 123, "ghi": "привет"}}
@@);
SELECT Yson::SerializeText($node.abc) AS `yson`;
-- {"def"=123;"ghi"="\xD0\xBF\xD1\x80\xD0\xB8\xD0\xB2\xD0\xB5\xD1\x82"}
$node = Yson(@@
<a=z;x=y>[
{abc=123; def=456};
{abc=234; xyz=789};
]
@@);
$attrs = Yson::YPath($node, "/@");
SELECT
ListMap(Yson::ConvertToList($node), ($x) -> { return Yson::LookupInt64($x, "abc") }) AS abcs,
Yson::ConvertToStringDict($attrs) AS attrs,
Yson::SerializePretty(Yson::Lookup($node, "7", Yson::Options(false AS Strict))) AS miss;
/*
- abcs: `[123; 234]`
- attrs: `{"a"="z";"x"="y"}`
- miss: `NULL`
*/
Yson::Parse...
Yson::Parse(Yson{Flags:AutoMap}) -> Resource<'Yson2.Node'>
Yson::ParseJson(Json{Flags:AutoMap}) -> Resource<'Yson2.Node'>
Yson::ParseJsonDecodeUtf8(Json{Flags:AutoMap}) -> Resource<'Yson2.Node'>
Yson::Parse(String{Flags:AutoMap}) -> Resource<'Yson2.Node'>? -- принимает YSON в любом формате
Yson::ParseJson(String{Flags:AutoMap}) -> Resource<'Yson2.Node'>?
Yson::ParseJsonDecodeUtf8(String{Flags:AutoMap}) -> Resource<'Yson2.Node'>?
Результат всех трёх функций является несериализуемым: его можно только передать на вход другой функции из библиотеки Yson, но нельзя сохранить в таблицу или вернуть на клиент в результате операции — попытка так сделать приведет к ошибке типизации. Также запрещено возвращать его за пределы подзапросов: если это требуется, то надо вызвать Yson::Serialize, а оптимизатор уберёт лишнюю сериализию и десериализацию, если материализация в конечном счёте не потребуется.
Примечание
Функция Yson::ParseJsonDecodeUtf8 ожидает, что символы, выходящие за пределы ASCII, должны быть дополнительно заэкранированы.
Yson::From
Yson::From(T) -> Resource<'Yson2.Node'>
Yson::From является полиморфной функцией, преобразующей в Yson ресурс большинство примитивных типов данных и контейнеров (списки, словари, кортежи, структуры и т.п.). Тип исходного объекта должен быть совместим с Yson. Например, в ключах словарей допустимы только типы String или Utf8, а вот String? или Utf8? уже нет.
Пример
SELECT Yson::Serialize(Yson::From(TableRow())) FROM table1;
Yson::WithAttributes
Yson::WithAttributes(Resource<'Yson2.Node'>{Flags:AutoMap}, Resource<'Yson2.Node'>{Flags:AutoMap}) -> Resource<'Yson2.Node'>?
Добавляет к узлу Yson (первый аргумент) атрибуты (второй аргумент). Атрибуты должны представлять из себя узел map.
Yson::Equals
Yson::Equals(Resource<'Yson2.Node'>{Flags:AutoMap}, Resource<'Yson2.Node'>{Flags:AutoMap}) -> Bool
Проверка деревьев в памяти на равенство, толерантная к исходному формату сериализации и порядку перечисления ключей в словарях.
Yson::GetHash
Yson::GetHash(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Uint64
Вычисление 64-битного хэша от дерева объектов.
Yson::Is...
Yson::IsEntity(Resource<'Yson2.Node'>{Flags:AutoMap}) -> bool
Yson::IsString(Resource<'Yson2.Node'>{Flags:AutoMap}) -> bool
Yson::IsDouble(Resource<'Yson2.Node'>{Flags:AutoMap}) -> bool
Yson::IsUint64(Resource<'Yson2.Node'>{Flags:AutoMap}) -> bool
Yson::IsInt64(Resource<'Yson2.Node'>{Flags:AutoMap}) -> bool
Yson::IsBool(Resource<'Yson2.Node'>{Flags:AutoMap}) -> bool
Yson::IsList(Resource<'Yson2.Node'>{Flags:AutoMap}) -> bool
Yson::IsDict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> bool
Проверка, что текущий узел имеет соответствующий тип. Entity это #.
Yson::GetLength
Yson::GetLength(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Uint64?
Получение числа элементов в списке или словаре.
Yson::ConvertTo...
Yson::ConvertTo(Resource<'Yson2.Node'>{Flags:AutoMap}, Type<T>) -> T
Yson::ConvertToBool(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Bool?
Yson::ConvertToInt64(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Int64?
Yson::ConvertToUint64(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Uint64?
Yson::ConvertToDouble(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Double?
Yson::ConvertToString(Resource<'Yson2.Node'>{Flags:AutoMap}) -> String?
Yson::ConvertToList(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<Resource<'Yson2.Node'>>
Yson::ConvertToBoolList(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<Bool>
Yson::ConvertToInt64List(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<Int64>
Yson::ConvertToUint64List(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<Uint64>
Yson::ConvertToDoubleList(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<Double>
Yson::ConvertToStringList(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<String>
Yson::ConvertToDict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String,Resource<'Yson2.Node'>>
Yson::ConvertToBoolDict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String,Bool>
Yson::ConvertToInt64Dict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String,Int64>
Yson::ConvertToUint64Dict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String,Uint64>
Yson::ConvertToDoubleDict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String,Double>
Yson::ConvertToStringDict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String,String>
Важно
Данные функции по умолчанию не делают неявного приведения типов, то есть значение в аргументе должно в точности соответствовать вызываемой функции.
Yson::ConvertTo является полиморфной функцией, преобразующей Yson ресурс в указанный во втором аргументе тип данных с поддержкой вложенных контейнеров (списки, словари, кортежи, структуры и т.п.).
Пример
$data = Yson(@@{
"name" = "Anya";
"age" = 15u;
"params" = {
"ip" = "95.106.17.32";
"last_time_on_site" = 0.5;
"region" = 213;
"user_agent" = "Mozilla/5.0"
}
}@@);
SELECT Yson::ConvertTo($data,
Struct<
name: String,
age: Uint32,
params: Dict<String,Yson>
>
);
Yson::Contains
Yson::Contains(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Bool?
Проверяет наличие ключа в словаре. Если тип объекта map, то ищем среди ключей.
Если тип объекта список, то ключ должен быть десятичным числом - индексом в списке.
Yson::Lookup...
Yson::Lookup(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Resource<'Yson2.Node'>?
Yson::LookupBool(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Bool?
Yson::LookupInt64(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Int64?
Yson::LookupUint64(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Uint64?
Yson::LookupDouble(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Double?
Yson::LookupString(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> String?
Yson::LookupDict(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Dict<String,Resource<'Yson2.Node'>>?
Yson::LookupList(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> List<Resource<'Yson2.Node'>>?
Перечисленные выше функции представляют собой краткую форму записи для типичного сценария использования: Yson::YPath — переход в словарь на один уровень с последующим извлечением значения — Yson::ConvertTo***. Второй аргумент для всех перечисленных функций — имя ключа в словаре (в отличие от YPath, без префикса /) или индекс в списке (например, 7). Упрощают запрос и дают небольшой выигрыш в скорости работы.
Yson::YPath
Yson::YPath(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Resource<'Yson2.Node'>?
Yson::YPathBool(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Bool?
Yson::YPathInt64(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Int64?
Yson::YPathUint64(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Uint64?
Yson::YPathDouble(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Double?
Yson::YPathString(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> String?
Yson::YPathDict(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> Dict<String,Resource<'Yson2.Node'>>?
Yson::YPathList(Resource<'Yson2.Node'>{Flags:AutoMap}, String) -> List<Resource<'Yson2.Node'>>?
Позволяет по входному ресурсу и пути на языке YPath получить ресурс, указывающий на соответствующую пути часть исходного ресурса.
Информация по YPath на странице документации YT.
Yson::Attributes
Yson::Attributes(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String,Resource<'Yson2.Node'>>
Получение всех атрибутов узла в виде словаря.
Yson::Serialize...
Yson::Serialize(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Yson -- бинарное представление
Yson::SerializeText(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Yson
Yson::SerializePretty(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Yson -- чтобы увидеть именно текстовый результат, можно обернуть его в ToBytes(...)
Yson::SerializeJson
Yson::SerializeJson(Resource<'Yson2.Node'>{Flags:AutoMap}, [Resource<'Yson2.Options'>?, SkipMapEntity:Bool?, EncodeUtf8:Bool?, WriteNanAsString:Bool?]) -> Json?
SkipMapEntityотвечает за сериализацию значений в словарях, имеющих значение#. На значение атрибутов флаг не влияет. По умолчаниюfalse.EncodeUtf8отвечает за экранирование символов, выходящих за пределы ASCII. По умолчаниюfalse.WriteNanAsStringразрешает сериализацию значенийNaNиInfв json в виде строк. По умолчаниюfalse.
Типы данных Yson и Json, возвращаемые функциями сериализации, представляет собой частный случай строки, про которую известно, что в ней находятся данные в соответствующем формате (Yson/Json).
Yson::Options
Yson::Options([AutoConvert:Bool?, Strict:Bool?]) -> Resource<'Yson2.Options'>
Передаётся последним опциональным аргументом (который для краткости не указан) в методы Parse..., ConvertTo..., Contains, Lookup... и YPath..., которые принимают результат вызова Yson::Options. По умолчанию все поля Yson::Options выключены (false), а при включении (true) модифицируют поведение следующим образом:
- AutoConvert — если переданное в Yson значение не в точности соответствует типу данных результата, то значение будет по возможности сконвертировано. Например,
Yson::ConvertToInt64в этом режиме будет делать Int64 даже из чисел типа Double. - Strict — по умолчанию все функции из библиотеки Yson возвращают ошибку в случае проблем в ходе выполнения запроса (например, попытка парсинга строки не являющейся Yson/Json, или попытка поиска по ключу в скалярном типе, или запрошено преобразование в несовместимый тип данных, и т.п.), а если отключить строгий режим, то вместо ошибки в большинстве случаев будет возвращаться
NULL. При преобразовании в словарь или список (ConvertTo<Type>DictилиConvertTo<Type>List) плохие элементы будут выброшены из полученной коллекции.
Пример
$yson = @@{y = true; x = 5.5}@@y;
SELECT Yson::LookupBool($yson, "z"); --- null
SELECT Yson::LookupBool($yson, "y"); --- true
-- SELECT Yson::LookupInt64($yson, "x"); --- Ошибка
SELECT Yson::LookupInt64($yson, "x", Yson::Options(false as Strict)); --- null
SELECT Yson::LookupInt64($yson, "x", Yson::Options(true as AutoConvert)); --- 5
-- SELECT Yson::ConvertToBoolDict($yson); --- Ошибка
SELECT Yson::ConvertToBoolDict($yson, Yson::Options(false as Strict)); --- { "y": true }
SELECT Yson::ConvertToDoubleDict($yson, Yson::Options(false as Strict)); --- { "x": 5.5 }
Если во всём запросе требуется применять одинаковые значения настроек библиотеки Yson, то удобнее воспользоваться PRAGMA yson.AutoConvert; и/или PRAGMA yson.Strict;. Также эти PRAGMA являются единственным способом повлиять на неявные вызовы библиотеки Yson, которые возникают при работе с типами данных Yson/Json.
Yson::Iterate
Yson::Iterate(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<Variant<
'BeginAttributes':Void,
'BeginList':Void,
'BeginMap':Void,
'EndAttributes':Void,
'EndList':Void,
'EndMap':Void,
'Item':Void,
'Key':String,
'PostValue':Resource<'Yson2.Node'>,
'PreValue':Resource<'Yson2.Node'>,
'Value':Resource<'Yson2.Node'>>>
Доступна начиная с версии 2025.05.
Получение списка всех событий при обходе Yson-дерева.
Листовые узлы (Entity, Bool, Int64, Uint64, Double, String) передаются в виде события Value.
Для узла с типом List выдается такая последовательность:
PreValueс самим узломBeginListItem- перед каждым элементомList- события для элемента
List EndListPostValueс самим узлом
Для узла с типом Map выдается такая последовательность:
PreValueс самим узломBeginMapKey- перед каждым элементомMap- события для элемента
Map EndMapPostValueс самим узлом
Порядок выдачи ключей может быть произвольным.
Для узла с непустыми атрибутами выдается такая последовательность:
PreValueс самим узломBeginAttributesKey- перед каждым именем атрибута- события для атрибута
EndAttributes- события для узла без атрибутов
PostValueс самим узлом
Порядок выдачи атрибутов может быть произвольным.
Примеры
-- Просмотр всей выдачи функции Yson::Iterate
$dump = ($x) -> (
(
Way($x),
$x.Key,
Yson::Serialize($x.PreValue),
Yson::Serialize($x.Value),
Yson::Serialize($x.PostValue)
)
);
SELECT ListMap(Yson::Iterate('{a=1;b=<c="foo">[2u;%true;#;-3.2]}'y), $dump);
/*
События:
PreValue [1]
BeginMap
Key a
Value 1
Key b
PreValue [2]
BeginAttributes
Key c
Value foo
EndAttributes
PreValue [3]
BeginList
Item
Value 2
Item
Value %true
Item
Value #
Item
Value -3.2
EndList
PostValue [3]
PostValue [2]
EndMap
PostValue [1]
*/
-- Получение всех листовых значений - раскрытие всех списков
$yson = '[[1;2];[3;4]]'y;
SELECT ListFlatMap(Yson::Iterate($yson), ($x)->(IF($x.Value IS NOT NULL, $x.Value))); -- [1;2;3;4]
-- Поиск ключа с заданным именем на любом уровне
$yson = '{a={b={c=1}};e={f=2}}'y;
SELECT ListHasItems(ListFilter(Yson::Iterate($yson), ($x)->($x.Key == 'b'))); -- true
-- Поиск строки в значениях любом уровне
$yson = '{a={b={c="x"}};e={f="y"}}'y;
SELECT ListHasItems(ListFilter(Yson::Iterate($yson), ($x)->(Yson::ConvertToString($x.Value) == 'y'))); -- true
-- Получение атрибутов name для всех Map узлов без атрибута children
$yson = @@{
name=foo;
children=[
{
name=bar
}
]
}@@y;
SELECT ListFlatMap(Yson::Iterate($yson), ($x)->(
IF(Yson::IsDict($x.PreValue) and not Yson::Contains($x.PreValue,'children'), Yson::LookupString($x.PreValue, 'name')))); -- [bar]
Yson::As... и Yson::TryAs...
Yson::AsString(Resource<'Yson2.Node'>{Flags:AutoMap}) -> String
Yson::AsDouble(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Double
Yson::AsUint64(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Uint64
Yson::AsInt64(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Int64
Yson::AsBool(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Bool
Yson::AsList(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<Resource<'Yson2.Node'>>
Yson::AsDict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String, Resource<'Yson2.Node'>>
Yson::TryAsString(Resource<'Yson2.Node'>{Flags:AutoMap}) -> String?
Yson::TryAsDouble(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Double?
Yson::TryAsUint64(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Uint64?
Yson::TryAsInt64(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Int64?
Yson::TryAsBool(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Bool?
Yson::TryAsList(Resource<'Yson2.Node'>{Flags:AutoMap}) -> List<Resource<'Yson2.Node'>>?
Yson::TryAsDict(Resource<'Yson2.Node'>{Flags:AutoMap}) -> Dict<String, Resource<'Yson2.Node'>>?
Доступны начиная с версии 2025.05.
Приводят Yson узел к заданному типу.
TryAs* функции при неверном типе Yson узла возвращают NULL, а As* функции в этом случае приведут к ошибке запроса.
Для обработки узла с типом Entity ('#') следует использовать функцию IsEntity.
In-place изменения Yson узлов
Начиная с версии 2025.05 появилась возможность работать с изменяемым Yson деревом на базе linear типов. Для создания нового дерева используется функция MutCreate, а для импорта существующего неизменяемого дерева в изменяемую форму - Mutate.
Само построение или трансформацию дерева следует выполнять внутри функции Block, при этом изменяемое Yson дерево в конечном счете должно быть преобразовано в неизменяемое с помощью функции MutFreeze.
В каждом экземпляре изменяемого Yson дерева есть единственный текущий узел, который можно перемещать, попутно создавая новые узлы при необходимости.
Yson::MutCreate
Yson::MutCreate() -> Linear<Resource<Yson2.MutNode>>
Доступна начиная с версии 2025.05.
Создает новое пустое дерево с единственным узлом в Invalid состоянии - с неопределенным типом (значение этого узла можно поменять, например, через функцию MutUpsert). При этом текущий узел выставляется на корень дерева.
Т.к. эта функция создает экземпляр linear типа, то она должна использоваться с указанием зависимых узлов через функцию Udf.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::MutCreate, $arg as Depends)();
return Yson::MutFreeze($m); -- Ошибка: Invalid or deleted node is not allowed
})
SELECT Block(($arg)->{
$m = Udf(Yson::MutCreate, $arg as Depends)();
$m = Yson::MutUpsert($m, '1'y);
return Yson::MutFreeze($m); -- 1
});
Yson::Mutate
Yson::Mutate(Resource<Yson2.Node>) -> Linear<Resource<Yson2.MutNode>>
Доступна начиная с версии 2025.05.
Создает изменяемое Yson дерево из неизменяемого. При этом текущий узел выставляется на корень дерева.
Т.к. эта функция создает экземпляр linear типа, то она должна использоваться с указанием зависимых узлов через функцию Udf.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('1'y);
return Yson::MutFreeze($m); -- 1
});
Yson::MutFreeze
Yson::MutFreeze(Linear<Resource<Yson2.MutNode>>) -> Resource<Yson2.Node>
Доступна начиная с версии 2025.05.
Преобразует изменяемое Yson дерево в неизменяемое, при этом удаляются все узлы из словарей и списков в состоянии Deleted.
Корневой узел в изменяемом дереве не должен быть в состоянии Invalid или Deleted, а дочерние узлы не должны быть в состоянии Invalid, иначе преобразование завершится ошибкой.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('1'y);
return Yson::MutFreeze($m); -- 1
});
SELECT Block(($arg)->{
$m = Udf(Yson::MutCreate, $arg as Depends)(); -- Корневой узел в состоянии Invalid
return Yson::MutFreeze($m); -- Ошибка: Invalid or deleted node is not allowed
})
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('1'y);
$m = Yson::MutRemove($m); -- Переводим узел в состояние Deleted
return Yson::MutFreeze($m); -- Ошибка: Invalid or deleted node is not allowed
})
Yson::MutUpsert
Yson::MutUpsert(Linear<Resource<Yson2.MutNode>>, Resource<Yson2.MutNode>) -> Linear<Resource<Yson2.MutNode>>
Доступна начиная с версии 2025.05.
Заменяет текущий узел в изменяемом Yson дереве на заданное Yson поддерево.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('1'y);
$m = Yson::MutUpsert($m, '2'y);
return Yson::MutFreeze($m); -- 2
});
Yson::MutInsert
Yson::MutInsert(Linear<Resource<Yson2.MutNode>>, Resource<Yson2.MutNode>) -> Linear<Resource<Yson2.MutNode>>
Доступна начиная с версии 2025.05.
Заменяет текущий узел в изменяемом Yson дереве на заданное Yson поддерево если текущий узел находится в Invalid или Deleted состоянии.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::MutCreate, $arg as Depends)();
$m = Yson::MutInsert($m, '1'y);
$m = Yson::MutInsert($m, '2'y); -- не имеет эффекта
return Yson::MutFreeze($m); -- 1
});
Yson::MutUpdate
Yson::MutUpdate(Linear<Resource<Yson2.MutNode>>, Resource<Yson2.MutNode>) -> Linear<Resource<Yson2.MutNode>>
Доступна начиная с версии 2025.05.
Заменяет текущий узел в изменяемом Yson дереве на заданное Yson поддерево если текущий узел не находится в Invalid или Deleted состоянии.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::MutCreate, $arg as Depends)();
$m = Yson::MutUpdate($m, '1'y); -- не имеет эффекта
$m = Yson::MutInsert($m, '2'y);
$m = Yson::MutUpdate($m, '3'y);
return Yson::MutFreeze($m); -- 3
});
Yson::MutRemove
Yson::MutRemove(Linear<Resource<Yson2.MutNode>>) -> Linear<Resource<Yson2.MutNode>>
Доступна начиная с версии 2025.05.
Переводит текущий узел в Deleted состояние.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('1'y);
$m = Yson::MutRemove($m);
$m = Yson::MutInsert($m, '2'y);
return Yson::MutFreeze($m); -- 2
});
Yson::MutRewind
Yson::MutRewind(Linear<Resource<Yson2.MutNode>>) -> Linear<Resource<Yson2.MutNode>>
Доступна начиная с версии 2025.05.
Передвигает текущий узел в корень дерева.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('{a=1;b=2}'y);
$m = Yson::MutDown($m, 'a'); -- переместились к ключу 'a' в словаре
$m = Yson::MutRewind($m); -- переместились в корень
$m = Yson::MutUpsert($m, '1'y); -- заменили дерево в корне
return Yson::MutFreeze($m); -- 1
});
Yson::MutUp
Yson::MutUp(Linear<Resource<Yson2.MutNode>>) -> Linear<Resource<Yson2.MutNode>>
Доступна начиная с версии 2025.05.
Передвигает текущий узел на уровень ближе к корню дерева. Если текущий узел уже находится в корне дерева, то генерируется ошибка.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('{a=1;b=2}'y);
$m = Yson::MutDown($m, 'a'); -- переместились к ключу 'a' в словаре
$m = Yson::MutUp($m); -- переместились в корень
$m = Yson::MutUpsert($m, '1'y); -- заменили дерево в корне
return Yson::MutFreeze($m); -- 1
});
Yson::MutDown, Yson::MutDownOrCreate, Yson::MutTryDown
Yson::MutDown(Linear<Resource<Yson2.MutNode>>,location:String) -> Linear<Resource<Yson2.MutNode>>
Yson::MutDownOrCreate(Linear<Resource<Yson2.MutNode>>,location:String) -> Linear<Resource<Yson2.MutNode>>
Yson::MutTryDown(Linear<Resource<Yson2.MutNode>>,location:String) -> Tuple<Linear<Resource<Yson2.MutNode>>, Bool>
Доступна начиная с версии 2025.05.
Передвигает текущий узел к элементу словаря или списка, при этом может быть создан новый узел в состоянии Invalid.
В строке location используется экранирование символов через обратный слеш \.
Если строка location начинается с символов <, = или >, то за ним должен быть указан индекс списка в формате десятичного числа (начиная с 0), first (0-й элемент списка) или last (последний элемент списка). Иначе location интерпретируется как ключ в словаре.
Для пустого списка first и last описывают положение за концом списка.
Для функции MutDown запрещается создавать новые ключи в словарях, в ней можно использовать только модификатор = для списков и она генерирует ошибку если перемещение текущего узла невозможно.
Функция MutTryDown в отличии от MutDown возвращает false во втором элементе кортежа в случае невозможности перемещения.
Если текущий узел имеет состояние Invalid или Deleted, то функция MutDownOrCreate сначала создает его в виде пустого списка или словаря, в зависимости от вида location.
Если тип текущего узла (список или словарь) не совпадает с видом location, то MutDown и MutDownOrCreate выдадут ошибку.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('{a=1;b=2}'y);
$m = Yson::MutDown($m, 'a'); -- переместились к ключу 'a' в словаре
$m = Yson::MutUpsert($m, '3'y); -- заменили текущий узел
return Yson::MutFreeze($m); -- {a=3;b=2}
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('{a=1;b=2}'y);
$m = Yson::MutDownOrCreate($m, 'c'); -- переместились к ключу 'c' в словаре - он создан в состоянии Invalid
$m = Yson::MutUpsert($m, '3'y); -- заменили текущий узел
return Yson::MutFreeze($m); -- {a=1;b=2;c=3}
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('{a=1;b=2}'y);
$m = Yson::MutDown($m, 'c'); -- переместились к ключу 'c' в словаре, ошибка, т.к. он не существует
return Yson::MutFreeze($m);
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('{a=1;b=2}'y);
$m, $ok = Yson::MutTryDown($m, 'c'); -- пытались переместились к ключу 'c' в словаре, вернули false
return (Yson::MutFreeze($m), $ok); -- ({a=1;b=2}, false)
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('[1;2]'y);
$m = Yson::MutDownOrCreate($m, '>last'); -- добавили узел в конец списка в состоянии Invalid
$m = Yson::MutUpsert($m, '3'y); -- заменили текущий узел
return Yson::MutFreeze($m); -- [1;2;3]
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('[1;2]'y);
$m = Yson::MutDownOrCreate($m, '<1'); -- добавили узел перед узлом с индексом 1 (значение '2')
$m = Yson::MutUpsert($m, '3'y); -- заменили текущий узел
return Yson::MutFreeze($m); -- [1;3;2]
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('[1;2]'y);
$m = Yson::MutDownOrCreate($m, '<first'); -- добавили узел в начало списка
$m = Yson::MutUpsert($m, '3'y); -- заменили текущий узел
return Yson::MutFreeze($m); -- [3;1;2]
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('[1;2]'y);
$m = Yson::MutDown($m, '=last'); -- переместились на последний узел списка
$m = Yson::MutUpsert($m, '3'y); -- заменили текущий узел
return Yson::MutFreeze($m); -- [1;3]
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('[1;2]'y);
$m = Yson::MutDownOrCreate($m, 'a'); -- ошибка, текущий узел не словарь
return Yson::MutFreeze($m);
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('[a=1;b=2]'y);
$m = Yson::MutDownOrCreate($m, '=0'); -- ошибка, текущий узел не список
return Yson::MutFreeze($m);
});
Yson::MutExists
Yson::MutExists(Linear<Resource<Yson2.MutNode>>) -> Tuple<Linear<Resource<Yson2.MutNode>>,Bool>
Доступна начиная с версии 2025.05.
Проверяет, что текущий узел не находится в состоянии Invalid или Deleted.
Само изменяемое дерево при этом не меняется, как и положение текущего узла в нем.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::MutCreate, $arg as Depends)();
$m, $exists = Yson::MutExists($m); -- текущий узел в состоянии Invalid
return LinearDestroy($exists, $m); -- возвращаем значение $exists=false, поглощая linear тип $m
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('1'y);
$m, $exists = Yson::MutExists($m); -- текущий узел не является Invalid
return LinearDestroy($exists, $m); -- возвращаем значение $exists=true, поглощая linear тип $m
});
Yson::MutView
Yson::MutView(Linear<Resource<Yson2.MutNode>>) -> Tuple<Linear<Resource<Yson2.MutNode>>, Optional<Resource<Yson2.Node>>>
Доступна начиная с версии 2025.05.
Возвращает неизменяемое Yson дерево начиная с текущего узла, если он не находится в состоянии Invalid или Deleted, в противном случае возвращает NULL.
Как и в функции MutFreeze среди дочерних узлов по отношению к текущему не должно быть узлов в состоянии Invalid, а Deleted узлы будут пропущены при построении результирующего Yson дерева.
Само изменяемое дерево при этом не меняется, как и положение текущего узла в нем.
Примеры
SELECT Block(($arg)->{
$m = Udf(Yson::MutCreate, $arg as Depends)();
$m, $view = Yson::MutView($m); -- текущий узел в состоянии Invalid
return LinearDestroy($view, $m); -- возвращаем значение $view=NULL, поглощая linear тип $m
});
SELECT Block(($arg)->{
$m = Udf(Yson::Mutate, $arg as Depends)('1'y);
$m, $view = Yson::MutView($m); -- текущий узел не является Invalid
return LinearDestroy($view, $m); -- возвращаем значение $view='1'y, поглощая linear тип $m
});