2019-06-01 06:01:01 +00:00
|
|
|
// Copyright 2016-2018, Pulumi Corporation.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2021-08-10 18:31:59 +00:00
|
|
|
/* eslint-disable */
|
2019-06-01 06:01:01 +00:00
|
|
|
|
|
|
|
import * as assert from "assert";
|
2022-05-30 08:31:28 +00:00
|
|
|
import { all } from "../output";
|
2019-06-01 06:01:01 +00:00
|
|
|
import * as runtime from "../runtime";
|
2023-02-27 18:21:31 +00:00
|
|
|
import {
|
2023-04-28 22:27:10 +00:00
|
|
|
allAliases,
|
|
|
|
createUrn,
|
|
|
|
ProviderResource,
|
|
|
|
CustomResource,
|
|
|
|
ComponentResource,
|
|
|
|
ComponentResourceOptions,
|
|
|
|
CustomResourceOptions,
|
|
|
|
DependencyProviderResource,
|
2023-02-27 18:21:31 +00:00
|
|
|
} from "../resource";
|
2019-06-01 06:01:01 +00:00
|
|
|
|
|
|
|
class MyResource extends ComponentResource {
|
2023-02-27 18:21:31 +00:00
|
|
|
constructor(name: string, opts?: ComponentResourceOptions) {
|
2019-06-01 06:01:01 +00:00
|
|
|
super("my:mod:MyResource", name, {}, opts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class MyParentResource extends ComponentResource {
|
|
|
|
child: MyResource;
|
2023-02-27 18:21:31 +00:00
|
|
|
constructor(name: string, opts?: ComponentResourceOptions) {
|
2019-06-01 06:01:01 +00:00
|
|
|
super("my:mod:MyParentResource", name, {}, opts);
|
|
|
|
this.child = new MyResource(`${name}-child`, { parent: this });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
describe("createUrn", () => {
|
|
|
|
before(() => {
|
|
|
|
runtime._setProject("myproject");
|
2020-10-21 17:21:47 +00:00
|
|
|
runtime._setStack("mystack");
|
2019-06-01 06:01:01 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
after(() => {
|
|
|
|
runtime._setProject(undefined);
|
|
|
|
runtime._setStack(undefined);
|
2023-04-28 22:27:10 +00:00
|
|
|
});
|
2019-06-01 06:01:01 +00:00
|
|
|
|
2022-11-09 13:01:25 +00:00
|
|
|
it("handles name and type", async () => {
|
2019-06-01 06:01:01 +00:00
|
|
|
const urn = await createUrn("n", "t").promise();
|
2020-10-21 17:21:47 +00:00
|
|
|
assert.strictEqual(urn, "urn:pulumi:mystack::myproject::t::n");
|
2022-11-09 13:01:25 +00:00
|
|
|
});
|
2019-06-01 06:01:01 +00:00
|
|
|
|
2022-11-09 13:01:25 +00:00
|
|
|
it("handles name and type and parent", async () => {
|
2019-06-01 06:01:01 +00:00
|
|
|
const res = new MyResource("myres");
|
|
|
|
const urn = await createUrn("n", "t", res).promise();
|
2020-10-21 17:21:47 +00:00
|
|
|
assert.strictEqual(urn, "urn:pulumi:mystack::myproject::my:mod:MyResource$t::n");
|
2022-11-09 13:01:25 +00:00
|
|
|
});
|
2019-06-01 06:01:01 +00:00
|
|
|
|
2022-11-09 13:01:25 +00:00
|
|
|
it("handles name and type and parent with parent", async () => {
|
2019-06-01 06:01:01 +00:00
|
|
|
const res = new MyParentResource("myres");
|
|
|
|
const urn = await createUrn("n", "t", res.child).promise();
|
2020-10-21 17:21:47 +00:00
|
|
|
assert.strictEqual(urn, "urn:pulumi:mystack::myproject::my:mod:MyParentResource$my:mod:MyResource$t::n");
|
2022-11-09 13:01:25 +00:00
|
|
|
});
|
2020-10-21 17:21:47 +00:00
|
|
|
});
|
2021-07-27 13:50:24 +00:00
|
|
|
|
2021-12-29 17:07:55 +00:00
|
|
|
class TestResource extends ComponentResource {
|
|
|
|
constructor(name: string, opts?: CustomResourceOptions) {
|
|
|
|
super("test:resource:type", name, {}, opts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
describe("allAliases", () => {
|
|
|
|
before(() => {
|
|
|
|
runtime._setProject("project");
|
|
|
|
runtime._setStack("stack");
|
|
|
|
});
|
|
|
|
|
|
|
|
after(() => {
|
|
|
|
runtime._setProject(undefined);
|
|
|
|
runtime._setStack(undefined);
|
2023-04-28 22:27:10 +00:00
|
|
|
});
|
2021-12-29 17:07:55 +00:00
|
|
|
|
|
|
|
const testCases = [
|
|
|
|
{
|
|
|
|
name: "no aliases",
|
|
|
|
parentAliases: [],
|
|
|
|
childAliases: [],
|
|
|
|
results: [],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "one child alias (type), no parent aliases",
|
|
|
|
parentAliases: [],
|
|
|
|
childAliases: [{ type: "test:resource:child2" }],
|
|
|
|
results: ["urn:pulumi:stack::project::test:resource:type$test:resource:child2::myres-child"],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "one child alias (name), no parent aliases",
|
|
|
|
parentAliases: [],
|
|
|
|
childAliases: [{ name: "child2" }],
|
|
|
|
results: ["urn:pulumi:stack::project::test:resource:type$test:resource:child::child2"],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "one child alias (name), one parent alias (type)",
|
2023-02-27 18:21:31 +00:00
|
|
|
parentAliases: [{ type: "test:resource:type3" }],
|
|
|
|
childAliases: [{ name: "myres-child2" }],
|
|
|
|
results: [
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres-child2",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type3$test:resource:child::myres-child",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type3$test:resource:child::myres-child2",
|
2021-12-29 17:07:55 +00:00
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "one child alias (name), one parent alias (name)",
|
2023-02-27 18:21:31 +00:00
|
|
|
parentAliases: [{ name: "myres2" }],
|
|
|
|
childAliases: [{ name: "myres-child2" }],
|
|
|
|
results: [
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres-child2",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres2-child",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres2-child2",
|
2021-12-29 17:07:55 +00:00
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "two child aliases, three parent aliases",
|
|
|
|
parentAliases: [{ name: "myres2" }, { type: "test:resource:type3" }, { name: "myres3" }],
|
|
|
|
childAliases: [{ name: "myres-child2" }, { type: "test:resource:child2" }],
|
|
|
|
results: [
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres-child2",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child2::myres-child",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres2-child",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres2-child2",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child2::myres2-child",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type3$test:resource:child::myres-child",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type3$test:resource:child::myres-child2",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type3$test:resource:child2::myres-child",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres3-child",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres3-child2",
|
|
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child2::myres3-child",
|
|
|
|
],
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
for (const testCase of testCases) {
|
2022-11-09 13:01:25 +00:00
|
|
|
it(testCase.name, async () => {
|
2021-12-29 17:07:55 +00:00
|
|
|
const res = new TestResource("myres", { aliases: testCase.parentAliases });
|
|
|
|
const aliases = allAliases(testCase.childAliases, "myres-child", "test:resource:child", res, "myres");
|
|
|
|
assert.strictEqual(aliases.length, testCase.results.length);
|
|
|
|
const aliasURNs = await all(aliases).promise();
|
|
|
|
for (let i = 0; i < aliasURNs.length; i++) {
|
|
|
|
assert.strictEqual(aliasURNs[i], testCase.results[i]);
|
|
|
|
}
|
2022-11-09 13:01:25 +00:00
|
|
|
});
|
2021-12-29 17:07:55 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2021-07-27 13:50:24 +00:00
|
|
|
describe("DependencyProviderResource", () => {
|
|
|
|
describe("getPackage", () => {
|
|
|
|
it("returns the expected package", () => {
|
2023-04-28 22:27:10 +00:00
|
|
|
const res = new DependencyProviderResource(
|
|
|
|
"urn:pulumi:stack::project::pulumi:providers:aws::default_4_13_0",
|
|
|
|
);
|
2021-07-27 13:50:24 +00:00
|
|
|
assert.strictEqual(res.getPackage(), "aws");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2023-02-27 18:21:31 +00:00
|
|
|
|
[sdk/nodejs] Fix provider for resource methods (#13796)
The `Resource` class in the Node.js SDK has the following internal
property:
```typescript
/** @internal */
readonly __prov?: ProviderResource;
```
When a resource is created, the provider specified for the resource is
stored in this property. If it is set, it is passed along in the `Call`
request when a method is called on the resource.
Prior to #13282, the property was only set for custom resources in
`Resource`'s constructor:
```typescript
this.__prov = custom ? opts.provider : undefined;
```
With #13282, it was changed to also store the value for remote
components:
```diff
- this.__prov = custom ? opts.provider : undefined;
+ this.__prov = custom || remote ? opts.provider : undefined;
```
This regressed the behavior when calling a method on a remote component
that had an explicit provider that wasn't the component provider, but
some other provider (e.g. AWS provider) specified as:
```typescript
const component = new MyRemoteComponent("comp", {
}, { provider: awsProvider });
```
The `awsProvider` was being stored in `Resource.__prov`, and when making
the method call on the resource, it would try to invoke `Call` on the
AWS provider, rather than calling the remote component provider's
`Call`, which resulted in an error.
Note that specifying the AWS provider using the more verbose `providers:
[awsProvider]` works around the issue.
The fix is to only set `__prov` if the provider's package is the same as
the resource's package. Otherwise, don't set it, because the user is
specifying a provider with the `provider: awsProvider` syntax as
shorthand for `providers: [awsProvider]`.
Fixes #13777
2023-08-30 14:49:53 +00:00
|
|
|
describe("CustomResource", () => {
|
|
|
|
runtime.setMocks({
|
|
|
|
call: (_) => {
|
|
|
|
throw new Error("unexpected call");
|
|
|
|
},
|
|
|
|
newResource: (args) => {
|
|
|
|
return { id: `${args.name}_id`, state: {} };
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://github.com/pulumi/pulumi/issues/13777
|
|
|
|
it("saves provider with same package as the resource in __prov", async () => {
|
|
|
|
const provider = new MyProvider("prov");
|
|
|
|
const custom = new MyCustomResource("custom", { provider: provider });
|
|
|
|
assert.strictEqual(custom.__prov, provider);
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://github.com/pulumi/pulumi/issues/13777
|
|
|
|
it("does not save provider with different package as the resource in __prov", async () => {
|
|
|
|
const provider = new MyOtherProvider("prov");
|
|
|
|
const custom = new MyCustomResource("custom", { provider: provider });
|
|
|
|
assert.strictEqual(custom.__prov, undefined);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-02-27 18:21:31 +00:00
|
|
|
describe("ComponentResource", () => {
|
|
|
|
runtime.setMocks({
|
|
|
|
call: (_) => {
|
|
|
|
throw new Error("unexpected call");
|
|
|
|
},
|
|
|
|
newResource: (args) => {
|
|
|
|
return { id: `${args.name}_id`, state: {} };
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://github.com/pulumi/pulumi/issues/12161
|
|
|
|
it("propagates provider to children", async () => {
|
|
|
|
const provider = new MyProvider("prov");
|
|
|
|
const component = new MyResource("comp", { provider: provider });
|
|
|
|
const custom = new MyCustomResource("custom", { parent: component });
|
|
|
|
assert.strictEqual(custom.__prov, provider);
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://github.com/pulumi/pulumi/issues/12161
|
|
|
|
it("propagates providers list to children", async () => {
|
|
|
|
const provider = new MyProvider("prov");
|
|
|
|
const component = new MyResource("comp", { providers: [provider] });
|
|
|
|
const custom = new MyCustomResource("custom", { parent: component });
|
|
|
|
assert.strictEqual(custom.__prov, provider);
|
2023-04-28 22:27:10 +00:00
|
|
|
});
|
|
|
|
});
|
2023-02-27 18:21:31 +00:00
|
|
|
|
[sdk/nodejs] Fix provider for resource methods (#13796)
The `Resource` class in the Node.js SDK has the following internal
property:
```typescript
/** @internal */
readonly __prov?: ProviderResource;
```
When a resource is created, the provider specified for the resource is
stored in this property. If it is set, it is passed along in the `Call`
request when a method is called on the resource.
Prior to #13282, the property was only set for custom resources in
`Resource`'s constructor:
```typescript
this.__prov = custom ? opts.provider : undefined;
```
With #13282, it was changed to also store the value for remote
components:
```diff
- this.__prov = custom ? opts.provider : undefined;
+ this.__prov = custom || remote ? opts.provider : undefined;
```
This regressed the behavior when calling a method on a remote component
that had an explicit provider that wasn't the component provider, but
some other provider (e.g. AWS provider) specified as:
```typescript
const component = new MyRemoteComponent("comp", {
}, { provider: awsProvider });
```
The `awsProvider` was being stored in `Resource.__prov`, and when making
the method call on the resource, it would try to invoke `Call` on the
AWS provider, rather than calling the remote component provider's
`Call`, which resulted in an error.
Note that specifying the AWS provider using the more verbose `providers:
[awsProvider]` works around the issue.
The fix is to only set `__prov` if the provider's package is the same as
the resource's package. Otherwise, don't set it, because the user is
specifying a provider with the `provider: awsProvider` syntax as
shorthand for `providers: [awsProvider]`.
Fixes #13777
2023-08-30 14:49:53 +00:00
|
|
|
describe("RemoteComponentResource", () => {
|
|
|
|
runtime.setMocks({
|
|
|
|
call: (_) => {
|
|
|
|
throw new Error("unexpected call");
|
|
|
|
},
|
|
|
|
newResource: (args) => {
|
|
|
|
return { id: `${args.name}_id`, state: {} };
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://github.com/pulumi/pulumi/issues/13777
|
|
|
|
it("saves provider with same package as the resource in __prov", async () => {
|
|
|
|
const provider = new MyProvider("prov");
|
|
|
|
const comp = new MyRemoteComponentResource("comp", { provider: provider });
|
|
|
|
assert.strictEqual(comp.__prov, provider);
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://github.com/pulumi/pulumi/issues/13777
|
|
|
|
it("does not save provider with different package as the resource in __prov", async () => {
|
|
|
|
const provider = new MyOtherProvider("prov");
|
|
|
|
const comp = new MyRemoteComponentResource("comp", { provider: provider });
|
|
|
|
assert.strictEqual(comp.__prov, undefined);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-02-27 18:21:31 +00:00
|
|
|
class MyProvider extends ProviderResource {
|
|
|
|
constructor(name: string) {
|
|
|
|
super("test", name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[sdk/nodejs] Fix provider for resource methods (#13796)
The `Resource` class in the Node.js SDK has the following internal
property:
```typescript
/** @internal */
readonly __prov?: ProviderResource;
```
When a resource is created, the provider specified for the resource is
stored in this property. If it is set, it is passed along in the `Call`
request when a method is called on the resource.
Prior to #13282, the property was only set for custom resources in
`Resource`'s constructor:
```typescript
this.__prov = custom ? opts.provider : undefined;
```
With #13282, it was changed to also store the value for remote
components:
```diff
- this.__prov = custom ? opts.provider : undefined;
+ this.__prov = custom || remote ? opts.provider : undefined;
```
This regressed the behavior when calling a method on a remote component
that had an explicit provider that wasn't the component provider, but
some other provider (e.g. AWS provider) specified as:
```typescript
const component = new MyRemoteComponent("comp", {
}, { provider: awsProvider });
```
The `awsProvider` was being stored in `Resource.__prov`, and when making
the method call on the resource, it would try to invoke `Call` on the
AWS provider, rather than calling the remote component provider's
`Call`, which resulted in an error.
Note that specifying the AWS provider using the more verbose `providers:
[awsProvider]` works around the issue.
The fix is to only set `__prov` if the provider's package is the same as
the resource's package. Otherwise, don't set it, because the user is
specifying a provider with the `provider: awsProvider` syntax as
shorthand for `providers: [awsProvider]`.
Fixes #13777
2023-08-30 14:49:53 +00:00
|
|
|
class MyOtherProvider extends ProviderResource {
|
|
|
|
constructor(name: string) {
|
|
|
|
super("other", name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-27 18:21:31 +00:00
|
|
|
class MyCustomResource extends CustomResource {
|
|
|
|
constructor(name: string, opts?: CustomResourceOptions) {
|
|
|
|
super("test:index:MyCustomResource", name, {}, opts);
|
|
|
|
}
|
|
|
|
}
|
[sdk/nodejs] Fix hang due to component children cycles
When a resource depends on a local component resource, rather than setting the component resource itself as a dependency, each of the component's descendants is added as a dependency. This can lead to hangs when cycles are introduced.
For example, consider the following parent/child hierarchy, where `ComponentA` is the parent of `ComponentB` and `ComponentB` is the parent of `CustomC`:
```
ComponentA
|
ComponentB
|
CustomC
```
If `ComponentB` specifies it has a dependency on `ComponentA`, the following takes place as part determining the full set of transitive dependencies:
1. `ComponentA` is a component resource so it isn't added as a dependency, its children are.
2. `ComponentA` has one child: `ComponentB`
3. `ComponentB` is a component resource so it isn't added as a dependency, its children are.
4. `ComponentB` has one child: `CustomC`, a custom resource.
5. Since `CustomC` is a custom resource, it is added to the set of dependencies.
6. We try to await its URN, but we'll never get it because `RegisterResource` hasn't yet been called for it. And we hang waiting.
To address this, skip looking at a component's children if it is the component from which the dependency is being added.
In the case of the example, at step 3 the dependency expansion will stop: we won't look at `ComponentB`'s children because we're adding the dependency from `ComponentB`.
fix
2023-03-27 01:50:24 +00:00
|
|
|
|
[sdk/nodejs] Fix provider for resource methods (#13796)
The `Resource` class in the Node.js SDK has the following internal
property:
```typescript
/** @internal */
readonly __prov?: ProviderResource;
```
When a resource is created, the provider specified for the resource is
stored in this property. If it is set, it is passed along in the `Call`
request when a method is called on the resource.
Prior to #13282, the property was only set for custom resources in
`Resource`'s constructor:
```typescript
this.__prov = custom ? opts.provider : undefined;
```
With #13282, it was changed to also store the value for remote
components:
```diff
- this.__prov = custom ? opts.provider : undefined;
+ this.__prov = custom || remote ? opts.provider : undefined;
```
This regressed the behavior when calling a method on a remote component
that had an explicit provider that wasn't the component provider, but
some other provider (e.g. AWS provider) specified as:
```typescript
const component = new MyRemoteComponent("comp", {
}, { provider: awsProvider });
```
The `awsProvider` was being stored in `Resource.__prov`, and when making
the method call on the resource, it would try to invoke `Call` on the
AWS provider, rather than calling the remote component provider's
`Call`, which resulted in an error.
Note that specifying the AWS provider using the more verbose `providers:
[awsProvider]` works around the issue.
The fix is to only set `__prov` if the provider's package is the same as
the resource's package. Otherwise, don't set it, because the user is
specifying a provider with the `provider: awsProvider` syntax as
shorthand for `providers: [awsProvider]`.
Fixes #13777
2023-08-30 14:49:53 +00:00
|
|
|
class MyRemoteComponentResource extends ComponentResource {
|
|
|
|
constructor(name: string, opts?: ComponentResourceOptions) {
|
|
|
|
super("test:index:MyRemoteComponentResource", name, {}, opts, true /*remote*/);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[sdk/nodejs] Fix hang due to component children cycles
When a resource depends on a local component resource, rather than setting the component resource itself as a dependency, each of the component's descendants is added as a dependency. This can lead to hangs when cycles are introduced.
For example, consider the following parent/child hierarchy, where `ComponentA` is the parent of `ComponentB` and `ComponentB` is the parent of `CustomC`:
```
ComponentA
|
ComponentB
|
CustomC
```
If `ComponentB` specifies it has a dependency on `ComponentA`, the following takes place as part determining the full set of transitive dependencies:
1. `ComponentA` is a component resource so it isn't added as a dependency, its children are.
2. `ComponentA` has one child: `ComponentB`
3. `ComponentB` is a component resource so it isn't added as a dependency, its children are.
4. `ComponentB` has one child: `CustomC`, a custom resource.
5. Since `CustomC` is a custom resource, it is added to the set of dependencies.
6. We try to await its URN, but we'll never get it because `RegisterResource` hasn't yet been called for it. And we hang waiting.
To address this, skip looking at a component's children if it is the component from which the dependency is being added.
In the case of the example, at step 3 the dependency expansion will stop: we won't look at `ComponentB`'s children because we're adding the dependency from `ComponentB`.
fix
2023-03-27 01:50:24 +00:00
|
|
|
// Regression test for https://github.com/pulumi/pulumi/issues/12032
|
|
|
|
describe("parent and dependsOn are the same 12032", () => {
|
|
|
|
runtime.setMocks({
|
|
|
|
call: (_) => {
|
|
|
|
throw new Error("unexpected call");
|
|
|
|
},
|
|
|
|
newResource: (args) => {
|
|
|
|
return { id: `${args.name}_id`, state: {} };
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://github.com/pulumi/pulumi/issues/12161
|
|
|
|
it("runs without error", async () => {
|
2023-04-28 22:27:10 +00:00
|
|
|
const parent = new ComponentResource("pkg:index:first", "first");
|
|
|
|
const child = new ComponentResource(
|
|
|
|
"pkg:index:second",
|
|
|
|
"second",
|
|
|
|
{},
|
|
|
|
{
|
|
|
|
parent,
|
|
|
|
dependsOn: parent,
|
|
|
|
},
|
|
|
|
);
|
[sdk/nodejs] Fix hang due to component children cycles
When a resource depends on a local component resource, rather than setting the component resource itself as a dependency, each of the component's descendants is added as a dependency. This can lead to hangs when cycles are introduced.
For example, consider the following parent/child hierarchy, where `ComponentA` is the parent of `ComponentB` and `ComponentB` is the parent of `CustomC`:
```
ComponentA
|
ComponentB
|
CustomC
```
If `ComponentB` specifies it has a dependency on `ComponentA`, the following takes place as part determining the full set of transitive dependencies:
1. `ComponentA` is a component resource so it isn't added as a dependency, its children are.
2. `ComponentA` has one child: `ComponentB`
3. `ComponentB` is a component resource so it isn't added as a dependency, its children are.
4. `ComponentB` has one child: `CustomC`, a custom resource.
5. Since `CustomC` is a custom resource, it is added to the set of dependencies.
6. We try to await its URN, but we'll never get it because `RegisterResource` hasn't yet been called for it. And we hang waiting.
To address this, skip looking at a component's children if it is the component from which the dependency is being added.
In the case of the example, at step 3 the dependency expansion will stop: we won't look at `ComponentB`'s children because we're adding the dependency from `ComponentB`.
fix
2023-03-27 01:50:24 +00:00
|
|
|
|
|
|
|
// This would result in warnings about leaked promises before the fix.
|
|
|
|
new MyCustomResource("myresource", {
|
|
|
|
parent: child,
|
|
|
|
});
|
|
|
|
});
|
2023-04-28 22:27:10 +00:00
|
|
|
});
|