UNPKG

faastjs

Version:

Serverless batch computing made simple.

106 lines 17.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.testCosts = void 0; const ava_1 = require("ava"); const ppp = require("papaparse"); const index_1 = require("../index"); const funcs = require("./fixtures/functions"); const util_1 = require("./fixtures/util"); async function work(faastModule) { await faastModule.functions.monteCarloPI(20000000); } const repetitions = 10; const memorySizes = [1024, 2048]; function filter(configurations) { return configurations .filter(c => memorySizes.includes(c.options.memorySize)) .map(c => ({ ...c, repetitions })); } async function testCostAnalyzer(t, configurations) { const profile = await index_1.CostAnalyzer.analyze({ funcs, work, configurations, silent: true }); t.is(profile.estimates.length, configurations.length); for (const { costSnapshot } of profile.estimates) { t.true(costSnapshot.stats.completed > 0, `completed > 0`); t.is(costSnapshot.stats.errors, 0, `errors === 0`); t.true(costSnapshot.stats.estimatedBilledTime.mean > 0, `billed time > 0`); t.true(costSnapshot.total() > 0, `total > 0`); } const parsed = ppp.parse(profile.csv(), { header: true, skipEmptyLines: true, dynamicTyping: true }); index_1.log.info(`%O`, parsed.data); t.is(parsed.data.length, configurations.length); for (const row of parsed.data) { // cloud,memory,useQueue,options,completed,errors,retries,cost,executionTime,billedTime t.is(typeof row.cloud, "string"); t.is(typeof row.memory, "number"); t.is(typeof row.mode, "string"); t.is(typeof row.options, "string"); t.is(typeof row.completed, "number"); t.is(typeof row.errors, "number"); t.is(typeof row.retries, "number"); t.is(typeof row.cost, "string"); t.is(typeof row.executionTime, "number"); t.is(typeof row.billedTime, "number"); } } async function testCosts(t, provider) { const args = { timeout: 30, memorySize: 512, mode: "queue", maxRetries: 0, gc: "off", description: t.title }; const faastModule = await (0, index_1.faast)(provider, funcs, args); try { await faastModule.functions.hello("there"); const costs = await faastModule.costSnapshot(); const { estimatedBilledTime } = faastModule.stats(); t.is((estimatedBilledTime.mean * estimatedBilledTime.samples) / 1000, costs.costMetrics.find(m => m.name === "functionCallDuration").measured); t.true(costs.costMetrics.length > 1, "cost metrics exist"); t.true(costs.find("functionCallRequests").measured === 1, "functionCallRequests === 1"); const output = costs.toString(); const csvOutput = costs.csv(); let hasPricedMetric = false; for (const metric of costs.costMetrics) { t.regex(output, new RegExp(metric.name)); t.regex(csvOutput, new RegExp(metric.name)); if (!metric.informationalOnly) { t.true(metric.cost() > 0, `${metric.name}.cost() > 0`); t.true(metric.measured > 0, `${metric.name}.measured > 0`); t.true(metric.pricing > 0, `${metric.name}.pricing > 0`); } hasPricedMetric = true; t.true(metric.cost() < 0.00001, `${metric.name}.cost() < 0.00001`); t.true(metric.name.length > 0, `${metric.name}.length > 0`); t.true(metric.unit.length > 0, `${metric.name}.unit.length > 0`); t.true(metric.cost() === metric.pricing * metric.measured, `${metric.name} cost is computed correctly`); } if (hasPricedMetric) { t.true(costs.total() >= 0, `costs.total > 0 (hasPricedMetric)`); } else { t.true(costs.total() === 0, `costs.total === 0 (!hasPricedMetric)`); } } finally { await faastModule.cleanup(); } } exports.testCosts = testCosts; const { awsConfigurations, googleConfigurations } = index_1.CostAnalyzer; (0, ava_1.default)((0, util_1.title)("aws", "cost analyzer"), testCostAnalyzer, filter(awsConfigurations)); (0, ava_1.default)((0, util_1.title)("google", "cost analyzer"), testCostAnalyzer, filter(googleConfigurations)); for (const provider of index_1.providers) { (0, ava_1.default)((0, util_1.title)(provider, `cost estimate for basic calls`), testCosts, provider); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cost.test.js","sourceRoot":"","sources":["../../test/cost.test.ts"],"names":[],"mappings":"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","sourcesContent":["import test, { ExecutionContext } from \"ava\";\nimport * as ppp from \"papaparse\";\nimport {\n    CommonOptions,\n    faast,\n    FaastModule,\n    log,\n    Provider,\n    providers,\n    CostAnalyzer\n} from \"../index\";\nimport * as funcs from \"./fixtures/functions\";\nimport { title } from \"./fixtures/util\";\n\nasync function work(faastModule: FaastModule<typeof funcs>) {\n    await faastModule.functions.monteCarloPI(20000000);\n}\n\nconst repetitions = 10;\nconst memorySizes = [1024, 2048];\n\nfunction filter(configurations: CostAnalyzer.Configuration[]) {\n    return configurations\n        .filter(c => memorySizes.includes(c.options.memorySize!))\n        .map(c => ({ ...c, repetitions }));\n}\n\nasync function testCostAnalyzer(\n    t: ExecutionContext,\n    configurations: CostAnalyzer.Configuration[]\n) {\n    const profile = await CostAnalyzer.analyze({\n        funcs,\n        work,\n        configurations,\n        silent: true\n    });\n    t.is(profile.estimates.length, configurations.length);\n    for (const { costSnapshot } of profile.estimates) {\n        t.true(costSnapshot.stats.completed > 0, `completed > 0`);\n        t.is(costSnapshot.stats.errors, 0, `errors === 0`);\n        t.true(costSnapshot.stats.estimatedBilledTime.mean > 0, `billed time > 0`);\n        t.true(costSnapshot.total() > 0, `total > 0`);\n    }\n\n    const parsed = ppp.parse<any>(profile.csv(), {\n        header: true,\n        skipEmptyLines: true,\n        dynamicTyping: true\n    });\n    log.info(`%O`, parsed.data);\n    t.is(parsed.data.length, configurations.length);\n\n    for (const row of parsed.data) {\n        // cloud,memory,useQueue,options,completed,errors,retries,cost,executionTime,billedTime\n        t.is(typeof row.cloud, \"string\");\n        t.is(typeof row.memory, \"number\");\n        t.is(typeof row.mode, \"string\");\n        t.is(typeof row.options, \"string\");\n        t.is(typeof row.completed, \"number\");\n        t.is(typeof row.errors, \"number\");\n        t.is(typeof row.retries, \"number\");\n        t.is(typeof row.cost, \"string\");\n        t.is(typeof row.executionTime, \"number\");\n        t.is(typeof row.billedTime, \"number\");\n    }\n}\n\nexport async function testCosts(t: ExecutionContext, provider: Provider) {\n    const args: CommonOptions = {\n        timeout: 30,\n        memorySize: 512,\n        mode: \"queue\",\n        maxRetries: 0,\n        gc: \"off\",\n        description: t.title\n    };\n    const faastModule = await faast(provider, funcs, args);\n\n    try {\n        await faastModule.functions.hello(\"there\");\n        const costs = await faastModule.costSnapshot();\n\n        const { estimatedBilledTime } = faastModule.stats();\n        t.is(\n            (estimatedBilledTime.mean * estimatedBilledTime.samples) / 1000,\n            costs.costMetrics.find(m => m.name === \"functionCallDuration\")!.measured\n        );\n\n        t.true(costs.costMetrics.length > 1, \"cost metrics exist\");\n        t.true(\n            costs.find(\"functionCallRequests\")!.measured === 1,\n            \"functionCallRequests === 1\"\n        );\n        const output = costs.toString();\n        const csvOutput = costs.csv();\n        let hasPricedMetric = false;\n        for (const metric of costs.costMetrics) {\n            t.regex(output, new RegExp(metric.name));\n            t.regex(csvOutput, new RegExp(metric.name));\n            if (!metric.informationalOnly) {\n                t.true(metric.cost() > 0, `${metric.name}.cost() > 0`);\n                t.true(metric.measured > 0, `${metric.name}.measured > 0`);\n                t.true(metric.pricing > 0, `${metric.name}.pricing > 0`);\n            }\n            hasPricedMetric = true;\n            t.true(metric.cost() < 0.00001, `${metric.name}.cost() < 0.00001`);\n            t.true(metric.name.length > 0, `${metric.name}.length > 0`);\n            t.true(metric.unit.length > 0, `${metric.name}.unit.length > 0`);\n            t.true(\n                metric.cost() === metric.pricing * metric.measured,\n                `${metric.name} cost is computed correctly`\n            );\n        }\n        if (hasPricedMetric) {\n            t.true(costs.total() >= 0, `costs.total > 0 (hasPricedMetric)`);\n        } else {\n            t.true(costs.total() === 0, `costs.total === 0 (!hasPricedMetric)`);\n        }\n    } finally {\n        await faastModule.cleanup();\n    }\n}\n\nconst { awsConfigurations, googleConfigurations } = CostAnalyzer;\ntest(title(\"aws\", \"cost analyzer\"), testCostAnalyzer, filter(awsConfigurations));\ntest(title(\"google\", \"cost analyzer\"), testCostAnalyzer, filter(googleConfigurations));\n\nfor (const provider of providers) {\n    test(title(provider, `cost estimate for basic calls`), testCosts, provider);\n}\n"]}