Skip to content

Commit

Permalink
feat(core): add viaProxy Utility to rewrite requests to proxy (motion…
Browse files Browse the repository at this point in the history
  • Loading branch information
WaldemarLehner committed Feb 16, 2023
1 parent 39443c4 commit 15264ba
Show file tree
Hide file tree
Showing 6 changed files with 138 additions and 1 deletion.
3 changes: 2 additions & 1 deletion packages/2d/src/components/Image.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import {
SignalValue,
SimpleSignal,
} from '@motion-canvas/core/lib/signals';
import {viaProxy} from '@motion-canvas/core/lib/utils';

export interface ImageProps extends RectProps {
src?: SignalValue<string>;
Expand Down Expand Up @@ -54,7 +55,7 @@ export class Image extends Rect {

@computed()
protected image(): HTMLImageElement {
const src = this.src();
const src = viaProxy(this.src());
if (Image.pool[src]) {
return Image.pool[src];
}
Expand Down
1 change: 1 addition & 0 deletions packages/core/src/utils/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,3 +19,4 @@ export * from './useThread';
export * from './useTime';
export * from './useContext';
export * from './useDuration';
export * from './viaProxy';
64 changes: 64 additions & 0 deletions packages/core/src/utils/viaProxy.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
import {afterEach, describe, expect, test, vi} from 'vitest';
import {viaProxy} from './viaProxy';

const windowMock = {
location: {
toString: () => 'https://mockhostname:1234',
},
};

describe('viaProxy()', () => {
afterEach(() => {
vi.clearAllMocks();
});

vi.mock('virtual:proxysettings', () => ({enabled: false}));

describe('Pass requests as-is if proxying is disabled', () => {
test.each([
['Remote Address', 'https://example.com/image'],
['Local Address', 'https://mockhostname:1234/image'],
['Local Address No Host', '/image'],
])('%s', (_, val) => {
expect(viaProxy(val, false)).toEqual(val);
});
});

describe('Pass requests as-is if proxying is enabled, but requests go to local host', () => {
vi.stubGlobal('window', windowMock);

test.each([
['Local Address', 'https://mockhostname:1234/image'],
['Local Address No Host', '/image'],
])('%s', (_, val) => {
expect(viaProxy(val, true)).toEqual(val);
});
});

test('Transform request if proxying is enabled and request is remote', () => {
vi.stubGlobal('window', windowMock);

const address = 'https://example.com/some/image_resource.png?with_query';
expect(viaProxy(address, true)).toEqual(
`/cors-proxy/${encodeURIComponent(address)}`,
);
});

test('Unusual formats like base64 are to be ignored', () => {
vi.stubGlobal('window', windowMock); //

const address =
'';

expect(viaProxy(address, true)).toEqual(address);
});

test('Calling viaProxy multiple times should not break the result', () => {
vi.stubGlobal('window', windowMock);
const address = 'https://example.com/some/image_resource.png?with_query';

expect(viaProxy(address, true)).toEqual(
viaProxy(viaProxy(viaProxy(address, true), true), true),
);
});
});
57 changes: 57 additions & 0 deletions packages/core/src/utils/viaProxy.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
/**
* Utility to redirect remote sources via Proxy
*
* This utility is used to rewrite a request to be routed through
* the Proxy instead.
*/

/**
* Virtual Module provided by the Motion Canvas
* Plugin. Returns `false` if proxy is set to `false`, else `true`
*/
//@ts-ignore
import {enabled} from 'virtual:proxysettings';

/**
* Helper-Function to route remote requests through a local proxy.
*
* @example This rewrites a remote url like `https://via.placeholder.com/300.png/09f/fff` into a URI-Component-Encoded string like `/cors-proxy/https%3A%2F%2Fvia.placeholder.com%2F300.png%2F09f%2Ffff`
*/
export function viaProxy(url: string, overrideProxySetting?: boolean) {
const isEnabled = overrideProxySetting ?? enabled;
if (!isEnabled) {
// Proxy is disabled, so we just pass as-is.
return url;
}

if (url.startsWith('/cors-proxy/')) {
// Already proxied, return as-is
return url;
}

// window.location.hostname is being passed here to ensure that
// this does not throw an Error for same-origin requests
// e.g. /some/image -> localhost:9000/some/image

const selfUrl = new URL(window.location.toString());
// inside a try-catch in case the URL cannot be understood
try {
const expandedUrl = new URL(url, selfUrl);
if (!expandedUrl.protocol.startsWith('http')) {
// this is probably some embedded image (e.g. image/png;base64).
// don't touch and pass as is
return url;
}
if (selfUrl.host === expandedUrl.host) {
// This is a request to a "local" resource.
// No need to rewrite
return url;
}
} catch (_) {
// in case of error just silently pass as-is
return url;
}

// Everything else is a "remote" resource and requires a rewrite.
return `/cors-proxy/${encodeURIComponent(url)}`;
}
1 change: 1 addition & 0 deletions packages/core/vitest.setup.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import {vi} from 'vitest';

vi.stubGlobal('AudioContext', class {});
vi.mock('virtual:proxysettings', () => ({enabled: true}));
13 changes: 13 additions & 0 deletions packages/vite-plugin/src/main.ts
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,8 @@ export default ({
const createHtml = (src: string) => htmlParts[0] + src + htmlParts[1];

const resolvedEditorId = '\0virtual:editor';
const proxySettingsKey = 'virtual:proxysettings';

const timeStamps: Record<string, number> = {};
const outputPath = path.resolve(output);
const projects: ProjectData[] = [];
Expand Down Expand Up @@ -136,6 +138,11 @@ export default ({
async configResolved(resolvedConfig) {
viteConfig = resolvedConfig;
},
resolveId(id) {
if (id === proxySettingsKey) {
return '\0' + proxySettingsKey;
}
},
async load(id) {
const [base, query] = id.split('?');
const {name, dir} = path.posix.parse(base);
Expand Down Expand Up @@ -167,6 +174,12 @@ export default ({
);
}

if (id.startsWith('\0' + proxySettingsKey)) {
return source(
`export const enabled = "${proxy === false ? 'false' : 'true'}" `,
);
}

if (query) {
const params = new URLSearchParams(query);
if (params.has('scene')) {
Expand Down

0 comments on commit 15264ba

Please sign in to comment.