yql_eval_params.cpp 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. #include "yql_eval_params.h"
  2. #include <yql/essentials/core/type_ann/type_ann_core.h>
  3. #include <yql/essentials/core/type_ann/type_ann_expr.h>
  4. #include <yql/essentials/providers/common/codec/yql_codec.h>
  5. #include <yql/essentials/providers/common/schema/expr/yql_expr_schema.h>
  6. #include <yql/essentials/providers/common/mkql/yql_type_mkql.h>
  7. #include <yql/essentials/minikql/mkql_alloc.h>
  8. #include <util/string/builder.h>
  9. namespace NYql {
  10. using namespace NKikimr;
  11. using namespace NKikimr::NMiniKQL;
  12. namespace {
  13. bool BuildParameterValuesAsNodes(const THashMap<TStringBuf, const TTypeAnnotationNode*>& paramTypes,
  14. const NYT::TNode& paramData, TExprContext& ctx, const IFunctionRegistry& functionRegistry,
  15. THashMap<TStringBuf, TExprNode::TPtr>& paramValues) {
  16. if (!paramData.IsMap()) {
  17. ctx.AddError(TIssue({}, TStringBuilder() << "ParamData is not a map"));
  18. return false;
  19. }
  20. TScopedAlloc alloc(__LOCATION__);
  21. TTypeEnvironment env(alloc);
  22. TMemoryUsageInfo memInfo("Parameters");
  23. THolderFactory holderFactory(alloc.Ref(), memInfo);
  24. NCommon::TMemoizedTypesMap typesMemoization;
  25. bool isOk = true;
  26. auto& paramDataMap = paramData.AsMap();
  27. for (auto& p : paramTypes) {
  28. auto name = p.first;
  29. TStringStream err;
  30. TProgramBuilder pgmBuilder(env, functionRegistry);
  31. TType* mkqlType = NCommon::BuildType(*p.second, pgmBuilder, typesMemoization, err);
  32. if (!mkqlType) {
  33. ctx.AddError(TIssue({}, TStringBuilder() << "Failed to process type for parameter: " << name << ", reason: " << err.Str()));
  34. isOk = false;
  35. continue;
  36. }
  37. const auto parameterItem = paramDataMap.FindPtr(name);
  38. if (parameterItem && (!parameterItem->IsMap() || !parameterItem->HasKey("Data"))) {
  39. ctx.AddError(TIssue({}, TStringBuilder() << "Parameter '" << name << "' value should be a map with key 'Data'"));
  40. isOk = false;
  41. continue;
  42. }
  43. if (!parameterItem && p.second->GetKind() != ETypeAnnotationKind::Optional && p.second->GetKind() != ETypeAnnotationKind::Null) {
  44. ctx.AddError(TIssue({}, TStringBuilder() << "Missing value for parameter: " << name));
  45. isOk = false;
  46. continue;
  47. }
  48. auto value = parameterItem ? NCommon::ParseYsonNodeInResultFormat(holderFactory, (*parameterItem)["Data"], mkqlType, &err) : MakeMaybe(NUdf::TUnboxedValue());
  49. if (!value) {
  50. ctx.AddError(TIssue({}, TStringBuilder() << "Failed to parse data for parameter: " << name << ", reason: " << err.Str()));
  51. isOk = false;
  52. continue;
  53. }
  54. paramValues[name] = NCommon::ValueToExprLiteral(p.second, *value, ctx);
  55. }
  56. return isOk;
  57. }
  58. bool ExtractParameterTypes(const TExprNode::TPtr& input, TTypeAnnotationContext& types,
  59. TExprContext& ctx, THashMap<TStringBuf, const TTypeAnnotationNode*>& paramTypes) {
  60. auto callableTransformer = CreateExtCallableTypeAnnotationTransformer(types);
  61. auto typeTransformer = CreateTypeAnnotationTransformer(callableTransformer, types);
  62. TVector<TTransformStage> transformers;
  63. const auto issueCode = TIssuesIds::CORE_TYPE_ANN;
  64. transformers.push_back(TTransformStage(typeTransformer, "TypeAnnotation", issueCode));
  65. auto fullTransformer = CreateCompositeGraphTransformer(transformers, false);
  66. TOptimizeExprSettings settings(nullptr);
  67. settings.VisitChanges = true;
  68. TExprNode::TPtr output = input;
  69. auto status1 = OptimizeExpr(input, output, [&](const TExprNode::TPtr& node, TExprContext& ctx)->TExprNode::TPtr {
  70. if (!node->IsCallable("Parameter")) {
  71. return node;
  72. }
  73. auto param = node;
  74. fullTransformer->Rewind();
  75. auto status = InstantTransform(*fullTransformer, param, ctx);
  76. if (status.Level == IGraphTransformer::TStatus::Error) {
  77. return nullptr;
  78. }
  79. auto name = param->Child(0)->Content();
  80. if (!param->GetTypeAnn()) {
  81. ctx.AddError(TIssue(ctx.GetPosition(param->Pos()), TStringBuilder() << "Failed to type check parameter: " << name));
  82. return nullptr;
  83. }
  84. auto& type = paramTypes[name];
  85. if (!type) {
  86. type = param->GetTypeAnn();
  87. } else if (!IsSameAnnotation(*type, *param->GetTypeAnn())) {
  88. ctx.AddError(TIssue(ctx.GetPosition(param->Pos()), TStringBuilder() << "Mismatch of types: " << *type << " != " << *param->GetTypeAnn()
  89. << " for parameter: " << name));
  90. return nullptr;
  91. }
  92. return param;
  93. }, ctx, settings);
  94. return status1.Level == IGraphTransformer::TStatus::Ok;
  95. }
  96. }
  97. bool ExtractParametersMetaAsYson(const TExprNode::TPtr& input, TTypeAnnotationContext& types,
  98. TExprContext& ctx, NYT::TNode& paramsMetaMap) {
  99. THashMap<TStringBuf, const TTypeAnnotationNode*> params;
  100. if (!ExtractParameterTypes(input, types, ctx, params)) {
  101. return false;
  102. }
  103. for (auto& p : params) {
  104. paramsMetaMap[p.first] = NCommon::TypeToYsonNode(p.second);
  105. }
  106. return true;
  107. }
  108. IGraphTransformer::TStatus EvaluateParameters(const TExprNode::TPtr& input, TExprNode::TPtr& output,
  109. TTypeAnnotationContext& types, TExprContext& ctx, const IFunctionRegistry& functionRegistry) {
  110. output = input;
  111. if (ctx.Step.IsDone(TExprStep::Params)) {
  112. return IGraphTransformer::TStatus::Ok;
  113. }
  114. THashMap<TStringBuf, const TTypeAnnotationNode*> paramTypes;
  115. if (!ExtractParameterTypes(input, types, ctx, paramTypes)) {
  116. return IGraphTransformer::TStatus::Error;
  117. }
  118. if (!paramTypes) {
  119. // no params - just exit
  120. return IGraphTransformer::TStatus::Ok;
  121. }
  122. const auto emptyMapNode = NYT::TNode::CreateMap();
  123. const auto& paramData = types.OperationOptions.ParametersYson.GetOrElse(emptyMapNode);
  124. THashMap<TStringBuf, TExprNode::TPtr> paramValues;
  125. if (!BuildParameterValuesAsNodes(paramTypes, paramData, ctx, functionRegistry, paramValues)) {
  126. return IGraphTransformer::TStatus::Error;
  127. }
  128. // inject param values into graph
  129. TOptimizeExprSettings settings(nullptr);
  130. settings.VisitChanges = true;
  131. auto status = OptimizeExpr(output, output, [&](const TExprNode::TPtr& node, TExprContext& ctx)->TExprNode::TPtr {
  132. if (!node->IsCallable("Parameter")) {
  133. return node;
  134. }
  135. auto name = node->Child(0)->Content();
  136. auto evaluated = paramValues.FindPtr(name);
  137. YQL_ENSURE(evaluated, "Missing parameter value: " << name);
  138. return ctx.ShallowCopyWithPosition(**evaluated, node->Pos());
  139. }, ctx, settings);
  140. if (status.Level == IGraphTransformer::TStatus::Error) {
  141. return status;
  142. }
  143. ctx.Step.Done(TExprStep::Params);
  144. return IGraphTransformer::TStatus::Ok;
  145. }
  146. } // namespace NYql