simplify $inspect

pull/12069/head
Rich Harris 7 months ago
parent 6588d85364
commit 36704e4b01

@ -16,6 +16,7 @@ export const EFFECT_RAN = 1 << 14;
export const EFFECT_TRANSPARENT = 1 << 15; export const EFFECT_TRANSPARENT = 1 << 15;
/** Svelte 4 legacy mode props need to be handled with deriveds and be recognized elsewhere, hence the dedicated flag */ /** Svelte 4 legacy mode props need to be handled with deriveds and be recognized elsewhere, hence the dedicated flag */
export const LEGACY_DERIVED_PROP = 1 << 16; export const LEGACY_DERIVED_PROP = 1 << 16;
export const INSPECT_EFFECT = 1 << 17;
export const STATE_SYMBOL = Symbol('$state'); export const STATE_SYMBOL = Symbol('$state');
export const STATE_FROZEN_SYMBOL = Symbol('$state.frozen'); export const STATE_FROZEN_SYMBOL = Symbol('$state.frozen');

@ -1,20 +1,7 @@
import { DEV } from 'esm-env';
import { snapshot } from '../proxy.js'; import { snapshot } from '../proxy.js';
import { render_effect, validate_effect } from '../reactivity/effects.js'; import { inspect_effect, validate_effect } from '../reactivity/effects.js';
import { deep_read, untrack } from '../runtime.js';
import { array_prototype, get_prototype_of, object_prototype } from '../utils.js'; import { array_prototype, get_prototype_of, object_prototype } from '../utils.js';
/** @type {Function | null} */
export let inspect_fn = null;
/** @param {Function | null} fn */
export function set_inspect_fn(fn) {
inspect_fn = fn;
}
/** @type {Array<import('#client').ValueDebug>} */
export let inspect_captured_signals = [];
/** /**
* @param {() => any[]} get_value * @param {() => any[]} get_value
* @param {Function} [inspector] * @param {Function} [inspector]
@ -25,32 +12,10 @@ export function inspect(get_value, inspector = console.log) {
let initial = true; let initial = true;
// we assign the function directly to signals, rather than just inspect_effect(() => {
// calling `inspector` directly inside the effect, so that const value = deep_snapshot(get_value());
// we get useful stack traces
var fn = () => {
const value = untrack(() => deep_snapshot(get_value()));
inspector(initial ? 'init' : 'update', ...value); inspector(initial ? 'init' : 'update', ...value);
}; initial = false;
render_effect(() => {
inspect_fn = fn;
deep_read(get_value());
inspect_fn = null;
const signals = inspect_captured_signals.slice();
inspect_captured_signals = [];
if (initial) {
fn();
initial = false;
}
return () => {
for (const s of signals) {
s.inspect.delete(fn);
}
};
}); });
} }

@ -1,11 +1,5 @@
import { DEV } from 'esm-env'; import { DEV } from 'esm-env';
import { import { get, current_component_context, untrack, current_effect } from './runtime.js';
get,
batch_inspect,
current_component_context,
untrack,
current_effect
} from './runtime.js';
import { import {
array_prototype, array_prototype,
define_property, define_property,
@ -230,11 +224,6 @@ const state_proxy_handler = {
return value === UNINITIALIZED ? undefined : value; return value === UNINITIALIZED ? undefined : value;
} }
if (DEV) {
if (typeof target[prop] === 'function' && prop !== Symbol.iterator) {
return batch_inspect(target, prop, receiver);
}
}
return Reflect.get(target, prop, receiver); return Reflect.get(target, prop, receiver);
}, },

@ -39,10 +39,6 @@ export function derived(fn) {
version: 0 version: 0
}; };
if (DEV) {
/** @type {import('#client').DerivedDebug} */ (signal).inspect = new Set();
}
if (current_reaction !== null && (current_reaction.f & DERIVED) !== 0) { if (current_reaction !== null && (current_reaction.f & DERIVED) !== 0) {
var current_derived = /** @type {import('#client').Derived<V>} */ (current_reaction); var current_derived = /** @type {import('#client').Derived<V>} */ (current_reaction);
if (current_derived.deriveds === null) { if (current_derived.deriveds === null) {
@ -107,10 +103,6 @@ export function update_derived(derived, force_schedule) {
derived.version = increment_version(); derived.version = increment_version();
mark_reactions(derived, DIRTY, force_schedule); mark_reactions(derived, DIRTY, force_schedule);
if (DEV && force_schedule) {
for (var fn of /** @type {import('#client').DerivedDebug} */ (derived).inspect) fn();
}
} }
} }

@ -30,7 +30,8 @@ import {
EFFECT_TRANSPARENT, EFFECT_TRANSPARENT,
DERIVED, DERIVED,
UNOWNED, UNOWNED,
CLEAN CLEAN,
INSPECT_EFFECT
} from '../constants.js'; } from '../constants.js';
import { set } from './sources.js'; import { set } from './sources.js';
import { remove } from '../dom/reconciler.js'; import { remove } from '../dom/reconciler.js';
@ -204,6 +205,11 @@ export function user_pre_effect(fn) {
return render_effect(fn); return render_effect(fn);
} }
/** @param {() => void | (() => void)} fn */
export function inspect_effect(fn) {
return create_effect(INSPECT_EFFECT, fn, true);
}
/** /**
* Internal representation of `$effect.root(...)` * Internal representation of `$effect.root(...)`
* @param {() => void | (() => void)} fn * @param {() => void | (() => void)} fn

@ -10,7 +10,6 @@ import { mutable_source, set, source } from './sources.js';
import { derived, derived_safe_equal } from './deriveds.js'; import { derived, derived_safe_equal } from './deriveds.js';
import { get, is_signals_recorded, untrack, update } from '../runtime.js'; import { get, is_signals_recorded, untrack, update } from '../runtime.js';
import { safe_equals } from './equality.js'; import { safe_equals } from './equality.js';
import { inspect_fn } from '../dev/inspect.js';
import * as e from '../errors.js'; import * as e from '../errors.js';
import { LEGACY_DERIVED_PROP } from '../constants.js'; import { LEGACY_DERIVED_PROP } from '../constants.js';
@ -308,7 +307,7 @@ export function prop(props, key, flags, fallback) {
// legacy nonsense — need to ensure the source is invalidated when necessary // legacy nonsense — need to ensure the source is invalidated when necessary
// also needed for when handling inspect logic so we can inspect the correct source signal // also needed for when handling inspect logic so we can inspect the correct source signal
if (is_signals_recorded || (DEV && inspect_fn)) { if (is_signals_recorded) {
// set this so that we don't reset to the parent value if `d` // set this so that we don't reset to the parent value if `d`
// is invalidated because of `invalidate_inner_signals` (rather // is invalidated because of `invalidate_inner_signals` (rather
// than because the parent or child value changed) // than because the parent or child value changed)

@ -6,7 +6,6 @@ import {
current_effect, current_effect,
current_untracked_writes, current_untracked_writes,
get, get,
is_batching_effect,
is_runes, is_runes,
mark_reactions, mark_reactions,
schedule_effect, schedule_effect,
@ -14,7 +13,9 @@ import {
set_last_inspected_signal, set_last_inspected_signal,
set_signal_status, set_signal_status,
untrack, untrack,
increment_version increment_version,
execute_effect,
inspect_effects
} from '../runtime.js'; } from '../runtime.js';
import { equals, safe_equals } from './equality.js'; import { equals, safe_equals } from './equality.js';
import { CLEAN, DERIVED, DIRTY, BRANCH_EFFECT } from '../constants.js'; import { CLEAN, DERIVED, DIRTY, BRANCH_EFFECT } from '../constants.js';
@ -37,10 +38,6 @@ export function source(value) {
version: 0 version: 0
}; };
if (DEV) {
/** @type {import('#client').ValueDebug<V>} */ (source).inspect = new Set();
}
return source; return source;
} }
@ -129,11 +126,11 @@ export function set(source, value) {
} }
if (DEV) { if (DEV) {
if (is_batching_effect) { for (const effect of inspect_effects) {
set_last_inspected_signal(/** @type {import('#client').ValueDebug} */ (source)); execute_effect(effect);
} else {
for (const fn of /** @type {import('#client').ValueDebug} */ (source).inspect) fn();
} }
inspect_effects.clear();
} }
} }

