如何在Foundry Functions中拥有灵活的分组列?

Edu*_*sko 3 typescript palantir-foundry foundry-functions

在我的 Workshop 应用程序中,我想要一个带有可更改 x 轴的条形图。下拉小部件将用于选择所需的 x 轴。

为此,我正在编写一个 TypeScript 函数,它将返回为图表小部件提供数据的数据。

我写了以下函数:

@Function()
public async flexibleCounter2DimensionTest(
    objects: ObjectSet<Data>,
    xaxis : string ): Promise<TwoDimensionalAggregation<string>> {
    return objects
            .groupBy(val => val[xaxis].topValues())
            .count()
}
Run Code Online (Sandbox Code Playgroud)

xaxis参数将定义要对其执行分组的列名称。

我在 Foundry Functions 中努力做到这一点,我总是在编译时收到以下错误:

{ "stdout": "src/hospital_provider_analysis.ts(170,9): 错误 TS2322: 类型 'TwoDimensionalAggregation<BucketKey, number>' 不可分配给类型 'TwoDimensionalAggregation<string, number>'。\n 类型 'BucketKey' 是不可分配给类型“string”。\n 类型“false”不可分配给类型“string”。\nsrc/hospital_provider_analysis.ts(171,33):错误 TS7053:元素隐式具有“any”类型,因为类型的表达式'string' 不能用于索引类型 'BucketableProperties'。\n 在类型 'BucketableProperties' 上未找到具有类型为 'string' 的参数的索引签名。\n", "stderr": "" }

有什么想法如何解决这个问题吗?

小智 5

Typescript 类型系统的一些更强大的部分允许我们指定此聚合函数的类型安全版本。我在下面的第 1 部分中详细介绍了一种方法。

当公开这个类型安全的聚合函数时,我们需要执行一些额外的检查,因为 Functions 目前不支持联合类型(请参阅 Palantir 文档中的“函数 API:输入和输出类型”)。我在下面的第 2 部分中列出了我的代码。

第 1 部分:类型安全聚合

首先,我们列出所有想要分组为文字类型的新联合类型的属性 API 名称。这些应该是您通常用来访问函数中这些属性的 API 名称。

type AggregatableStringProperties = "propertyA" | "propertyB" | ...;
Run Code Online (Sandbox Code Playgroud)

现在,我们可以定义一个类型安全的聚合函数,它接受一个对象集以及要聚合的属性(在 中AggregatableStringProperties):

private async flexibleCounter2DimensionTestImpl(
    objects: ObjectSet<Data>,
    xaxis: AggregatableStringProperties,
) : Promise<TwoDimensionalAggregation<string>> {
    return objects
        .groupBy(val => val[xaxis].topValues())
        .count();
}
Run Code Online (Sandbox Code Playgroud)

第 2 部分:将其公开为@Function()

使用我们在第 1 部分中定义的类型安全聚合函数,我们现在可以使用类型断言来公开它。在下面的代码中,我添加了一些可选逻辑,以为该函数的用户提供有用的调试信息。

@Function()
public async flexibleCounter2DimensionTest(
    objects: ObjectSet<Data>,
    xaxis : string
): Promise<TwoDimensionalAggregation<string>> {
    const xaxis_t = xaxis as AggregatableStringProperties;
    if (Data.properties[xaxis_t] === undefined || Data.properties[xaxis_t].baseType.type !== "string") {
        throw new UserFacingError("xaxis argument ('" + xaxis + "') is not a string property of the Data object");
    }
    return this.flexibleCounter2DimensionTestImpl(objects, xaxis_t);
}
Run Code Online (Sandbox Code Playgroud)