Merge branch 'bug/broken_cycle_detection' of https://github.com/colincasey/svelte into colincasey-bug/broken_cycle_detection

pull/3527/head
Richard Harris 5 years ago
commit 0f80ea7aea

@ -24,6 +24,7 @@ import unwrap_parens from './utils/unwrap_parens';
import Slot from './nodes/Slot'; import Slot from './nodes/Slot';
import { Node as ESTreeNode } from 'estree'; import { Node as ESTreeNode } from 'estree';
import add_to_set from './utils/add_to_set'; import add_to_set from './utils/add_to_set';
import checkGraphForCycles from './utils/checkGraphForCycles';
interface ComponentOptions { interface ComponentOptions {
namespace?: string; namespace?: string;
@ -1205,14 +1206,27 @@ export default class Component {
}); });
}); });
const add_declaration = declaration => { const cycle = checkGraphForCycles(unsorted_reactive_declarations.reduce((acc, declaration) => {
if (seen.has(declaration)) { declaration.assignees.forEach(v => {
declaration.dependencies.forEach(w => {
if (!declaration.assignees.has(w)) {
acc.push([v, w]);
}
});
});
return acc;
}, []));
if (cycle && cycle.length) {
const declarationList = lookup.get(cycle[0]);
const declaration = declarationList[0];
this.error(declaration.node, { this.error(declaration.node, {
code: 'cyclical-reactive-declaration', code: 'cyclical-reactive-declaration',
message: 'Cyclical dependency detected' message: `Cyclical dependency detected: ${cycle.join(' → ')}`
}); });
} }
const add_declaration = declaration => {
if (this.reactive_declarations.indexOf(declaration) !== -1) { if (this.reactive_declarations.indexOf(declaration) !== -1) {
return; return;
} }

@ -0,0 +1,36 @@
export default function checkGraphForCycles(edges: Array<[any, any]>): any[] {
const graph: Map<any, any[]> = edges.reduce((g, edge) => {
const [u, v] = edge;
if (!g.has(u)) g.set(u, []);
if (!g.has(v)) g.set(v, []);
g.get(u).push(v);
return g;
}, new Map());
const visited = new Set();
const onStack = new Set();
const cycles = [];
function visit (v) {
visited.add(v);
onStack.add(v);
graph.get(v).forEach(w => {
if (!visited.has(w)) {
visit(w);
} else if (onStack.has(w)) {
cycles.push([...onStack, w]);
}
});
onStack.delete(v);
}
graph.forEach((_, v) => {
if (!visited.has(v)) {
visit(v);
}
});
return cycles[0];
}

@ -0,0 +1,11 @@
export default {
html: `
<p>2+2=4</p>
`,
test({ assert, target }) {
assert.htmlEqual(target.innerHTML, `
<p>2+2=4</p>
`);
}
};

@ -0,0 +1,7 @@
<script>
$: c = a + b;
$: a = 2;
$: b = a;
</script>
<p>{a}+{b}={c}</p>

@ -1,5 +1,5 @@
[{ [{
"message": "Cyclical dependency detected", "message": "Cyclical dependency detected: a → b → a",
"code": "cyclical-reactive-declaration", "code": "cyclical-reactive-declaration",
"start": { "line": 5, "column": 1, "character": 35 }, "start": { "line": 5, "column": 1, "character": 35 },
"end": { "line": 5, "column": 14, "character": 48 }, "end": { "line": 5, "column": 14, "character": 48 },

Loading…
Cancel
Save