Skip to content

Commit

Permalink
fix(function-resource): lifecycle fixes
Browse files Browse the repository at this point in the history
Resolves an issue where, when a function-resource is directly used in a template,
the lifecycle of invoking and running cleanup was incorrect when wrapped
in an if statement (or any conditional rendering situation).  Now, when
a resource used within `{{ ... }}` is removed from the render tree, the
cleanup hook will run.
  • Loading branch information
NullVoxPopuli committed Jun 17, 2022
1 parent 3842d46 commit b5c6d9d
Show file tree
Hide file tree
Showing 7 changed files with 500 additions and 138 deletions.
213 changes: 118 additions & 95 deletions ember-resources/src/util/function-resource.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,9 @@ import { createCache, getValue } from '@glimmer/tracking/primitives/cache';
import { assert } from '@ember/debug';
import {
associateDestroyableChild,
destroy,
destroy,
registerDestructor,
unregisterDestructor,
// unregisterDestructor,
} from '@ember/destroyable';
// @ts-ignore
import { capabilities as helperCapabilities, invokeHelper, setHelperManager } from '@ember/helper';
Expand Down Expand Up @@ -155,26 +155,62 @@ export function resource<Value>(context: object, setup: ResourceFunction<Value>)
export function resource<Value>(
context: object | ResourceFunction<Value>,
setup?: ResourceFunction<Value>
):
| Value
| { [INTERNAL]: true; [INTERMEDIATE_VALUE]: ResourceFunction<Value> }
| ResourceFunction<Value> {
/**
* With only one argument, we have to do a bunch of lying to
* TS, because we need a special object to pass to `@use`
*/
if (typeof context === 'function' && !setup) {
setHelperManager(() => MANAGER, context);
): Value | InternalIntermediate<Value> | ResourceFunction<Value> {
if (!setup) {
assert(
`When using \`resource\` with @use, ` +
`the first argument to \`resource\` must be a function. ` +
`Instead, a ${typeof context} was received.`,
typeof context === 'function'
);

// Add secret key to help @use assert against
// using vanilla functions as resources without the resource wrapper
/**
* Functions have a different identity every time they are defined.
* The primary purpose of the `resource` wrapper is to individually
* register each function with our helper manager.
*/
setHelperManager(ResourceManagerFactory, context);

/**
* With only one argument, we have to do a bunch of lying to
* TS, because we need a special object to pass to `@use`
*
* Add secret key to help @use assert against
* using vanilla functions as resources without the resource wrapper
*/
(context as any)[INTERNAL] = true;

return context as ResourceFunction<Value>;
}

setHelperManager(() => MANAGER, setup);
assert(
`Mismatched argument typs passed to \`resource\`. ` +
`Expected the first arg, the context, to be a type of object. This is usually the \`this\`. ` +
`Received ${typeof context} instead.`,
typeof context === 'object'
);
assert(
`Mismatched argument type passed to \`resource\`. ` +
`Expected the second arg to be a function but instead received ${typeof setup}.`,
typeof setup === 'function'
);

setHelperManager(ResourceManagerFactory, setup);

return wrapForPlainUsage(context, setup);
}

const INTERMEDIATE_VALUE = '__Intermediate_Value__';
const INTERNAL = '__INTERNAL__';

/**
* This is what allows resource to be used withotu @use.
* The caveat though is that a property must be accessed
* on the return object.
*
* A resource not using use *must* be an object.
*/
function wrapForPlainUsage<Value>(context: object, setup: ResourceFunction<Value>) {
let cache: Cache;

/*
Expand All @@ -186,7 +222,7 @@ export function resource<Value>(
const target = {
get [INTERMEDIATE_VALUE]() {
if (!cache) {
cache = invokeHelper(context, setup, () => ({}));
cache = invokeHelper(context, setup);
}

return getValue<Value>(cache);
Expand Down Expand Up @@ -221,6 +257,15 @@ export function resource<Value>(
}) as never as Value;
}

/**
* Secret args to allow `resource` to be used without
* a decorator
*/
interface InternalIntermediate<Value> {
[INTERNAL]: true;
[INTERMEDIATE_VALUE]: ResourceFunction<Value>;
}

export type Hooks = {
on: {
/**
Expand Down Expand Up @@ -251,110 +296,83 @@ type Destructor = () => void;
type ResourceFunction<Value = unknown> = (hooks: Hooks) => Value;
type Cache = object;

const INTERMEDIATE_VALUE = '__Intermediate_Value__';
const INTERNAL = '__INTERNAL__';

let DESTROYERS = new WeakMap();

/**
* Note, a function-resource receives
* Note, a function-resource receives on object, hooks.
* We have to build that manually in this helper manager
*/
class FunctionResourceManager {
capabilities = helperCapabilities('3.23', {
hasValue: true,
hasDestroyable: true,
});

constructor(protected owner: unknown) {}

/**
* Resources do not take args.
* However, they can access tracked data
*/
createHelper(fn: ResourceFunction) {
/**
* The helper is only created once.
* It's the cache's callback that is invoked multiple times,
* based on reactive behavior
*
*/
let cache: Cache = createCache(() => {
let destroyer = DESTROYERS.get(fn);

/**
* Because every function invocation shares the same cache,
* we gotta take care of destruction manually.
*
* Glimmer will handle the last destruction for us when it tears down the cache
*
* It is not guaranteed if destruction is async or sync, and this may change in the future if it needs to
*/
if (destroyer) {
unregisterDestructor(fn, destroyer);
destroyer();
}
let thisFn = fn.bind(null);

let value = fn({
on: {
cleanup: (destroyer: Destructor) => {
associateDestroyableChild(cache, fn);
registerDestructor(fn, destroyer);
associateDestroyableChild(fn, thisFn);
return thisFn;
}

DESTROYERS.set(fn, destroyer);
},
},
});
previousFn?: object;

return value;
});
getValue(fn: ResourceFunction) {
if (this.previousFn) {
destroy(this.previousFn);
}

return cache;
}
let currentFn = fn.bind(null);
associateDestroyableChild(fn, currentFn);
this.previousFn = currentFn;

getValue(cache: Cache) {
return getValue(cache);
return currentFn({
on: {
cleanup: (destroyer: Destructor) => {
registerDestructor(currentFn, destroyer);
},
},
});
}

getDestroyable(cache: Cache) {
return cache;
getDestroyable(fn: ResourceFunction) {
return fn;
}
}

type ResourceFactory = (...args: any[]) => ReturnType<typeof resource>;
type InvokerState = { fn: ResourceFactory; args: any };

class ResourceInvokerManager {
capabilities = helperCapabilities('3.23', {
hasValue: true,
hasDestroyable: true,
});

helper?: object;
constructor(protected owner: unknown) {}

createHelper(fn: ResourceFactory, args: any): InvokerState {
return createCache(() => {
return fn(...args.positional);
});
createHelper(fn: ResourceFactory, args: any): Cache {
return { fn, args };
}

getValue(cache: Cache) {
if (this.helper) {
destroy(this.helper);
}

let helper = getValue(cache);

let result = invokeHelper(this, helper, () => ({}));

associateDestroyableChild(cache, helper);

this.helper = helper;
getValue({ fn, args }: { fn: ResourceFactory, args: any }) {
let helper = fn(...args.positional) as object;
let result = invokeHelper(this, helper);

return getValue(result);
}

getDestroyable(cache: Cache) {
return cache;
getDestroyable({ fn }: { fn: ResourceFactory }) {
return fn;
}
}

// Provide a singleton manager.
const MANAGER = new FunctionResourceManager();
const ResourceInvoker = new ResourceInvokerManager();
const ResourceManagerFactory = (owner: unknown) => new FunctionResourceManager(owner);
const ResourceInvokerFactory = (owner: unknown) => new ResourceInvokerManager(owner);

/**
* Allows wrapper functions to provide a [[resource]] for use in templates.
Expand All @@ -367,9 +385,9 @@ const ResourceInvoker = new ResourceInvokerManager();
*
* Example using strict mode + <template> syntax and a template-only component:
* ```js
* import { resource, registerResourceWrapper } from 'ember-resources/util/function-resource';
* import { resource, resourceFactory } from 'ember-resources/util/function-resource';
*
* function RemoteData(url) {
* const RemoteData = resourceFactory((url) => {
* return resource(({ on }) => {
* let state = new TrackedObject({});
* let controller = new AbortController();
Expand All @@ -387,12 +405,10 @@ const ResourceInvoker = new ResourceInvokerManager();
*
* return state;
* })
* }
*
* registerResourceWrapper(RemoteData)
* });
*
* <template>
* {{#let (load) as |state|}}
* {{#let (RemoteData) as |state|}}
* {{#if state.value}}
* ...
* {{else if state.error}}
Expand All @@ -402,22 +418,27 @@ const ResourceInvoker = new ResourceInvokerManager();
* </template>
* ```
*
* Alternatively, `registerResourceWrapper` can wrap the wrapper function.
* Alternatively, `resourceFactory` can wrap the wrapper function.
*
* ```js
* const RemoteData = registerResourceWrapper((url) => {
* const RemoteData = resourceFactory((url) => {
* return resource(({ on }) => {
* ...
* });
* })
* ```
*/
export function registerResourceWrapper(wrapperFn: ResourceFactory) {
setHelperManager(() => ResourceInvoker, wrapperFn);
export function resourceFactory(wrapperFn: ResourceFactory) {
setHelperManager(ResourceInvokerFactory, wrapperFn);

return wrapperFn;
}

/**
* @deprecated - use resourceFactory (same behavior, just renamed)
*/
export const registerResourceWrapper = resourceFactory;

interface Descriptor {
initializer: () => unknown;
}
Expand Down Expand Up @@ -458,8 +479,8 @@ export function use(_prototype: object, key: string, descriptor?: Descriptor): v

// https://github.com/pzuraq/ember-could-get-used-to-this/blob/master/addon/index.js
return {
get() {
let cache = caches.get(this as object);
get(this: object) {
let cache = caches.get(this);

if (!cache) {
let fn = initializer.call(this);
Expand All @@ -469,7 +490,9 @@ export function use(_prototype: object, key: string, descriptor?: Descriptor): v
isResourceInitializer(fn)
);

cache = invokeHelper(this, fn, () => ({}));
cache = invokeHelper(this, fn);

associateDestroyableChild(this, cache);

caches.set(this as object, cache);
}
Expand Down
4 changes: 2 additions & 2 deletions ember-resources/src/util/remote-data.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { tracked } from '@glimmer/tracking';
import { waitForPromise } from '@ember/test-waiters';

import { registerResourceWrapper, resource } from './function-resource';
import { resource, resourceFactory } from './function-resource';

import type { Hooks } from './function-resource';

Expand Down Expand Up @@ -237,4 +237,4 @@ export function RemoteData(
});
}

registerResourceWrapper(RemoteData);
resourceFactory(RemoteData);
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
"author": "NullVoxPopuli",
"scripts": {
"dev": "concurrently 'npm:dev:*' --restart-after 5000 --prefix-colors cyan,white,yellow",
"dev:ember": "pnpm run --filter ember-app start --port 0",
"dev:ember": "pnpm run --filter ember-app start",
"dev:addon": "pnpm run --filter ember-resources start --no-watch.clearScreen",
"dev:docs": "pnpm run --filter docs docs:watch --preserveWatchOutput",
"ci:update": "npx ember-ci-update",
Expand Down
1 change: 1 addition & 0 deletions testing/ember-app/app/config/environment.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,4 +11,5 @@ declare const config: {
locationType: 'history' | 'hash' | 'none' | 'auto';
rootURL: string;
APP: Record<string, unknown>;
isTestCli: boolean | undefined;
};
4 changes: 4 additions & 0 deletions testing/ember-app/config/environment.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
'use strict';

let isTestCli = process.argv.includes('test');

module.exports = function (environment) {
let ENV = {
modulePrefix: 'ember-app',
Expand All @@ -21,6 +23,8 @@ module.exports = function (environment) {
// Here you can pass flags/options to your application instance
// when it is created
},

isTestCli,
};

if (environment === 'development') {
Expand Down
Loading

0 comments on commit b5c6d9d

Please sign in to comment.