diff --git a/src/components/gltf-model-plus.js b/src/components/gltf-model-plus.js
index b1195922467eb57dd6b536e688b147c164e502ea..fad11f667fb6c879fbc1505e056d9b7f39130684 100644
--- a/src/components/gltf-model-plus.js
+++ b/src/components/gltf-model-plus.js
@@ -1,3 +1,4 @@
+import { isFarsparkUrl, decodeFarsparkUrl } from "../utils/media-utils";
 const GLTFCache = {};
 
 AFRAME.GLTFModelPlus = {
@@ -186,6 +187,10 @@ function cachedLoadGLTF(src, preferredTechnique, onProgress) {
   if (!GLTFCache[src]) {
     GLTFCache[src] = new Promise((resolve, reject) => {
       const gltfLoader = new THREE.GLTFLoader();
+      // Farspark urls can't handle relative paths. Use the original url as the base path.
+      if (isFarsparkUrl(src)) {
+        gltfLoader.path = THREE.LoaderUtils.extractUrlBase(decodeFarsparkUrl(src));
+      }
       gltfLoader.preferredTechnique = preferredTechnique;
       gltfLoader.load(src, resolve, onProgress, reject);
     });
diff --git a/src/utils/media-utils.js b/src/utils/media-utils.js
index 9c17813b926e5541eef96b586f8efb133e696f6c..65c13bb0149ce9e6c6d2b4fd853958efd523f236 100644
--- a/src/utils/media-utils.js
+++ b/src/utils/media-utils.js
@@ -1,6 +1,5 @@
 const whitelistedHosts = [/^.*\.reticulum\.io$/, /^.*hubs\.mozilla\.com$/, /^hubs\.local$/];
 const isHostWhitelisted = hostname => !!whitelistedHosts.filter(r => r.test(hostname)).length;
-
 let resolveMediaUrl = "/api/v1/media";
 if (process.env.NODE_ENV === "development") {
   resolveMediaUrl = `https://${process.env.DEV_RETICULUM_SERVER}${resolveMediaUrl}`;
@@ -18,9 +17,24 @@ export const resolveFarsparkUrl = async url => {
   }).then(r => r.json())).raw;
 };
 
-const fetchContentType = async url => fetch(url, { method: "HEAD" }).then(r => r.headers.get("content-type"));
-let interactableId = 0;
+const farsparkRegex = /^farspark.*\.reticulum\.io$/;
+export const isFarsparkUrl = url => farsparkRegex.test(new URL(url).hostname);
+export const decodeFarsparkUrl = url => {
+  const parts = url.split("/");
+  return atob(parts[parts.length - 1]);
+};
 
+const staticContentMappings = {
+  "poly.googleapis.com": "model/gltf"
+};
+const fetchContentType = async url => {
+  const staticContentType = staticContentMappings[new URL(decodeFarsparkUrl(url)).hostname];
+  return staticContentType
+    ? Promise.resolve(staticContentType)
+    : fetch(url, { method: "HEAD" }).then(r => r.headers.get("content-type"));
+};
+
+let interactableId = 0;
 const offset = { x: 0, y: 0, z: -1.5 };
 export const spawnNetworkedImage = (src, contentType) => {
   const scene = AFRAME.scenes[0];