Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix passing variables across compilations #367

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions lib/src/function-registry.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import {Value} from './value';
* execute them.
*/
export class FunctionRegistry<sync extends 'sync' | 'async'> {
public readonly compileContext = Symbol();
private readonly functionsByName = new Map<string, CustomFunction<sync>>();
private readonly functionsById = new Map<number, CustomFunction<sync>>();
private readonly idsByFunction = new Map<CustomFunction<sync>, number>();
Expand Down
47 changes: 36 additions & 11 deletions lib/src/protofier.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ export class Protofier {
get accessedArgumentLists(): number[] {
return this.argumentLists
.filter(list => list.keywordsAccessed)
.map(list => list.id);
.map(list => list.id as number);
}

constructor(
Expand Down Expand Up @@ -85,15 +85,23 @@ export class Protofier {
});
result.value = {case: 'list', value: list};
} else if (value instanceof SassArgumentList) {
const list = create(proto.Value_ArgumentListSchema, {
id: value.id,
separator: this.protofySeparator(value.separator),
contents: value.asList.map(element => this.protofy(element)).toArray(),
});
for (const [key, mapValue] of value.keywordsInternal) {
list.keywords[key] = this.protofy(mapValue);
if (value.compileContext === this.functions.compileContext) {
const list = create(proto.Value_ArgumentListSchema, {
id: value.id,
});
result.value = {case: 'argumentList', value: list};
} else {
const list = create(proto.Value_ArgumentListSchema, {
separator: this.protofySeparator(value.separator),
contents: value.asList
.map(element => this.protofy(element))
.toArray(),
});
for (const [key, mapValue] of value.keywordsInternal) {
list.keywords[key] = this.protofy(mapValue);
}
result.value = {case: 'argumentList', value: list};
}
result.value = {case: 'argumentList', value: list};
} else if (value instanceof SassMap) {
const map = create(proto.Value_MapSchema, {
entries: value.contents.toArray().map(([key, value]) => ({
Expand All @@ -104,6 +112,11 @@ export class Protofier {
result.value = {case: 'map', value: map};
} else if (value instanceof SassFunction) {
if (value.id !== undefined) {
if (value.compileContext !== this.functions.compileContext) {
throw utils.compilerError(
`Value ${value} does not belong to this compilation`,
);
}
const fn = create(proto.Value_CompilerFunctionSchema, value);
result.value = {case: 'compilerFunction', value: fn};
} else {
Expand All @@ -114,6 +127,11 @@ export class Protofier {
result.value = {case: 'hostFunction', value: fn};
}
} else if (value instanceof SassMixin) {
if (value.compileContext !== this.functions.compileContext) {
throw utils.compilerError(
`Value ${value} does not belong to this compilation`,
);
}
const mixin = create(proto.Value_CompilerMixinSchema, value);
result.value = {case: 'compilerMixin', value: mixin};
} else if (value instanceof SassCalculation) {
Expand Down Expand Up @@ -349,6 +367,7 @@ export class Protofier {
),
separator,
list.id,
this.functions.compileContext,
);
this.argumentLists.push(result);
return result;
Expand All @@ -369,15 +388,21 @@ export class Protofier {
);

case 'compilerFunction':
return new SassFunction(value.value.value.id);
return new SassFunction(
value.value.value.id,
this.functions.compileContext,
);

case 'hostFunction':
throw utils.compilerError(
'The compiler may not send Value.host_function.',
);

case 'compilerMixin':
return new SassMixin(value.value.value.id);
return new SassMixin(
value.value.value.id,
this.functions.compileContext,
);

case 'calculation':
return this.deprotofyCalculation(value.value.value);
Expand Down
17 changes: 15 additions & 2 deletions lib/src/value/argument-list.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,18 @@ export class SassArgumentList extends SassList {
* part of the package's public API and should not be accessed by user code.
* It may be renamed or removed without warning in the future.
*/
readonly id: number;
readonly id: number | undefined;

/**
* If this argument list is constructed in the compiler, this is the unique
* context that the host uses to determine which compilation this argument
* list belongs to.
*
* This is marked as public so that the protofier can access it, but it's not
* part of the package's public API and should not be accessed by user code.
* It may be renamed or removed without warning in the future.
*/
readonly compileContext: symbol | undefined;

/**
* The argument list's keywords. This isn't exposed directly so that we can
Expand Down Expand Up @@ -54,11 +65,13 @@ export class SassArgumentList extends SassList {
keywords: Record<string, Value> | OrderedMap<string, Value>,
separator?: ListSeparator,
id?: number,
compileContext?: symbol,
) {
super(contents, {separator});
this.keywordsInternal = isOrderedMap(keywords)
? keywords
: OrderedMap(keywords);
this.id = id ?? 0;
this.id = id;
this.compileContext = compileContext;
}
}
28 changes: 22 additions & 6 deletions lib/src/value/function.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,16 @@ export class SassFunction extends Value {
*/
readonly id: number | undefined;

/**
* If this function is defined in the compiler, this is the unique context
* that the host uses to determine which compilation this function belongs to.
*
* This is marked as public so that the protofier can access it, but it's not
* part of the package's public API and should not be accessed by user code.
* It may be renamed or removed without warning in the future.
*/
readonly compileContext: symbol | undefined;

/**
* If this function is defined in the host, this is the signature that
* describes how to pass arguments to it.
Expand All @@ -39,26 +49,32 @@ export class SassFunction extends Value {
*/
readonly callback: CustomFunction<'sync'> | undefined;

constructor(id: number);
constructor(id: number, compileContext: symbol);
constructor(signature: string, callback: CustomFunction<'sync'>);
constructor(
idOrSignature: number | string,
callback?: CustomFunction<'sync'>,
callbackOrCompileContext: CustomFunction<'sync'> | symbol,
) {
super();

if (typeof idOrSignature === 'number') {
if (
typeof idOrSignature === 'number' &&
typeof callbackOrCompileContext === 'symbol'
) {
this.id = idOrSignature;
this.compileContext = callbackOrCompileContext;
} else {
this.signature = idOrSignature;
this.callback = callback!;
this.signature = idOrSignature as string;
this.callback = callbackOrCompileContext as CustomFunction<'sync'>;
}
}

equals(other: Value): boolean {
return this.id === undefined
? other === this
: other instanceof SassFunction && other.id === this.id;
: other instanceof SassFunction &&
other.compileContext === this.compileContext &&
other.id === this.id;
}

hashCode(): number {
Expand Down
19 changes: 17 additions & 2 deletions lib/src/value/mixin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,13 +18,28 @@ export class SassMixin extends Value {
*/
readonly id: number;

constructor(id: number) {
/**
* This is the unique context that the host uses to determine which
* compilation this mixin belongs to.
*
* This is marked as public so that the protofier can access it, but it's not
* part of the package's public API and should not be accessed by user code.
* It may be renamed or removed without warning in the future.
*/
readonly compileContext: symbol;

constructor(id: number, compileContext: symbol) {
super();
this.id = id;
this.compileContext = compileContext;
}

equals(other: Value): boolean {
return other instanceof SassMixin && other.id === this.id;
return (
other instanceof SassMixin &&
other.compileContext === this.compileContext &&
other.id === this.id
);
}

hashCode(): number {
Expand Down