Skip to content

Commit

Permalink
Make feature discovery lazy
Browse files Browse the repository at this point in the history
While working toward [strict-es-modules](emberjs/rfcs#938), I found that under real ES modules, this feature-detection code will become eager, which makes it blow up inside all versions of ember-cli-htmlbars before ember-cli/ember-cli-htmlbars#786, because ember-cli-htmlbars insists on evaluating the template compiler bundle inside a VM with no valid `URL` global.

While that is now fixed in an upcoming major of ember-cli-htmlbars, people can't benefit from that until their very last addon upgrades ember-cli-htmlbars.

So it's also good to fix this directly, by making the feature detection code explicitly lazy.
  • Loading branch information
ef4 committed Dec 24, 2023
1 parent 8b470f4 commit 70e23c1
Showing 1 changed file with 55 additions and 48 deletions.
103 changes: 55 additions & 48 deletions packages/@glimmer/runtime/lib/dom/sanitized-values.ts
Original file line number Diff line number Diff line change
Expand Up @@ -37,55 +37,62 @@ interface NodeUrlModule {
parse(url: string): NodeUrlParseResult;
}

let protocolForUrl: (url: string) => string;

if (
typeof URL === 'object' &&
URL !== null &&
// this is super annoying, TS thinks that URL **must** be a function so `URL.parse` check
// thinks it is `never` without this `as unknown as any`
typeof (URL as unknown as any).parse === 'function'
) {
// In Ember-land the `fastboot` package sets the `URL` global to `require('url')`
// ultimately, this should be changed (so that we can either rely on the natural `URL` global
// that exists) but for now we have to detect the specific `FastBoot` case first
//
// a future version of `fastboot` will detect if this legacy URL setup is required (by
// inspecting Ember version) and if new enough, it will avoid shadowing the `URL` global
// constructor with `require('url')`.
let nodeURL = URL as NodeUrlModule;

protocolForUrl = (url: string) => {
let protocol = null;

if (typeof url === 'string') {
protocol = nodeURL.parse(url).protocol;
}
function findProtocolForURL() {
if (
typeof URL === 'object' &&
URL !== null &&
// this is super annoying, TS thinks that URL **must** be a function so `URL.parse` check
// thinks it is `never` without this `as unknown as any`
typeof ((URL as unknown) as any).parse === 'function'
) {
// In Ember-land the `fastboot` package sets the `URL` global to `require('url')`
// ultimately, this should be changed (so that we can either rely on the natural `URL` global
// that exists) but for now we have to detect the specific `FastBoot` case first
//
// a future version of `fastboot` will detect if this legacy URL setup is required (by
// inspecting Ember version) and if new enough, it will avoid shadowing the `URL` global
// constructor with `require('url')`.
let nodeURL = URL as NodeUrlModule;

return (url: string) => {
let protocol = null;

if (typeof url === 'string') {
protocol = nodeURL.parse(url).protocol;
}

return protocol === null ? ':' : protocol;
};
} else if (typeof URL === 'function') {
return (_url: string) => {
try {
let url = new URL(_url);

return url.protocol;
} catch (error) {
// any non-fully qualified url string will trigger an error (because there is no
// baseURI that we can provide; in that case we **know** that the protocol is
// "safe" because it isn't specifically one of the `badProtocols` listed above
// (and those protocols can never be the default baseURI)
return ':';
}
};
} else {
// fallback for IE11 support
let parsingNode = document.createElement('a');
return (url: string) => {
parsingNode.href = url;
return parsingNode.protocol;
};
}
}

return protocol === null ? ':' : protocol;
};
} else if (typeof URL === 'function') {
protocolForUrl = (_url: string) => {
try {
let url = new URL(_url);

return url.protocol;
} catch (error) {
// any non-fully qualified url string will trigger an error (because there is no
// baseURI that we can provide; in that case we **know** that the protocol is
// "safe" because it isn't specifically one of the `badProtocols` listed above
// (and those protocols can never be the default baseURI)
return ':';
}
};
} else {
// fallback for IE11 support
let parsingNode = document.createElement('a');

protocolForUrl = (url: string) => {
parsingNode.href = url;
return parsingNode.protocol;
};
let _protocolForUrlImplementation: typeof protocolForUrl | undefined;
function protocolForUrl(url: string): string {
if (!_protocolForUrlImplementation) {
_protocolForUrlImplementation = findProtocolForURL();
}
return _protocolForUrlImplementation(url);
}

export function sanitizeAttributeValue(
Expand Down

0 comments on commit 70e23c1

Please sign in to comment.