@ -29,13 +29,13 @@ import {
ROOT_EFFECT, ROOT_EFFECT,
LEGACY_DERIVED_PROP, LEGACY_DERIVED_PROP,
DISCONNECTED, DISCONNECTED,
STATE_FROZEN_SYMBOL STATE_FROZEN_SYMBOL,
INSPECT_EFFECT
} from './constants.js'; } from './constants.js';
import { flush_tasks } from './dom/task.js'; import { flush_tasks } from './dom/task.js';
import { add_owner } from './dev/ownership.js'; import { add_owner } from './dev/ownership.js';
import { mutate, set, source } from './reactivity/sources.js'; import { mutate, set, source } from './reactivity/sources.js';
import { update_derived } from './reactivity/deriveds.js'; import { update_derived } from './reactivity/deriveds.js';
import { inspect_captured_signals, inspect_fn, set_inspect_fn } from './dev/inspect.js';
import * as e from './errors.js'; import * as e from './errors.js';
import { lifecycle_outside_component } from '../shared/errors.js'; import { lifecycle_outside_component } from '../shared/errors.js';
@ -63,9 +63,7 @@ export function set_is_destroying_effect(value) {
is_destroying_effect = value; is_destroying_effect = value;
} }
// Used for $inspect export let inspect_effects = new Set();
export let is_batching_effect = false;
let is_inspecting_signal = false;
// Handle effect queues // Handle effect queues
@ -159,38 +157,6 @@ export function is_runes() {
return current_component_context !== null && current_component_context.l === null; return current_component_context !== null && current_component_context.l === null;
} }
/**
* @param {import('#client').ProxyStateObject} target
* @param {string | symbol} prop
* @param {any} receiver
*/
export function batch_inspect(target, prop, receiver) {
const value = Reflect.get(target, prop, receiver);
/**
* @this {any}
*/
return function () {
const previously_batching_effect = is_batching_effect;
is_batching_effect = true;
try {
return Reflect.apply(value, this, arguments);
} finally {
is_batching_effect = previously_batching_effect;
if (last_inspected_signal !== null && !is_inspecting_signal) {
is_inspecting_signal = true;
try {
for (const fn of last_inspected_signal.inspect) {
fn();
}
} finally {
is_inspecting_signal = false;
}
last_inspected_signal = null;
}
}
};
}
/** /**
* Determines whether a derived or effect is dirty. * Determines whether a derived or effect is dirty.
* If it is MAYBE_DIRTY, will set the status to CLEAN * If it is MAYBE_DIRTY, will set the status to CLEAN
@ -793,12 +759,6 @@ export async function tick() {
* @returns {V} * @returns {V}
*/ */
export function get(signal) { export function get(signal) {
if (DEV && inspect_fn) {
var s = /** @type {import('#client').ValueDebug} */ (signal);
s.inspect.add(inspect_fn);
inspect_captured_signals.push(s);
}
const flags = signal.f; const flags = signal.f;
if ((flags & DESTROYED) !== 0) { if ((flags & DESTROYED) !== 0) {
return signal.v; return signal.v;
@ -846,15 +806,7 @@ export function get(signal) {
(flags & DERIVED) !== 0 && (flags & DERIVED) !== 0 &&
check_dirtiness(/** @type {import('#client').Derived} */ (signal)) check_dirtiness(/** @type {import('#client').Derived} */ (signal))
) { ) {
if (DEV) { update_derived(/** @type {import('#client').Derived} **/ (signal), false);
// we want to avoid tracking indirect dependencies
const previous_inspect_fn = inspect_fn;
set_inspect_fn(null);
update_derived(/** @type {import('#client').Derived} **/ (signal), false);
set_inspect_fn(previous_inspect_fn);
} else {
update_derived(/** @type {import('#client').Derived} **/ (signal), false);
}
} }
return signal.v; return signal.v;
@ -914,6 +866,11 @@ export function mark_reactions(signal, to_status, force_schedule) {
var reaction = reactions[i]; var reaction = reactions[i];
var flags = reaction.f; var flags = reaction.f;
if (DEV && (flags & INSPECT_EFFECT) !== 0) {
inspect_effects.add(reaction);
continue;
}
// We skip any effects that are already dirty. Additionally, we also // We skip any effects that are already dirty. Additionally, we also
// skip if the reaction is the same as the current effect (except if we're not in runes or we // skip if the reaction is the same as the current effect (except if we're not in runes or we
// are in force schedule mode). // are in force schedule mode).

Loading…
Cancel
Save