@cdktn/provider-schema 0.23.0-pre.39 → 0.23.0-pre.40
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/__tests__/provider-schema.unit.test.d.ts +2 -0
- package/build/__tests__/provider-schema.unit.test.d.ts.map +1 -0
- package/build/__tests__/provider-schema.unit.test.js +58 -0
- package/build/provider-schema.d.ts.map +1 -1
- package/build/provider-schema.js +56 -3
- package/eslint.config.mjs +1 -74
- package/package.json +4 -12
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"provider-schema.unit.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/provider-schema.unit.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright (c) HashiCorp, Inc
|
|
3
|
+
// SPDX-License-Identifier: MPL-2.0
|
|
4
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
5
|
+
jest.mock("@cdktn/commons", () => ({
|
|
6
|
+
exec: jest.fn(),
|
|
7
|
+
}));
|
|
8
|
+
const commons_1 = require("@cdktn/commons");
|
|
9
|
+
const provider_schema_1 = require("../provider-schema");
|
|
10
|
+
const execMock = commons_1.exec;
|
|
11
|
+
function transientError(message = "502 Bad Gateway from github.com") {
|
|
12
|
+
const err = new Error("non-zero exit code 1");
|
|
13
|
+
err.stderr = message;
|
|
14
|
+
return err;
|
|
15
|
+
}
|
|
16
|
+
function fatalError(message = "Error: Invalid provider version") {
|
|
17
|
+
const err = new Error("non-zero exit code 1");
|
|
18
|
+
err.stderr = message;
|
|
19
|
+
return err;
|
|
20
|
+
}
|
|
21
|
+
describe("terraformInitWithRetry", () => {
|
|
22
|
+
let consoleErrorSpy;
|
|
23
|
+
beforeEach(() => {
|
|
24
|
+
execMock.mockReset();
|
|
25
|
+
consoleErrorSpy = jest
|
|
26
|
+
.spyOn(console, "error")
|
|
27
|
+
.mockImplementation(() => undefined);
|
|
28
|
+
});
|
|
29
|
+
afterEach(() => {
|
|
30
|
+
consoleErrorSpy.mockRestore();
|
|
31
|
+
});
|
|
32
|
+
it("returns immediately on first-attempt success", async () => {
|
|
33
|
+
execMock.mockResolvedValueOnce("ok");
|
|
34
|
+
const result = await (0, provider_schema_1.terraformInitWithRetry)({ cwd: "/tmp/x" }, 3, 0);
|
|
35
|
+
expect(result).toBe("ok");
|
|
36
|
+
expect(execMock).toHaveBeenCalledTimes(1);
|
|
37
|
+
});
|
|
38
|
+
it("retries on transient stderr and eventually succeeds", async () => {
|
|
39
|
+
execMock
|
|
40
|
+
.mockRejectedValueOnce(transientError())
|
|
41
|
+
.mockRejectedValueOnce(transientError("ECONNRESET"))
|
|
42
|
+
.mockResolvedValueOnce("ok");
|
|
43
|
+
const result = await (0, provider_schema_1.terraformInitWithRetry)({ cwd: "/tmp/x" }, 3, 0);
|
|
44
|
+
expect(result).toBe("ok");
|
|
45
|
+
expect(execMock).toHaveBeenCalledTimes(3);
|
|
46
|
+
});
|
|
47
|
+
it("does not retry on non-transient errors", async () => {
|
|
48
|
+
execMock.mockRejectedValueOnce(fatalError());
|
|
49
|
+
await expect((0, provider_schema_1.terraformInitWithRetry)({ cwd: "/tmp/x" }, 3, 0)).rejects.toMatchObject({ stderr: expect.stringContaining("Invalid") });
|
|
50
|
+
expect(execMock).toHaveBeenCalledTimes(1);
|
|
51
|
+
});
|
|
52
|
+
it("throws after exhausting retries on persistent transient errors", async () => {
|
|
53
|
+
execMock.mockRejectedValue(transientError("503 Service Unavailable"));
|
|
54
|
+
await expect((0, provider_schema_1.terraformInitWithRetry)({ cwd: "/tmp/x" }, 3, 0)).rejects.toMatchObject({ stderr: expect.stringContaining("503") });
|
|
55
|
+
expect(execMock).toHaveBeenCalledTimes(3);
|
|
56
|
+
});
|
|
57
|
+
});
|
|
58
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdmlkZXItc2NoZW1hLnVuaXQudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9fX3Rlc3RzX18vcHJvdmlkZXItc2NoZW1hLnVuaXQudGVzdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsK0JBQStCO0FBQy9CLG1DQUFtQzs7QUFFbkMsSUFBSSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxHQUFHLEVBQUUsQ0FBQyxDQUFDO0lBQ2pDLElBQUksRUFBRSxJQUFJLENBQUMsRUFBRSxFQUFFO0NBQ2hCLENBQUMsQ0FBQyxDQUFDO0FBRUosNENBQXNDO0FBQ3RDLHdEQUE0RDtBQUU1RCxNQUFNLFFBQVEsR0FBRyxjQUE0QixDQUFDO0FBRTlDLFNBQVMsY0FBYyxDQUFDLE9BQU8sR0FBRyxpQ0FBaUM7SUFDakUsTUFBTSxHQUFHLEdBQVEsSUFBSSxLQUFLLENBQUMsc0JBQXNCLENBQUMsQ0FBQztJQUNuRCxHQUFHLENBQUMsTUFBTSxHQUFHLE9BQU8sQ0FBQztJQUNyQixPQUFPLEdBQUcsQ0FBQztBQUNiLENBQUM7QUFFRCxTQUFTLFVBQVUsQ0FBQyxPQUFPLEdBQUcsaUNBQWlDO0lBQzdELE1BQU0sR0FBRyxHQUFRLElBQUksS0FBSyxDQUFDLHNCQUFzQixDQUFDLENBQUM7SUFDbkQsR0FBRyxDQUFDLE1BQU0sR0FBRyxPQUFPLENBQUM7SUFDckIsT0FBTyxHQUFHLENBQUM7QUFDYixDQUFDO0FBRUQsUUFBUSxDQUFDLHdCQUF3QixFQUFFLEdBQUcsRUFBRTtJQUN0QyxJQUFJLGVBQWlDLENBQUM7SUFFdEMsVUFBVSxDQUFDLEdBQUcsRUFBRTtRQUNkLFFBQVEsQ0FBQyxTQUFTLEVBQUUsQ0FBQztRQUNyQixlQUFlLEdBQUcsSUFBSTthQUNuQixLQUFLLENBQUMsT0FBTyxFQUFFLE9BQU8sQ0FBQzthQUN2QixrQkFBa0IsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUN6QyxDQUFDLENBQUMsQ0FBQztJQUVILFNBQVMsQ0FBQyxHQUFHLEVBQUU7UUFDYixlQUFlLENBQUMsV0FBVyxFQUFFLENBQUM7SUFDaEMsQ0FBQyxDQUFDLENBQUM7SUFFSCxFQUFFLENBQUMsOENBQThDLEVBQUUsS0FBSyxJQUFJLEVBQUU7UUFDNUQsUUFBUSxDQUFDLHFCQUFxQixDQUFDLElBQUksQ0FBQyxDQUFDO1FBRXJDLE1BQU0sTUFBTSxHQUFHLE1BQU0sSUFBQSx3Q0FBc0IsRUFBQyxFQUFFLEdBQUcsRUFBRSxRQUFRLEVBQUUsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFFckUsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMxQixNQUFNLENBQUMsUUFBUSxDQUFDLENBQUMscUJBQXFCLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDNUMsQ0FBQyxDQUFDLENBQUM7SUFFSCxFQUFFLENBQUMscURBQXFELEVBQUUsS0FBSyxJQUFJLEVBQUU7UUFDbkUsUUFBUTthQUNMLHFCQUFxQixDQUFDLGNBQWMsRUFBRSxDQUFDO2FBQ3ZDLHFCQUFxQixDQUFDLGNBQWMsQ0FBQyxZQUFZLENBQUMsQ0FBQzthQUNuRCxxQkFBcUIsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUUvQixNQUFNLE1BQU0sR0FBRyxNQUFNLElBQUEsd0NBQXNCLEVBQUMsRUFBRSxHQUFHLEVBQUUsUUFBUSxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBRXJFLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDMUIsTUFBTSxDQUFDLFFBQVEsQ0FBQyxDQUFDLHFCQUFxQixDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQzVDLENBQUMsQ0FBQyxDQUFDO0lBRUgsRUFBRSxDQUFDLHdDQUF3QyxFQUFFLEtBQUssSUFBSSxFQUFFO1FBQ3RELFFBQVEsQ0FBQyxxQkFBcUIsQ0FBQyxVQUFVLEVBQUUsQ0FBQyxDQUFDO1FBRTdDLE1BQU0sTUFBTSxDQUNWLElBQUEsd0NBQXNCLEVBQUMsRUFBRSxHQUFHLEVBQUUsUUFBUSxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUNoRCxDQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsRUFBRSxNQUFNLEVBQUUsTUFBTSxDQUFDLGdCQUFnQixDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUMsQ0FBQztRQUV4RSxNQUFNLENBQUMsUUFBUSxDQUFDLENBQUMscUJBQXFCLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDNUMsQ0FBQyxDQUFDLENBQUM7SUFFSCxFQUFFLENBQUMsZ0VBQWdFLEVBQUUsS0FBSyxJQUFJLEVBQUU7UUFDOUUsUUFBUSxDQUFDLGlCQUFpQixDQUFDLGNBQWMsQ0FBQyx5QkFBeUIsQ0FBQyxDQUFDLENBQUM7UUFFdEUsTUFBTSxNQUFNLENBQ1YsSUFBQSx3Q0FBc0IsRUFBQyxFQUFFLEdBQUcsRUFBRSxRQUFRLEVBQUUsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQ2hELENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxFQUFFLE1BQU0sRUFBRSxNQUFNLENBQUMsZ0JBQWdCLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBRXBFLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQyxxQkFBcUIsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUM1QyxDQUFDLENBQUMsQ0FBQztBQUNMLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IChjKSBIYXNoaUNvcnAsIEluY1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IE1QTC0yLjBcblxuamVzdC5tb2NrKFwiQGNka3RuL2NvbW1vbnNcIiwgKCkgPT4gKHtcbiAgZXhlYzogamVzdC5mbigpLFxufSkpO1xuXG5pbXBvcnQgeyBleGVjIH0gZnJvbSBcIkBjZGt0bi9jb21tb25zXCI7XG5pbXBvcnQgeyB0ZXJyYWZvcm1Jbml0V2l0aFJldHJ5IH0gZnJvbSBcIi4uL3Byb3ZpZGVyLXNjaGVtYVwiO1xuXG5jb25zdCBleGVjTW9jayA9IGV4ZWMgYXMgdW5rbm93biBhcyBqZXN0Lk1vY2s7XG5cbmZ1bmN0aW9uIHRyYW5zaWVudEVycm9yKG1lc3NhZ2UgPSBcIjUwMiBCYWQgR2F0ZXdheSBmcm9tIGdpdGh1Yi5jb21cIikge1xuICBjb25zdCBlcnI6IGFueSA9IG5ldyBFcnJvcihcIm5vbi16ZXJvIGV4aXQgY29kZSAxXCIpO1xuICBlcnIuc3RkZXJyID0gbWVzc2FnZTtcbiAgcmV0dXJuIGVycjtcbn1cblxuZnVuY3Rpb24gZmF0YWxFcnJvcihtZXNzYWdlID0gXCJFcnJvcjogSW52YWxpZCBwcm92aWRlciB2ZXJzaW9uXCIpIHtcbiAgY29uc3QgZXJyOiBhbnkgPSBuZXcgRXJyb3IoXCJub24temVybyBleGl0IGNvZGUgMVwiKTtcbiAgZXJyLnN0ZGVyciA9IG1lc3NhZ2U7XG4gIHJldHVybiBlcnI7XG59XG5cbmRlc2NyaWJlKFwidGVycmFmb3JtSW5pdFdpdGhSZXRyeVwiLCAoKSA9PiB7XG4gIGxldCBjb25zb2xlRXJyb3JTcHk6IGplc3QuU3B5SW5zdGFuY2U7XG5cbiAgYmVmb3JlRWFjaCgoKSA9PiB7XG4gICAgZXhlY01vY2subW9ja1Jlc2V0KCk7XG4gICAgY29uc29sZUVycm9yU3B5ID0gamVzdFxuICAgICAgLnNweU9uKGNvbnNvbGUsIFwiZXJyb3JcIilcbiAgICAgIC5tb2NrSW1wbGVtZW50YXRpb24oKCkgPT4gdW5kZWZpbmVkKTtcbiAgfSk7XG5cbiAgYWZ0ZXJFYWNoKCgpID0+IHtcbiAgICBjb25zb2xlRXJyb3JTcHkubW9ja1Jlc3RvcmUoKTtcbiAgfSk7XG5cbiAgaXQoXCJyZXR1cm5zIGltbWVkaWF0ZWx5IG9uIGZpcnN0LWF0dGVtcHQgc3VjY2Vzc1wiLCBhc3luYyAoKSA9PiB7XG4gICAgZXhlY01vY2subW9ja1Jlc29sdmVkVmFsdWVPbmNlKFwib2tcIik7XG5cbiAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0ZXJyYWZvcm1Jbml0V2l0aFJldHJ5KHsgY3dkOiBcIi90bXAveFwiIH0sIDMsIDApO1xuXG4gICAgZXhwZWN0KHJlc3VsdCkudG9CZShcIm9rXCIpO1xuICAgIGV4cGVjdChleGVjTW9jaykudG9IYXZlQmVlbkNhbGxlZFRpbWVzKDEpO1xuICB9KTtcblxuICBpdChcInJldHJpZXMgb24gdHJhbnNpZW50IHN0ZGVyciBhbmQgZXZlbnR1YWxseSBzdWNjZWVkc1wiLCBhc3luYyAoKSA9PiB7XG4gICAgZXhlY01vY2tcbiAgICAgIC5tb2NrUmVqZWN0ZWRWYWx1ZU9uY2UodHJhbnNpZW50RXJyb3IoKSlcbiAgICAgIC5tb2NrUmVqZWN0ZWRWYWx1ZU9uY2UodHJhbnNpZW50RXJyb3IoXCJFQ09OTlJFU0VUXCIpKVxuICAgICAgLm1vY2tSZXNvbHZlZFZhbHVlT25jZShcIm9rXCIpO1xuXG4gICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgdGVycmFmb3JtSW5pdFdpdGhSZXRyeSh7IGN3ZDogXCIvdG1wL3hcIiB9LCAzLCAwKTtcblxuICAgIGV4cGVjdChyZXN1bHQpLnRvQmUoXCJva1wiKTtcbiAgICBleHBlY3QoZXhlY01vY2spLnRvSGF2ZUJlZW5DYWxsZWRUaW1lcygzKTtcbiAgfSk7XG5cbiAgaXQoXCJkb2VzIG5vdCByZXRyeSBvbiBub24tdHJhbnNpZW50IGVycm9yc1wiLCBhc3luYyAoKSA9PiB7XG4gICAgZXhlY01vY2subW9ja1JlamVjdGVkVmFsdWVPbmNlKGZhdGFsRXJyb3IoKSk7XG5cbiAgICBhd2FpdCBleHBlY3QoXG4gICAgICB0ZXJyYWZvcm1Jbml0V2l0aFJldHJ5KHsgY3dkOiBcIi90bXAveFwiIH0sIDMsIDApLFxuICAgICkucmVqZWN0cy50b01hdGNoT2JqZWN0KHsgc3RkZXJyOiBleHBlY3Quc3RyaW5nQ29udGFpbmluZyhcIkludmFsaWRcIikgfSk7XG5cbiAgICBleHBlY3QoZXhlY01vY2spLnRvSGF2ZUJlZW5DYWxsZWRUaW1lcygxKTtcbiAgfSk7XG5cbiAgaXQoXCJ0aHJvd3MgYWZ0ZXIgZXhoYXVzdGluZyByZXRyaWVzIG9uIHBlcnNpc3RlbnQgdHJhbnNpZW50IGVycm9yc1wiLCBhc3luYyAoKSA9PiB7XG4gICAgZXhlY01vY2subW9ja1JlamVjdGVkVmFsdWUodHJhbnNpZW50RXJyb3IoXCI1MDMgU2VydmljZSBVbmF2YWlsYWJsZVwiKSk7XG5cbiAgICBhd2FpdCBleHBlY3QoXG4gICAgICB0ZXJyYWZvcm1Jbml0V2l0aFJldHJ5KHsgY3dkOiBcIi90bXAveFwiIH0sIDMsIDApLFxuICAgICkucmVqZWN0cy50b01hdGNoT2JqZWN0KHsgc3RkZXJyOiBleHBlY3Quc3RyaW5nQ29udGFpbmluZyhcIjUwM1wiKSB9KTtcblxuICAgIGV4cGVjdChleGVjTW9jaykudG9IYXZlQmVlbkNhbGxlZFRpbWVzKDMpO1xuICB9KTtcbn0pO1xuIl19
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"provider-schema.d.ts","sourceRoot":"","sources":["../src/provider-schema.ts"],"names":[],"mappings":"AAMA,OAAO,EAIL,2BAA2B,EAC3B,6BAA6B,EAG7B,YAAY,EACZ,cAAc,EAMf,MAAM,gBAAgB,CAAC;
|
|
1
|
+
{"version":3,"file":"provider-schema.d.ts","sourceRoot":"","sources":["../src/provider-schema.ts"],"names":[],"mappings":"AAMA,OAAO,EAIL,2BAA2B,EAC3B,6BAA6B,EAG7B,YAAY,EACZ,cAAc,EAMf,MAAM,gBAAgB,CAAC;AAiExB;;;GAGG;AACH,MAAM,MAAM,IAAI,GAAG,MAAM,GAAG;IAAE,MAAM,EAAE,4BAA4B,CAAA;CAAE,CAAC;AACrE,MAAM,MAAM,gBAAgB,GAAG,MAAM,GAAG;IAAE,MAAM,EAAE,kBAAkB,CAAA;CAAE,CAAC;AACvE,MAAM,MAAM,iBAAiB,GAAG,MAAM,GAAG;IAAE,MAAM,EAAE,mBAAmB,CAAA;CAAE,CAAC;AACzE,MAAM,MAAM,YAAY,GAAG,MAAM,GAAG;IAAE,MAAM,EAAE,cAAc,CAAA;CAAE,CAAC;AAE/D,eAAO,MAAM,SAAS,MAAO,IAAI;cAMnB,gBAAgB;eACf,iBAAiB;UACtB,YAAY;CAErB,CAAC;AAkIF,MAAM,WAAW,eAAe;IAC9B,QAAQ,CAAC,EAAE;QAAE,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;KAAE,CAAC;IACnD,SAAS,EAAE;QACT,kBAAkB,CAAC,EAAE;YACnB,CAAC,IAAI,EAAE,MAAM,GAAG;gBAAE,MAAM,CAAC,EAAE,MAAM,CAAC;gBAAC,OAAO,CAAC,EAAE,MAAM,CAAA;aAAE,CAAC;SACvD,CAAC;KACH,CAAC;IACF,MAAM,CAAC,EAAE;QAAE,CAAC,IAAI,EAAE,MAAM,GAAG;YAAE,MAAM,EAAE,MAAM,CAAC;YAAC,OAAO,CAAC,EAAE,MAAM,CAAA;SAAE,CAAA;KAAE,CAAC;CACnE;AAED,wBAAsB,kBAAkB,CACtC,MAAM,EAAE,6BAA6B,2BAuCtC;AAID,wBAAgB,sBAAsB,CAAC,MAAM,EAAE,cAAc,GAAG,cAAc,CAsC7E;AAED,wBAAsB,gBAAgB,CAAC,MAAM,EAAE,2BAA2B,yCAqCzE"}
|
package/build/provider-schema.js
CHANGED
|
@@ -23,7 +23,7 @@ var __importStar = (this && this.__importStar) || function (mod) {
|
|
|
23
23
|
return result;
|
|
24
24
|
};
|
|
25
25
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
|
-
exports.readModuleSchema = exports.sanitizeProviderSchema = exports.readProviderSchema = exports.parseFQPN = void 0;
|
|
26
|
+
exports.readModuleSchema = exports.sanitizeProviderSchema = exports.readProviderSchema = exports.parseFQPN = exports.terraformInitWithRetry = void 0;
|
|
27
27
|
// Copyright (c) HashiCorp, Inc
|
|
28
28
|
// SPDX-License-Identifier: MPL-2.0
|
|
29
29
|
const fs = __importStar(require("fs-extra"));
|
|
@@ -31,6 +31,59 @@ const path = __importStar(require("path"));
|
|
|
31
31
|
const hcl2json_1 = require("@cdktn/hcl2json");
|
|
32
32
|
const commons_1 = require("@cdktn/commons");
|
|
33
33
|
const terraformBinaryName = process.env.TERRAFORM_BINARY_NAME || "terraform";
|
|
34
|
+
// Provider binaries are downloaded from GitHub release CDNs during
|
|
35
|
+
// `terraform init`, which intermittently returns 5xx. Retry on those
|
|
36
|
+
// signatures only — real config/schema errors should still fail fast.
|
|
37
|
+
const TRANSIENT_INIT_ERROR_PATTERNS = [
|
|
38
|
+
/\b50[234]\b/,
|
|
39
|
+
/Bad Gateway/i,
|
|
40
|
+
/Service Unavailable/i,
|
|
41
|
+
/Gateway Timeout/i,
|
|
42
|
+
/ECONNRESET/i,
|
|
43
|
+
/ETIMEDOUT/i,
|
|
44
|
+
/EAI_AGAIN/i,
|
|
45
|
+
/unexpected EOF/i,
|
|
46
|
+
];
|
|
47
|
+
/**
|
|
48
|
+
* Runs `terraform init` and retries on transient HTTP/network failures.
|
|
49
|
+
*
|
|
50
|
+
* Provider binaries are pulled from GitHub's release CDN, which
|
|
51
|
+
* intermittently returns 5xx. The retry only fires when stderr matches a
|
|
52
|
+
* known transient pattern — real errors (bad version, schema parse, etc.)
|
|
53
|
+
* still fail on the first attempt.
|
|
54
|
+
*
|
|
55
|
+
* @param options - spawn options forwarded to `exec` (`cwd` is required so
|
|
56
|
+
* `terraform init` runs inside the temp dir holding `main.tf.json`)
|
|
57
|
+
* @param maxAttempts - total attempts including the initial call; the number
|
|
58
|
+
* of retries is `maxAttempts - 1`. Defaults to 3 (1 initial + 2 retries)
|
|
59
|
+
* @param baseDelayMs - delay before the first retry. Tests pass `0` to skip
|
|
60
|
+
* waits. Defaults to 1000ms
|
|
61
|
+
* @param backoffMultiplier - factor applied to the delay on each subsequent
|
|
62
|
+
* retry (so retries wait `baseDelayMs * multiplier^(attempt-1)`). Defaults
|
|
63
|
+
* to 2 (1s → 2s → 4s …)
|
|
64
|
+
* @returns stdout from the successful `terraform init` invocation
|
|
65
|
+
* @internal exposed for testing
|
|
66
|
+
*/
|
|
67
|
+
async function terraformInitWithRetry(options, maxAttempts = 3, baseDelayMs = 1000, backoffMultiplier = 2) {
|
|
68
|
+
var _a;
|
|
69
|
+
for (let attempt = 1; attempt <= maxAttempts; attempt++) {
|
|
70
|
+
try {
|
|
71
|
+
return await (0, commons_1.exec)(terraformBinaryName, ["init"], options);
|
|
72
|
+
}
|
|
73
|
+
catch (error) {
|
|
74
|
+
const stderr = (_a = error === null || error === void 0 ? void 0 : error.stderr) !== null && _a !== void 0 ? _a : "";
|
|
75
|
+
const transient = TRANSIENT_INIT_ERROR_PATTERNS.some((p) => p.test(stderr));
|
|
76
|
+
if (!transient || attempt === maxAttempts) {
|
|
77
|
+
throw error;
|
|
78
|
+
}
|
|
79
|
+
const delayMs = baseDelayMs * backoffMultiplier ** (attempt - 1);
|
|
80
|
+
console.error(`terraform init failed with a transient error, retrying in ${delayMs}ms (attempt ${attempt}/${maxAttempts - 1})`);
|
|
81
|
+
await new Promise((resolve) => setTimeout(resolve, delayMs));
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
throw new Error("unreachable");
|
|
85
|
+
}
|
|
86
|
+
exports.terraformInitWithRetry = terraformInitWithRetry;
|
|
34
87
|
const parseFQPN = (f) => {
|
|
35
88
|
const [hostname, namespace, name] = f.split("/");
|
|
36
89
|
if (!name) {
|
|
@@ -145,7 +198,7 @@ async function readProviderSchema(target) {
|
|
|
145
198
|
const outdir = process.cwd();
|
|
146
199
|
const filePath = path.join(outdir, "main.tf.json");
|
|
147
200
|
await fs.writeFile(filePath, JSON.stringify(config));
|
|
148
|
-
await (
|
|
201
|
+
await terraformInitWithRetry({ cwd: outdir });
|
|
149
202
|
providerSchema = JSON.parse(await (0, commons_1.exec)(terraformBinaryName, ["providers", "schema", "-json"], {
|
|
150
203
|
cwd: outdir,
|
|
151
204
|
}));
|
|
@@ -222,4 +275,4 @@ async function readModuleSchema(target) {
|
|
|
222
275
|
return moduleSchema;
|
|
223
276
|
}
|
|
224
277
|
exports.readModuleSchema = readModuleSchema;
|
|
225
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"provider-schema.js","sourceRoot":"","sources":["../src/provider-schema.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) HashiCorp, Inc\n// SPDX-License-Identifier: MPL-2.0\nimport * as fs from \"fs-extra\";\nimport * as path from \"path\";\n\nimport { convertFiles } from \"@cdktn/hcl2json\";\nimport {\n  Attribute,\n  AttributeType,\n  Block,\n  ConstructsMakerModuleTarget,\n  ConstructsMakerProviderTarget,\n  Input,\n  ModuleIndex,\n  ModuleSchema,\n  ProviderSchema,\n  TerraformModuleConstraint,\n  VersionSchema,\n  exec,\n  isNestedTypeAttribute,\n  withTempDir,\n} from \"@cdktn/commons\";\n\nconst terraformBinaryName = process.env.TERRAFORM_BINARY_NAME || \"terraform\";\n\n/**\n * Fully Qualified provider name in the format:\n * like e.g. registry.terraform.io/hashicorp/aws\n */\nexport type FQPN = string & { __type: \"FullyQualifiedProviderName\" };\nexport type ProviderHostname = string & { __type: \"ProviderHostname\" };\nexport type ProviderNamespace = string & { __type: \"ProviderNamespace\" };\nexport type ProviderName = string & { __type: \"ProviderName\" };\n\nexport const parseFQPN = (f: FQPN) => {\n  const [hostname, namespace, name] = f.split(\"/\");\n  if (!name) {\n    throw new Error(`can't handle ${f}`);\n  }\n  return { hostname, namespace, name } as {\n    hostname: ProviderHostname;\n    namespace: ProviderNamespace;\n    name: ProviderName;\n  };\n};\n\nconst unwrapIfArray = <T>(item: T | T[]): T =>\n  Array.isArray(item) ? item[0] : item;\n\nconst transformVariables = (variables: any) => {\n  const result: Input[] = [];\n\n  if (!variables) return result;\n\n  for (const name of Object.keys(variables)) {\n    const variable = unwrapIfArray(variables[name]);\n    let variableType: string;\n\n    if (\n      // eslint-disable-next-line no-prototype-builtins\n      variable.hasOwnProperty(\"type\") == false &&\n      // eslint-disable-next-line no-prototype-builtins\n      variable.hasOwnProperty(\"default\") == true\n    ) {\n      switch (typeof variable[\"default\"]) {\n        case \"boolean\":\n          variableType = \"bool\";\n          break;\n        case \"number\":\n          variableType = \"number\";\n          break;\n        default:\n          variableType = \"any\";\n      }\n    } else {\n      const rawVariableType = variable[\"type\"];\n      if (\n        typeof rawVariableType === \"string\" &&\n        rawVariableType.startsWith(\"${\") &&\n        rawVariableType.endsWith(\"}\") &&\n        !rawVariableType.includes(\"\\n\")\n      ) {\n        variableType = rawVariableType.slice(2, -1);\n      } else {\n        variableType = \"any\";\n      }\n    }\n\n    const item: Input = {\n      name,\n      type: variableType,\n      description: variable[\"description\"],\n      // eslint-disable-next-line no-prototype-builtins\n      required: variable.hasOwnProperty(\"default\") == false,\n    };\n\n    if (!item.required) {\n      item[\"default\"] = variable[\"default\"];\n    }\n\n    result.push(item);\n  }\n\n  return result;\n};\n\nconst transformOutputs = (outputs: any) => {\n  const result = [];\n\n  if (outputs) {\n    for (const name of Object.keys(outputs)) {\n      const output = unwrapIfArray(outputs[name]);\n\n      const item: any = {\n        name,\n        description: output[\"description\"],\n      };\n\n      result.push(item);\n    }\n  }\n\n  return result;\n};\n\nconst harvestModuleSchema = async (\n  workingDirectory: string,\n  modules: string[],\n): Promise<Record<string, any>> => {\n  const fileName = path.join(\n    workingDirectory,\n    \".terraform\",\n    \"modules\",\n    \"modules.json\",\n  );\n  const result: Record<string, any> = {};\n\n  if (!fs.existsSync(fileName)) {\n    throw new Error(\n      `Modules were not generated properly - couldn't find ${fileName}`,\n    );\n  }\n\n  const moduleIndex = JSON.parse(\n    fs.readFileSync(fileName, \"utf-8\"),\n  ) as ModuleIndex;\n\n  for (const mod of modules) {\n    const m = moduleIndex.Modules.find((other) => mod === other.Key);\n\n    if (!m) {\n      throw new Error(`Couldn't find ${m}`);\n    }\n\n    const parsed = await convertFiles(path.join(workingDirectory, m.Dir));\n\n    if (!parsed) {\n      throw new Error(\n        `Modules were not generated properly - couldn't parse ${m.Dir}`,\n      );\n    }\n\n    const schema: ModuleSchema = {\n      inputs: transformVariables(parsed.variable),\n      outputs: transformOutputs(parsed.output),\n      name: mod,\n    };\n\n    result[mod] = schema;\n  }\n\n  return result;\n};\n\nexport interface TerraformConfig {\n  provider?: { [name: string]: Record<string, any> };\n  terraform: {\n    required_providers?: {\n      [name: string]: { source?: string; version?: string };\n    };\n  };\n  module?: { [name: string]: { source: string; version?: string } };\n}\n\nexport async function readProviderSchema(\n  target: ConstructsMakerProviderTarget,\n) {\n  const config: TerraformConfig = {\n    provider: {},\n    terraform: {\n      required_providers: {},\n    },\n  };\n\n  config.provider![target.name] = {};\n  config.terraform.required_providers![target.name] = {\n    version: target.version,\n    source: target.source,\n  };\n\n  let providerSchema: ProviderSchema = { format_version: \"0.1\" };\n\n  await withTempDir(\"fetchProviderSchema\", async () => {\n    const outdir = process.cwd();\n    const filePath = path.join(outdir, \"main.tf.json\");\n    await fs.writeFile(filePath, JSON.stringify(config));\n\n    await exec(terraformBinaryName, [\"init\"], { cwd: outdir });\n    providerSchema = JSON.parse(\n      await exec(terraformBinaryName, [\"providers\", \"schema\", \"-json\"], {\n        cwd: outdir,\n      }),\n    ) as ProviderSchema;\n\n    const versionSchema = JSON.parse(\n      await exec(terraformBinaryName, [\"version\", \"-json\"], {\n        cwd: outdir,\n      }),\n    ) as VersionSchema;\n\n    providerSchema.provider_versions = versionSchema.provider_selections;\n  });\n\n  return sanitizeProviderSchema(providerSchema);\n}\n\n// The providers have some potential bugs that we want to pro-actively\n// fix here so that the rest of the code can assume a consistent schema.\nexport function sanitizeProviderSchema(schema: ProviderSchema): ProviderSchema {\n  // Mainly some attributes are \"doubled\", e.g. [\"list\", \"string\", \"list\", \"string\"]\n  // instead of [\"list\", \"string\"]\n  function attributeDoublingFix(attribute: Attribute): Attribute {\n    if (isNestedTypeAttribute(attribute) || !Array.isArray(attribute.type)) {\n      return attribute;\n    }\n\n    const type =\n      attribute.type.length === 2\n        ? attribute.type\n        : (attribute.type as string[]).slice(0, 2); // The types tell us this can't happen, reality begs to differ\n\n    attribute.type = type as AttributeType;\n    return attribute;\n  }\n\n  // Mutates block with the fix\n  function sanitizeBlock(block: Block) {\n    Object.values(block.attributes || {}).forEach(attributeDoublingFix);\n    Object.values(block.block_types || {}).forEach((blockType) => {\n      sanitizeBlock(blockType.block);\n    });\n  }\n\n  Object.values(schema.provider_schemas || {}).forEach((provider) => {\n    const entities = [\n      provider.provider,\n      ...Object.values(provider.resource_schemas || {}),\n      ...Object.values(provider.data_source_schemas || {}),\n    ];\n\n    entities.forEach((entity) => {\n      sanitizeBlock(entity.block);\n    });\n  });\n\n  return schema;\n}\n\nexport async function readModuleSchema(target: ConstructsMakerModuleTarget) {\n  let moduleSchema: Record<string, ModuleSchema> = {};\n\n  await withTempDir(\"fetchSchema\", async () => {\n    const config: TerraformConfig = {\n      terraform: {},\n    };\n\n    if (!config.module) config.module = {};\n    let source: string = target.source;\n\n    const localSource = (target.constraint as TerraformModuleConstraint)\n      .localSourceAbsolutePath;\n    if (localSource) {\n      // create relative path to module in the user project\n      source = path.relative(process.cwd(), localSource);\n    }\n\n    config.module[target.moduleKey] = { source: source };\n    if (target.version) {\n      config.module[target.moduleKey][\"version\"] = target.version;\n    }\n\n    const outdir = process.cwd();\n    const filePath = path.join(outdir, \"main.tf.json\");\n    await fs.writeFile(filePath, JSON.stringify(config));\n\n    await exec(terraformBinaryName, [\"get\"], { cwd: outdir });\n    if (config.module) {\n      moduleSchema = await harvestModuleSchema(\n        outdir,\n        Object.keys(config.module),\n      );\n    }\n  });\n\n  return moduleSchema;\n}\n"]}
|
|
278
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"provider-schema.js","sourceRoot":"","sources":["../src/provider-schema.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) HashiCorp, Inc\n// SPDX-License-Identifier: MPL-2.0\nimport * as fs from \"fs-extra\";\nimport * as path from \"path\";\n\nimport { convertFiles } from \"@cdktn/hcl2json\";\nimport {\n  Attribute,\n  AttributeType,\n  Block,\n  ConstructsMakerModuleTarget,\n  ConstructsMakerProviderTarget,\n  Input,\n  ModuleIndex,\n  ModuleSchema,\n  ProviderSchema,\n  TerraformModuleConstraint,\n  VersionSchema,\n  exec,\n  isNestedTypeAttribute,\n  withTempDir,\n} from \"@cdktn/commons\";\n\nconst terraformBinaryName = process.env.TERRAFORM_BINARY_NAME || \"terraform\";\n\n// Provider binaries are downloaded from GitHub release CDNs during\n// `terraform init`, which intermittently returns 5xx. Retry on those\n// signatures only — real config/schema errors should still fail fast.\nconst TRANSIENT_INIT_ERROR_PATTERNS = [\n  /\\b50[234]\\b/,\n  /Bad Gateway/i,\n  /Service Unavailable/i,\n  /Gateway Timeout/i,\n  /ECONNRESET/i,\n  /ETIMEDOUT/i,\n  /EAI_AGAIN/i,\n  /unexpected EOF/i,\n];\n\n/**\n * Runs `terraform init` and retries on transient HTTP/network failures.\n *\n * Provider binaries are pulled from GitHub's release CDN, which\n * intermittently returns 5xx. The retry only fires when stderr matches a\n * known transient pattern — real errors (bad version, schema parse, etc.)\n * still fail on the first attempt.\n *\n * @param options - spawn options forwarded to `exec` (`cwd` is required so\n *   `terraform init` runs inside the temp dir holding `main.tf.json`)\n * @param maxAttempts - total attempts including the initial call; the number\n *   of retries is `maxAttempts - 1`. Defaults to 3 (1 initial + 2 retries)\n * @param baseDelayMs - delay before the first retry. Tests pass `0` to skip\n *   waits. Defaults to 1000ms\n * @param backoffMultiplier - factor applied to the delay on each subsequent\n *   retry (so retries wait `baseDelayMs * multiplier^(attempt-1)`). Defaults\n *   to 2 (1s → 2s → 4s …)\n * @returns stdout from the successful `terraform init` invocation\n * @internal exposed for testing\n */\nexport async function terraformInitWithRetry(\n  options: { cwd: string },\n  maxAttempts = 3,\n  baseDelayMs = 1000,\n  backoffMultiplier = 2,\n): Promise<string> {\n  for (let attempt = 1; attempt <= maxAttempts; attempt++) {\n    try {\n      return await exec(terraformBinaryName, [\"init\"], options);\n    } catch (error: any) {\n      const stderr: string = error?.stderr ?? \"\";\n      const transient = TRANSIENT_INIT_ERROR_PATTERNS.some((p) =>\n        p.test(stderr),\n      );\n      if (!transient || attempt === maxAttempts) {\n        throw error;\n      }\n      const delayMs = baseDelayMs * backoffMultiplier ** (attempt - 1);\n      console.error(\n        `terraform init failed with a transient error, retrying in ${delayMs}ms (attempt ${attempt}/${maxAttempts - 1})`,\n      );\n      await new Promise((resolve) => setTimeout(resolve, delayMs));\n    }\n  }\n  throw new Error(\"unreachable\");\n}\n\n/**\n * Fully Qualified provider name in the format:\n * like e.g. registry.terraform.io/hashicorp/aws\n */\nexport type FQPN = string & { __type: \"FullyQualifiedProviderName\" };\nexport type ProviderHostname = string & { __type: \"ProviderHostname\" };\nexport type ProviderNamespace = string & { __type: \"ProviderNamespace\" };\nexport type ProviderName = string & { __type: \"ProviderName\" };\n\nexport const parseFQPN = (f: FQPN) => {\n  const [hostname, namespace, name] = f.split(\"/\");\n  if (!name) {\n    throw new Error(`can't handle ${f}`);\n  }\n  return { hostname, namespace, name } as {\n    hostname: ProviderHostname;\n    namespace: ProviderNamespace;\n    name: ProviderName;\n  };\n};\n\nconst unwrapIfArray = <T>(item: T | T[]): T =>\n  Array.isArray(item) ? item[0] : item;\n\nconst transformVariables = (variables: any) => {\n  const result: Input[] = [];\n\n  if (!variables) return result;\n\n  for (const name of Object.keys(variables)) {\n    const variable = unwrapIfArray(variables[name]);\n    let variableType: string;\n\n    if (\n      // eslint-disable-next-line no-prototype-builtins\n      variable.hasOwnProperty(\"type\") == false &&\n      // eslint-disable-next-line no-prototype-builtins\n      variable.hasOwnProperty(\"default\") == true\n    ) {\n      switch (typeof variable[\"default\"]) {\n        case \"boolean\":\n          variableType = \"bool\";\n          break;\n        case \"number\":\n          variableType = \"number\";\n          break;\n        default:\n          variableType = \"any\";\n      }\n    } else {\n      const rawVariableType = variable[\"type\"];\n      if (\n        typeof rawVariableType === \"string\" &&\n        rawVariableType.startsWith(\"${\") &&\n        rawVariableType.endsWith(\"}\") &&\n        !rawVariableType.includes(\"\\n\")\n      ) {\n        variableType = rawVariableType.slice(2, -1);\n      } else {\n        variableType = \"any\";\n      }\n    }\n\n    const item: Input = {\n      name,\n      type: variableType,\n      description: variable[\"description\"],\n      // eslint-disable-next-line no-prototype-builtins\n      required: variable.hasOwnProperty(\"default\") == false,\n    };\n\n    if (!item.required) {\n      item[\"default\"] = variable[\"default\"];\n    }\n\n    result.push(item);\n  }\n\n  return result;\n};\n\nconst transformOutputs = (outputs: any) => {\n  const result = [];\n\n  if (outputs) {\n    for (const name of Object.keys(outputs)) {\n      const output = unwrapIfArray(outputs[name]);\n\n      const item: any = {\n        name,\n        description: output[\"description\"],\n      };\n\n      result.push(item);\n    }\n  }\n\n  return result;\n};\n\nconst harvestModuleSchema = async (\n  workingDirectory: string,\n  modules: string[],\n): Promise<Record<string, any>> => {\n  const fileName = path.join(\n    workingDirectory,\n    \".terraform\",\n    \"modules\",\n    \"modules.json\",\n  );\n  const result: Record<string, any> = {};\n\n  if (!fs.existsSync(fileName)) {\n    throw new Error(\n      `Modules were not generated properly - couldn't find ${fileName}`,\n    );\n  }\n\n  const moduleIndex = JSON.parse(\n    fs.readFileSync(fileName, \"utf-8\"),\n  ) as ModuleIndex;\n\n  for (const mod of modules) {\n    const m = moduleIndex.Modules.find((other) => mod === other.Key);\n\n    if (!m) {\n      throw new Error(`Couldn't find ${m}`);\n    }\n\n    const parsed = await convertFiles(path.join(workingDirectory, m.Dir));\n\n    if (!parsed) {\n      throw new Error(\n        `Modules were not generated properly - couldn't parse ${m.Dir}`,\n      );\n    }\n\n    const schema: ModuleSchema = {\n      inputs: transformVariables(parsed.variable),\n      outputs: transformOutputs(parsed.output),\n      name: mod,\n    };\n\n    result[mod] = schema;\n  }\n\n  return result;\n};\n\nexport interface TerraformConfig {\n  provider?: { [name: string]: Record<string, any> };\n  terraform: {\n    required_providers?: {\n      [name: string]: { source?: string; version?: string };\n    };\n  };\n  module?: { [name: string]: { source: string; version?: string } };\n}\n\nexport async function readProviderSchema(\n  target: ConstructsMakerProviderTarget,\n) {\n  const config: TerraformConfig = {\n    provider: {},\n    terraform: {\n      required_providers: {},\n    },\n  };\n\n  config.provider![target.name] = {};\n  config.terraform.required_providers![target.name] = {\n    version: target.version,\n    source: target.source,\n  };\n\n  let providerSchema: ProviderSchema = { format_version: \"0.1\" };\n\n  await withTempDir(\"fetchProviderSchema\", async () => {\n    const outdir = process.cwd();\n    const filePath = path.join(outdir, \"main.tf.json\");\n    await fs.writeFile(filePath, JSON.stringify(config));\n\n    await terraformInitWithRetry({ cwd: outdir });\n    providerSchema = JSON.parse(\n      await exec(terraformBinaryName, [\"providers\", \"schema\", \"-json\"], {\n        cwd: outdir,\n      }),\n    ) as ProviderSchema;\n\n    const versionSchema = JSON.parse(\n      await exec(terraformBinaryName, [\"version\", \"-json\"], {\n        cwd: outdir,\n      }),\n    ) as VersionSchema;\n\n    providerSchema.provider_versions = versionSchema.provider_selections;\n  });\n\n  return sanitizeProviderSchema(providerSchema);\n}\n\n// The providers have some potential bugs that we want to pro-actively\n// fix here so that the rest of the code can assume a consistent schema.\nexport function sanitizeProviderSchema(schema: ProviderSchema): ProviderSchema {\n  // Mainly some attributes are \"doubled\", e.g. [\"list\", \"string\", \"list\", \"string\"]\n  // instead of [\"list\", \"string\"]\n  function attributeDoublingFix(attribute: Attribute): Attribute {\n    if (isNestedTypeAttribute(attribute) || !Array.isArray(attribute.type)) {\n      return attribute;\n    }\n\n    const type =\n      attribute.type.length === 2\n        ? attribute.type\n        : (attribute.type as string[]).slice(0, 2); // The types tell us this can't happen, reality begs to differ\n\n    attribute.type = type as AttributeType;\n    return attribute;\n  }\n\n  // Mutates block with the fix\n  function sanitizeBlock(block: Block) {\n    Object.values(block.attributes || {}).forEach(attributeDoublingFix);\n    Object.values(block.block_types || {}).forEach((blockType) => {\n      sanitizeBlock(blockType.block);\n    });\n  }\n\n  Object.values(schema.provider_schemas || {}).forEach((provider) => {\n    const entities = [\n      provider.provider,\n      ...Object.values(provider.resource_schemas || {}),\n      ...Object.values(provider.data_source_schemas || {}),\n    ];\n\n    entities.forEach((entity) => {\n      sanitizeBlock(entity.block);\n    });\n  });\n\n  return schema;\n}\n\nexport async function readModuleSchema(target: ConstructsMakerModuleTarget) {\n  let moduleSchema: Record<string, ModuleSchema> = {};\n\n  await withTempDir(\"fetchSchema\", async () => {\n    const config: TerraformConfig = {\n      terraform: {},\n    };\n\n    if (!config.module) config.module = {};\n    let source: string = target.source;\n\n    const localSource = (target.constraint as TerraformModuleConstraint)\n      .localSourceAbsolutePath;\n    if (localSource) {\n      // create relative path to module in the user project\n      source = path.relative(process.cwd(), localSource);\n    }\n\n    config.module[target.moduleKey] = { source: source };\n    if (target.version) {\n      config.module[target.moduleKey][\"version\"] = target.version;\n    }\n\n    const outdir = process.cwd();\n    const filePath = path.join(outdir, \"main.tf.json\");\n    await fs.writeFile(filePath, JSON.stringify(config));\n\n    await exec(terraformBinaryName, [\"get\"], { cwd: outdir });\n    if (config.module) {\n      moduleSchema = await harvestModuleSchema(\n        outdir,\n        Object.keys(config.module),\n      );\n    }\n  });\n\n  return moduleSchema;\n}\n"]}
|
package/eslint.config.mjs
CHANGED
|
@@ -3,77 +3,4 @@
|
|
|
3
3
|
* SPDX-License-Identifier: MPL-2.0
|
|
4
4
|
*/
|
|
5
5
|
|
|
6
|
-
|
|
7
|
-
import eslint from "@eslint/js";
|
|
8
|
-
import tseslint from "typescript-eslint";
|
|
9
|
-
import tsParser from "@typescript-eslint/parser";
|
|
10
|
-
import react from "eslint-plugin-react";
|
|
11
|
-
import prettierConfig from "eslint-plugin-prettier/recommended";
|
|
12
|
-
import path from "node:path";
|
|
13
|
-
import { fileURLToPath } from "node:url";
|
|
14
|
-
import js from "@eslint/js";
|
|
15
|
-
import { FlatCompat } from "@eslint/eslintrc";
|
|
16
|
-
|
|
17
|
-
const __filename = fileURLToPath(import.meta.url);
|
|
18
|
-
const __dirname = path.dirname(__filename);
|
|
19
|
-
const compat = new FlatCompat({
|
|
20
|
-
baseDirectory: __dirname,
|
|
21
|
-
recommendedConfig: js.configs.recommended,
|
|
22
|
-
allConfig: js.configs.all,
|
|
23
|
-
});
|
|
24
|
-
|
|
25
|
-
export default [
|
|
26
|
-
{
|
|
27
|
-
ignores: [
|
|
28
|
-
"**/node_modules",
|
|
29
|
-
"**/dist",
|
|
30
|
-
"**/coverage",
|
|
31
|
-
"**/*.d.ts",
|
|
32
|
-
"**/*.js",
|
|
33
|
-
"**/*.mjs",
|
|
34
|
-
],
|
|
35
|
-
},
|
|
36
|
-
...tseslint.config(eslint.configs.recommended, tseslint.configs.recommended),
|
|
37
|
-
{
|
|
38
|
-
settings: {
|
|
39
|
-
react: {
|
|
40
|
-
version: "detect",
|
|
41
|
-
},
|
|
42
|
-
},
|
|
43
|
-
},
|
|
44
|
-
react.configs.flat.recommended,
|
|
45
|
-
...fixupConfigRules(compat.extends("plugin:react-hooks/recommended")).map(
|
|
46
|
-
(config) => ({
|
|
47
|
-
...config,
|
|
48
|
-
files: ["**/*.ts", "**/*.tsx"],
|
|
49
|
-
}),
|
|
50
|
-
),
|
|
51
|
-
{
|
|
52
|
-
files: ["**/*.ts", "**/*.tsx"],
|
|
53
|
-
|
|
54
|
-
languageOptions: {
|
|
55
|
-
parser: tsParser,
|
|
56
|
-
parserOptions: {
|
|
57
|
-
tsconfigRootDir: import.meta.dirname,
|
|
58
|
-
},
|
|
59
|
-
},
|
|
60
|
-
|
|
61
|
-
rules: {
|
|
62
|
-
"@typescript-eslint/no-explicit-any": 0,
|
|
63
|
-
"@typescript-eslint/explicit-function-return-type": 0,
|
|
64
|
-
"@typescript-eslint/no-use-before-define": 0,
|
|
65
|
-
"@typescript-eslint/explicit-module-boundary-types": 0,
|
|
66
|
-
"@typescript-eslint/no-var-requires": 0,
|
|
67
|
-
"react/no-unescaped-entities": 0,
|
|
68
|
-
"no-sequences": "error",
|
|
69
|
-
|
|
70
|
-
"no-irregular-whitespace": [
|
|
71
|
-
"error",
|
|
72
|
-
{
|
|
73
|
-
skipTemplates: true,
|
|
74
|
-
},
|
|
75
|
-
],
|
|
76
|
-
},
|
|
77
|
-
},
|
|
78
|
-
prettierConfig,
|
|
79
|
-
];
|
|
6
|
+
export { default } from "../../../eslint.config.mjs";
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@cdktn/provider-schema",
|
|
3
|
-
"version": "0.23.0-pre.
|
|
3
|
+
"version": "0.23.0-pre.40",
|
|
4
4
|
"publishConfig": {
|
|
5
5
|
"access": "public"
|
|
6
6
|
},
|
|
@@ -40,30 +40,22 @@
|
|
|
40
40
|
],
|
|
41
41
|
"license": "MPL-2.0",
|
|
42
42
|
"dependencies": {
|
|
43
|
-
"@cdktn/commons": "0.23.0-pre.
|
|
44
|
-
"@cdktn/hcl2json": "0.23.0-pre.
|
|
43
|
+
"@cdktn/commons": "0.23.0-pre.40",
|
|
44
|
+
"@cdktn/hcl2json": "0.23.0-pre.40",
|
|
45
45
|
"deepmerge": "4.3.1",
|
|
46
46
|
"fs-extra": "11.3.4"
|
|
47
47
|
},
|
|
48
48
|
"devDependencies": {
|
|
49
|
-
"@eslint/compat": "^1.4.1",
|
|
50
|
-
"@eslint/eslintrc": "^3.3.5",
|
|
51
|
-
"@eslint/js": "^9.39.4",
|
|
52
49
|
"@types/follow-redirects": "1.14.4",
|
|
53
50
|
"@types/fs-extra": "11.0.4",
|
|
54
51
|
"@types/json-stable-stringify": "1.2.0",
|
|
55
52
|
"@types/uuid": "9.0.8",
|
|
56
|
-
"@typescript-eslint/parser": "^8.58.0",
|
|
57
|
-
"eslint": "^9.39.4",
|
|
58
|
-
"eslint-plugin-prettier": "^5.5.5",
|
|
59
|
-
"eslint-plugin-react": "7.37.5",
|
|
60
53
|
"jest": "^30.3.0",
|
|
61
54
|
"json-stable-stringify": "1.3.0",
|
|
62
55
|
"lint-staged": "^15.5.2",
|
|
63
56
|
"ts-jest": "29.4.9",
|
|
64
57
|
"tsc-files": "1.1.4",
|
|
65
|
-
"typescript": "5.4.5"
|
|
66
|
-
"typescript-eslint": "^8.58.0"
|
|
58
|
+
"typescript": "5.4.5"
|
|
67
59
|
},
|
|
68
60
|
"lint-staged": {
|
|
69
61
|
"src/**/*.{ts,tsx}": "tsc-files ambient.d.ts --noEmit"
|