faastjs
Version:
Serverless batch computing made simple.
106 lines • 17.2 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.testCosts = void 0;
const tslib_1 = require("tslib");
const ava_1 = tslib_1.__importDefault(require("ava"));
const papaparse_1 = tslib_1.__importDefault(require("papaparse"));
const index_1 = require("../index");
const funcs = tslib_1.__importStar(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 = papaparse_1.default.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 } = index_1.CostAnalyzer;
(0, ava_1.default)((0, util_1.title)("aws", "cost analyzer"), testCostAnalyzer, filter(awsConfigurations));
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 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 } = CostAnalyzer;\ntest(title(\"aws\", \"cost analyzer\"), testCostAnalyzer, filter(awsConfigurations));\n\nfor (const provider of providers) {\n    test(title(provider, `cost estimate for basic calls`), testCosts, provider);\n}\n"]}