@nuxt/scripts 1.0.0-beta.3 → 1.0.0-beta.30

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (217) hide show
  1. package/README.md +3 -3
  2. package/dist/client/200.html +1 -1
  3. package/dist/client/404.html +1 -1
  4. package/dist/client/_nuxt/{Ds2G8aQM.js → AwAKM0sG.js} +1 -1
  5. package/dist/client/_nuxt/Bl23o3st.js +162 -0
  6. package/dist/client/_nuxt/{DdVDSbUA.js → CYlYSSNW.js} +1 -1
  7. package/dist/client/_nuxt/{CD5B-xvT.js → D5FIkDae.js} +1 -1
  8. package/dist/client/_nuxt/builds/latest.json +1 -1
  9. package/dist/client/_nuxt/builds/meta/f0b4dd20-8496-4003-b7a3-05cbae515923.json +1 -0
  10. package/dist/client/_nuxt/entry.C5SUNdim.css +1 -0
  11. package/dist/client/_nuxt/error-404.1K8v8Su2.css +1 -0
  12. package/dist/client/_nuxt/error-500.B9qvKpQm.css +1 -0
  13. package/dist/client/index.html +1 -1
  14. package/dist/module.d.mts +7 -19
  15. package/dist/module.d.ts +164 -0
  16. package/dist/module.json +1 -1
  17. package/dist/module.mjs +935 -645
  18. package/dist/registry.d.ts +6 -0
  19. package/dist/registry.mjs +235 -69
  20. package/dist/runtime/components/GoogleMaps/ScriptGoogleMaps.d.vue.ts +16 -9
  21. package/dist/runtime/components/GoogleMaps/ScriptGoogleMaps.vue +57 -30
  22. package/dist/runtime/components/GoogleMaps/ScriptGoogleMaps.vue.d.ts +16 -9
  23. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsAdvancedMarkerElement.d.vue.ts +22 -39
  24. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsAdvancedMarkerElement.vue +69 -72
  25. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsAdvancedMarkerElement.vue.d.ts +22 -39
  26. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsCircle.d.vue.ts +5 -1
  27. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsCircle.vue +25 -38
  28. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsCircle.vue.d.ts +5 -1
  29. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsGeoJson.d.vue.ts +43 -0
  30. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsGeoJson.vue +61 -0
  31. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsGeoJson.vue.d.ts +43 -0
  32. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsHeatmapLayer.d.vue.ts +4 -0
  33. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsHeatmapLayer.vue +22 -26
  34. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsHeatmapLayer.vue.d.ts +4 -0
  35. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsInfoWindow.d.vue.ts +9 -5
  36. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsInfoWindow.vue +62 -53
  37. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsInfoWindow.vue.d.ts +9 -5
  38. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsMarker.d.vue.ts +26 -11
  39. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsMarker.vue +48 -45
  40. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsMarker.vue.d.ts +26 -11
  41. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsMarkerClusterer.d.vue.ts +10 -2
  42. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsMarkerClusterer.vue +38 -37
  43. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsMarkerClusterer.vue.d.ts +10 -2
  44. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsOverlayView.d.vue.ts +63 -0
  45. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsOverlayView.vue +160 -0
  46. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsOverlayView.vue.d.ts +63 -0
  47. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPinElement.d.vue.ts +4 -0
  48. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPinElement.vue +23 -32
  49. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPinElement.vue.d.ts +4 -0
  50. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPolygon.d.vue.ts +7 -3
  51. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPolygon.vue +24 -38
  52. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPolygon.vue.d.ts +7 -3
  53. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPolyline.d.vue.ts +7 -3
  54. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPolyline.vue +24 -38
  55. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsPolyline.vue.d.ts +7 -3
  56. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsRectangle.d.vue.ts +7 -3
  57. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsRectangle.vue +25 -38
  58. package/dist/runtime/components/GoogleMaps/ScriptGoogleMapsRectangle.vue.d.ts +7 -3
  59. package/dist/runtime/components/GoogleMaps/bindGoogleMapsEvents.d.ts +13 -0
  60. package/dist/runtime/components/GoogleMaps/bindGoogleMapsEvents.js +8 -0
  61. package/dist/runtime/components/GoogleMaps/injectionKeys.d.ts +13 -0
  62. package/dist/runtime/components/GoogleMaps/injectionKeys.js +3 -0
  63. package/dist/runtime/components/GoogleMaps/useGoogleMapsResource.d.ts +26 -0
  64. package/dist/runtime/components/GoogleMaps/useGoogleMapsResource.js +42 -0
  65. package/dist/runtime/components/ScriptBlueskyEmbed.d.vue.ts +87 -0
  66. package/dist/runtime/components/ScriptBlueskyEmbed.vue +85 -0
  67. package/dist/runtime/components/ScriptBlueskyEmbed.vue.d.ts +87 -0
  68. package/dist/runtime/components/ScriptCrisp.vue +1 -1
  69. package/dist/runtime/components/ScriptGoogleAdsense.vue +1 -1
  70. package/dist/runtime/components/ScriptGravatar.d.vue.ts +22 -0
  71. package/dist/runtime/components/ScriptGravatar.vue +46 -0
  72. package/dist/runtime/components/ScriptGravatar.vue.d.ts +22 -0
  73. package/dist/runtime/components/ScriptInstagramEmbed.d.vue.ts +2 -2
  74. package/dist/runtime/components/ScriptInstagramEmbed.vue +5 -2
  75. package/dist/runtime/components/ScriptInstagramEmbed.vue.d.ts +2 -2
  76. package/dist/runtime/components/ScriptIntercom.vue +4 -3
  77. package/dist/runtime/components/ScriptPayPalButtons.d.vue.ts +43 -32
  78. package/dist/runtime/components/ScriptPayPalButtons.vue +48 -79
  79. package/dist/runtime/components/ScriptPayPalButtons.vue.d.ts +43 -32
  80. package/dist/runtime/components/ScriptPayPalMessages.d.vue.ts +37 -23
  81. package/dist/runtime/components/ScriptPayPalMessages.vue +46 -50
  82. package/dist/runtime/components/ScriptPayPalMessages.vue.d.ts +37 -23
  83. package/dist/runtime/components/ScriptStripePricingTable.vue +2 -2
  84. package/dist/runtime/components/ScriptVimeoPlayer.d.vue.ts +9 -0
  85. package/dist/runtime/components/ScriptVimeoPlayer.vue +13 -10
  86. package/dist/runtime/components/ScriptVimeoPlayer.vue.d.ts +9 -0
  87. package/dist/runtime/components/ScriptXEmbed.d.vue.ts +2 -2
  88. package/dist/runtime/components/ScriptXEmbed.vue +6 -3
  89. package/dist/runtime/components/ScriptXEmbed.vue.d.ts +2 -2
  90. package/dist/runtime/components/ScriptYouTubePlayer.d.vue.ts +2 -2
  91. package/dist/runtime/components/ScriptYouTubePlayer.vue +11 -5
  92. package/dist/runtime/components/ScriptYouTubePlayer.vue.d.ts +2 -2
  93. package/dist/runtime/composables/useScript.js +13 -6
  94. package/dist/runtime/composables/useScriptEventPage.js +2 -2
  95. package/dist/runtime/composables/useScriptTriggerConsent.d.ts +10 -0
  96. package/dist/runtime/composables/useScriptTriggerConsent.js +33 -20
  97. package/dist/runtime/composables/useScriptTriggerElement.js +1 -1
  98. package/dist/runtime/composables/useScriptTriggerIdleTimeout.js +1 -1
  99. package/dist/runtime/registry/bing-uet.d.ts +20 -0
  100. package/dist/runtime/registry/bing-uet.js +29 -0
  101. package/dist/runtime/registry/bluesky-embed.d.ts +116 -0
  102. package/dist/runtime/registry/bluesky-embed.js +72 -0
  103. package/dist/runtime/registry/clarity.d.ts +10 -15
  104. package/dist/runtime/registry/clarity.js +22 -31
  105. package/dist/runtime/registry/cloudflare-web-analytics.d.ts +2 -13
  106. package/dist/runtime/registry/cloudflare-web-analytics.js +2 -14
  107. package/dist/runtime/registry/crisp.d.ts +10 -40
  108. package/dist/runtime/registry/crisp.js +2 -33
  109. package/dist/runtime/registry/databuddy-analytics.d.ts +2 -35
  110. package/dist/runtime/registry/databuddy-analytics.js +20 -45
  111. package/dist/runtime/registry/fathom-analytics.d.ts +7 -26
  112. package/dist/runtime/registry/fathom-analytics.js +2 -24
  113. package/dist/runtime/registry/google-adsense.d.ts +3 -11
  114. package/dist/runtime/registry/google-adsense.js +2 -11
  115. package/dist/runtime/registry/google-analytics.d.ts +3 -5
  116. package/dist/runtime/registry/google-analytics.js +3 -8
  117. package/dist/runtime/registry/google-maps.d.ts +3 -9
  118. package/dist/runtime/registry/google-maps.js +2 -8
  119. package/dist/runtime/registry/google-recaptcha.d.ts +2 -6
  120. package/dist/runtime/registry/google-recaptcha.js +4 -12
  121. package/dist/runtime/registry/google-sign-in.d.ts +2 -13
  122. package/dist/runtime/registry/google-sign-in.js +2 -22
  123. package/dist/runtime/registry/google-tag-manager.d.ts +3 -28
  124. package/dist/runtime/registry/google-tag-manager.js +4 -27
  125. package/dist/runtime/registry/gravatar.d.ts +26 -0
  126. package/dist/runtime/registry/gravatar.js +33 -0
  127. package/dist/runtime/registry/hotjar.d.ts +4 -6
  128. package/dist/runtime/registry/hotjar.js +2 -5
  129. package/dist/runtime/registry/instagram-embed.d.ts +3 -18
  130. package/dist/runtime/registry/instagram-embed.js +4 -19
  131. package/dist/runtime/registry/intercom.d.ts +4 -12
  132. package/dist/runtime/registry/intercom.js +2 -12
  133. package/dist/runtime/registry/matomo-analytics.d.ts +3 -12
  134. package/dist/runtime/registry/matomo-analytics.js +3 -12
  135. package/dist/runtime/registry/meta-pixel.d.ts +4 -6
  136. package/dist/runtime/registry/meta-pixel.js +2 -4
  137. package/dist/runtime/registry/mixpanel-analytics.d.ts +22 -0
  138. package/dist/runtime/registry/mixpanel-analytics.js +46 -0
  139. package/dist/runtime/registry/npm.d.ts +3 -7
  140. package/dist/runtime/registry/npm.js +2 -9
  141. package/dist/runtime/registry/paypal.d.ts +4 -25
  142. package/dist/runtime/registry/paypal.js +3 -66
  143. package/dist/runtime/registry/plausible-analytics.js +18 -13
  144. package/dist/runtime/registry/posthog.d.ts +10 -12
  145. package/dist/runtime/registry/posthog.js +7 -14
  146. package/dist/runtime/registry/reddit-pixel.d.ts +5 -6
  147. package/dist/runtime/registry/reddit-pixel.js +2 -4
  148. package/dist/runtime/registry/rybbit-analytics.d.ts +2 -14
  149. package/dist/runtime/registry/rybbit-analytics.js +10 -20
  150. package/dist/runtime/registry/schemas.d.ts +982 -0
  151. package/dist/runtime/registry/schemas.js +937 -0
  152. package/dist/runtime/registry/segment.d.ts +2 -5
  153. package/dist/runtime/registry/segment.js +2 -5
  154. package/dist/runtime/registry/snapchat-pixel.d.ts +4 -33
  155. package/dist/runtime/registry/snapchat-pixel.js +2 -20
  156. package/dist/runtime/registry/stripe.d.ts +3 -4
  157. package/dist/runtime/registry/stripe.js +2 -4
  158. package/dist/runtime/registry/tiktok-pixel.d.ts +4 -7
  159. package/dist/runtime/registry/tiktok-pixel.js +2 -6
  160. package/dist/runtime/registry/umami-analytics.d.ts +2 -31
  161. package/dist/runtime/registry/umami-analytics.js +2 -36
  162. package/dist/runtime/registry/vercel-analytics.d.ts +29 -0
  163. package/dist/runtime/registry/vercel-analytics.js +84 -0
  164. package/dist/runtime/registry/vimeo-player.d.ts +2 -2
  165. package/dist/runtime/registry/vimeo-player.js +1 -1
  166. package/dist/runtime/registry/x-embed.d.ts +3 -17
  167. package/dist/runtime/registry/x-embed.js +3 -18
  168. package/dist/runtime/registry/x-pixel.d.ts +4 -7
  169. package/dist/runtime/registry/x-pixel.js +2 -5
  170. package/dist/runtime/registry/youtube-player.d.ts +7 -7
  171. package/dist/runtime/registry/youtube-player.js +1 -1
  172. package/dist/runtime/server/{sw-handler.d.ts → bluesky-embed-image.d.ts} +1 -1
  173. package/dist/runtime/server/bluesky-embed-image.js +7 -0
  174. package/dist/runtime/server/bluesky-embed.d.ts +16 -0
  175. package/dist/runtime/server/bluesky-embed.js +59 -0
  176. package/dist/runtime/server/google-maps-geocode-proxy.d.ts +2 -0
  177. package/dist/runtime/server/google-maps-geocode-proxy.js +34 -0
  178. package/dist/runtime/server/google-static-maps-proxy.js +2 -13
  179. package/dist/runtime/server/gravatar-proxy.d.ts +2 -0
  180. package/dist/runtime/server/gravatar-proxy.js +46 -0
  181. package/dist/runtime/server/instagram-embed-asset.js +8 -41
  182. package/dist/runtime/server/instagram-embed-image.js +6 -53
  183. package/dist/runtime/server/instagram-embed.d.ts +16 -0
  184. package/dist/runtime/server/instagram-embed.js +173 -35
  185. package/dist/runtime/server/proxy-handler.js +134 -93
  186. package/dist/runtime/server/utils/image-proxy.d.ts +12 -0
  187. package/dist/runtime/server/utils/image-proxy.js +70 -0
  188. package/dist/runtime/server/utils/privacy.d.ts +1 -2
  189. package/dist/runtime/server/utils/privacy.js +54 -34
  190. package/dist/runtime/server/x-embed-image.js +5 -49
  191. package/dist/runtime/server/x-embed.js +3 -2
  192. package/dist/runtime/types.d.ts +74 -40
  193. package/dist/runtime/utils/pure.d.ts +1 -5
  194. package/dist/runtime/utils/pure.js +0 -67
  195. package/dist/runtime/utils.d.ts +4 -3
  196. package/dist/runtime/utils.js +24 -10
  197. package/dist/shared/scripts.ViOoYQXH.mjs +381 -0
  198. package/dist/stats.d.mts +202 -0
  199. package/dist/stats.d.ts +202 -0
  200. package/dist/stats.mjs +3868 -0
  201. package/dist/types-source.d.mts +17 -0
  202. package/dist/types-source.d.ts +17 -0
  203. package/dist/types-source.mjs +3600 -0
  204. package/package.json +52 -38
  205. package/dist/client/_nuxt/D-kOnTuH.js +0 -162
  206. package/dist/client/_nuxt/builds/meta/f1474569-6922-450d-bc3f-4fd5f3e1391a.json +0 -1
  207. package/dist/client/_nuxt/entry.D45OuV0w.css +0 -1
  208. package/dist/client/_nuxt/error-404.B57D-jUQ.css +0 -1
  209. package/dist/client/_nuxt/error-500.DTHUW7BI.css +0 -1
  210. package/dist/runtime/components/ScriptPayPalMarks.d.vue.ts +0 -52
  211. package/dist/runtime/components/ScriptPayPalMarks.vue +0 -69
  212. package/dist/runtime/components/ScriptPayPalMarks.vue.d.ts +0 -52
  213. package/dist/runtime/plugins/sw-register.client.d.ts +0 -2
  214. package/dist/runtime/plugins/sw-register.client.js +0 -12
  215. package/dist/runtime/server/sw-handler.js +0 -25
  216. package/dist/runtime/sw/proxy-sw.template.d.ts +0 -1
  217. package/dist/runtime/sw/proxy-sw.template.js +0 -54
@@ -0,0 +1,34 @@
1
+ import { useRuntimeConfig } from "#imports";
2
+ import { createError, defineEventHandler, getQuery, setHeader } from "h3";
3
+ import { $fetch } from "ofetch";
4
+ import { withQuery } from "ufo";
5
+ export default defineEventHandler(async (event) => {
6
+ const runtimeConfig = useRuntimeConfig();
7
+ const privateConfig = runtimeConfig["nuxt-scripts"]?.googleMapsGeocodeProxy;
8
+ const apiKey = privateConfig?.apiKey;
9
+ if (!apiKey) {
10
+ throw createError({
11
+ statusCode: 500,
12
+ statusMessage: "Google Maps API key not configured for geocode proxy"
13
+ });
14
+ }
15
+ const query = getQuery(event);
16
+ const { key: _clientKey, ...safeQuery } = query;
17
+ const geocodeUrl = withQuery("https://maps.googleapis.com/maps/api/geocode/json", {
18
+ ...safeQuery,
19
+ key: apiKey
20
+ });
21
+ const data = await $fetch(geocodeUrl, {
22
+ headers: {
23
+ "User-Agent": "Nuxt Scripts Google Geocode Proxy"
24
+ }
25
+ }).catch((error) => {
26
+ throw createError({
27
+ statusCode: error.statusCode || 500,
28
+ statusMessage: error.statusMessage || "Failed to geocode"
29
+ });
30
+ });
31
+ setHeader(event, "Content-Type", "application/json");
32
+ setHeader(event, "Cache-Control", "public, max-age=86400, s-maxage=86400");
33
+ return data;
34
+ });
@@ -1,7 +1,7 @@
1
- import { createError, defineEventHandler, getHeader, getQuery, setHeader } from "h3";
1
+ import { useRuntimeConfig } from "#imports";
2
+ import { createError, defineEventHandler, getQuery, setHeader } from "h3";
2
3
  import { $fetch } from "ofetch";
3
4
  import { withQuery } from "ufo";
4
- import { useRuntimeConfig } from "#imports";
5
5
  export default defineEventHandler(async (event) => {
6
6
  const runtimeConfig = useRuntimeConfig();
7
7
  const publicConfig = runtimeConfig.public["nuxt-scripts"]?.googleStaticMapsProxy;
@@ -19,17 +19,6 @@ export default defineEventHandler(async (event) => {
19
19
  statusMessage: "Google Maps API key not configured for proxy"
20
20
  });
21
21
  }
22
- const referer = getHeader(event, "referer");
23
- const host = getHeader(event, "host");
24
- if (referer && host) {
25
- const refererUrl = new URL(referer).host;
26
- if (refererUrl !== host) {
27
- throw createError({
28
- statusCode: 403,
29
- statusMessage: "Invalid referer"
30
- });
31
- }
32
- }
33
22
  const query = getQuery(event);
34
23
  const { key: _clientKey, ...safeQuery } = query;
35
24
  const googleMapsUrl = withQuery("https://maps.googleapis.com/maps/api/staticmap", {
@@ -0,0 +1,2 @@
1
+ declare const _default: import("h3").EventHandler<import("h3").EventHandlerRequest, Promise<any>>;
2
+ export default _default;
@@ -0,0 +1,46 @@
1
+ import { useRuntimeConfig } from "#imports";
2
+ import { createError, defineEventHandler, getQuery, setHeader } from "h3";
3
+ import { $fetch } from "ofetch";
4
+ import { withQuery } from "ufo";
5
+ export default defineEventHandler(async (event) => {
6
+ const runtimeConfig = useRuntimeConfig();
7
+ const proxyConfig = runtimeConfig.public["nuxt-scripts"]?.gravatarProxy;
8
+ const query = getQuery(event);
9
+ let hash = query.hash;
10
+ const email = query.email;
11
+ if (!hash && email) {
12
+ const encoder = new TextEncoder();
13
+ const data = encoder.encode(email.trim().toLowerCase());
14
+ const hashBuffer = await crypto.subtle.digest("SHA-256", data);
15
+ hash = Array.from(new Uint8Array(hashBuffer), (b) => b.toString(16).padStart(2, "0")).join("");
16
+ }
17
+ if (!hash) {
18
+ throw createError({
19
+ statusCode: 400,
20
+ statusMessage: "Either hash or email parameter is required"
21
+ });
22
+ }
23
+ const size = query.s || "80";
24
+ const defaultImg = query.d || "mp";
25
+ const rating = query.r || "g";
26
+ const gravatarUrl = withQuery(`https://www.gravatar.com/avatar/${hash}`, {
27
+ s: size,
28
+ d: defaultImg,
29
+ r: rating
30
+ });
31
+ const response = await $fetch.raw(gravatarUrl, {
32
+ headers: {
33
+ "User-Agent": "Nuxt Scripts Gravatar Proxy"
34
+ }
35
+ }).catch((error) => {
36
+ throw createError({
37
+ statusCode: error.statusCode || 500,
38
+ statusMessage: error.statusMessage || "Failed to fetch Gravatar avatar"
39
+ });
40
+ });
41
+ const cacheMaxAge = proxyConfig?.cacheMaxAge ?? 3600;
42
+ setHeader(event, "Content-Type", response.headers.get("content-type") || "image/jpeg");
43
+ setHeader(event, "Cache-Control", `public, max-age=${cacheMaxAge}, s-maxage=${cacheMaxAge}`);
44
+ setHeader(event, "Vary", "Accept-Encoding");
45
+ return response._data;
46
+ });
@@ -1,42 +1,9 @@
1
- import { createError, defineEventHandler, getQuery, setHeader } from "h3";
2
- import { $fetch } from "ofetch";
3
- export default defineEventHandler(async (event) => {
4
- const query = getQuery(event);
5
- const url = query.url?.replace(/&amp;/g, "&");
6
- if (!url) {
7
- throw createError({
8
- statusCode: 400,
9
- statusMessage: "Asset URL is required"
10
- });
11
- }
12
- let parsedUrl;
13
- try {
14
- parsedUrl = new URL(url);
15
- } catch {
16
- throw createError({
17
- statusCode: 400,
18
- statusMessage: "Invalid asset URL"
19
- });
20
- }
21
- if (parsedUrl.hostname !== "static.cdninstagram.com") {
22
- throw createError({
23
- statusCode: 403,
24
- statusMessage: "Domain not allowed"
25
- });
26
- }
27
- const response = await $fetch.raw(url, {
28
- headers: {
29
- "Accept": "*/*",
30
- "User-Agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36"
31
- }
32
- }).catch((error) => {
33
- throw createError({
34
- statusCode: error.statusCode || 500,
35
- statusMessage: error.statusMessage || "Failed to fetch asset"
36
- });
37
- });
38
- const contentType = response.headers.get("content-type") || "application/octet-stream";
39
- setHeader(event, "Content-Type", contentType);
40
- setHeader(event, "Cache-Control", "public, max-age=86400, s-maxage=86400");
41
- return response._data;
1
+ import { createImageProxyHandler } from "./utils/image-proxy.js";
2
+ export default createImageProxyHandler({
3
+ allowedDomains: ["static.cdninstagram.com"],
4
+ accept: "*/*",
5
+ userAgent: "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36",
6
+ cacheMaxAge: 86400,
7
+ contentType: "application/octet-stream",
8
+ decodeAmpersands: true
42
9
  });
@@ -1,54 +1,7 @@
1
- import { createError, defineEventHandler, getQuery, setHeader } from "h3";
2
- import { $fetch } from "ofetch";
3
- export default defineEventHandler(async (event) => {
4
- const query = getQuery(event);
5
- const url = query.url?.replace(/&amp;/g, "&");
6
- if (!url) {
7
- throw createError({
8
- statusCode: 400,
9
- statusMessage: "Image URL is required"
10
- });
11
- }
12
- let parsedUrl;
13
- try {
14
- parsedUrl = new URL(url);
15
- } catch {
16
- throw createError({
17
- statusCode: 400,
18
- statusMessage: "Invalid image URL"
19
- });
20
- }
21
- if (parsedUrl.protocol !== "http:" && parsedUrl.protocol !== "https:") {
22
- throw createError({
23
- statusCode: 400,
24
- statusMessage: "Invalid URL scheme"
25
- });
26
- }
27
- if (!parsedUrl.hostname.endsWith(".cdninstagram.com") && parsedUrl.hostname !== "scontent.cdninstagram.com") {
28
- throw createError({
29
- statusCode: 403,
30
- statusMessage: "Domain not allowed"
31
- });
32
- }
33
- const response = await $fetch.raw(url, {
34
- redirect: "manual",
35
- headers: {
36
- "Accept": "image/webp,image/jpeg,image/png,image/*,*/*;q=0.8",
37
- "User-Agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36"
38
- }
39
- }).catch((error) => {
40
- throw createError({
41
- statusCode: error.statusCode || 500,
42
- statusMessage: error.statusMessage || "Failed to fetch image"
43
- });
44
- });
45
- if (response.status >= 300 && response.status < 400) {
46
- throw createError({
47
- statusCode: 403,
48
- statusMessage: "Redirects not allowed"
49
- });
50
- }
51
- setHeader(event, "Content-Type", response.headers.get("content-type") || "image/jpeg");
52
- setHeader(event, "Cache-Control", "public, max-age=3600, s-maxage=3600");
53
- return response._data;
1
+ import { createImageProxyHandler } from "./utils/image-proxy.js";
2
+ export default createImageProxyHandler({
3
+ allowedDomains: (hostname) => hostname.endsWith(".cdninstagram.com") || hostname === "scontent.cdninstagram.com",
4
+ userAgent: "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36",
5
+ followRedirects: false,
6
+ decodeAmpersands: true
54
7
  });
@@ -1,2 +1,18 @@
1
+ export declare const RSRC_RE: RegExp;
2
+ export declare const AMP_RE: RegExp;
3
+ export declare const SCONTENT_RE: RegExp;
4
+ export declare const STATIC_CDN_RE: RegExp;
5
+ export declare const LOOKASIDE_RE: RegExp;
6
+ export declare const INSTAGRAM_IMAGE_HOSTS: string[];
7
+ export declare const INSTAGRAM_ASSET_HOST = "static.cdninstagram.com";
8
+ export declare function proxyImageUrl(url: string): string;
9
+ export declare function proxyAssetUrl(url: string): string;
10
+ export declare function rewriteUrl(url: string): string;
11
+ export declare function rewriteUrlsInText(text: string): string;
12
+ /**
13
+ * Scope CSS rules under a parent selector and strip global/page-level rules.
14
+ * Removes :root, html, body selectors and @charset/@import at-rules.
15
+ */
16
+ export declare function scopeCss(css: string, scopeSelector: string): string;
1
17
  declare const _default: import("h3").EventHandler<import("h3").EventHandlerRequest, Promise<string>>;
2
18
  export default _default;
@@ -1,5 +1,128 @@
1
1
  import { createError, defineEventHandler, getQuery, setHeader } from "h3";
2
2
  import { $fetch } from "ofetch";
3
+ import { ELEMENT_NODE, parse, renderSync, TEXT_NODE, walkSync } from "ultrahtml";
4
+ export const RSRC_RE = /url\(\/rsrc\.php([^)]+)\)/g;
5
+ export const AMP_RE = /&amp;/g;
6
+ export const SCONTENT_RE = /https:\/\/scontent[^"'\s),]+\.cdninstagram\.com[^"'\s),]+/g;
7
+ export const STATIC_CDN_RE = /https:\/\/static\.cdninstagram\.com[^"'\s),]+/g;
8
+ export const LOOKASIDE_RE = /https:\/\/lookaside\.instagram\.com[^"'\s),]+/g;
9
+ export const INSTAGRAM_IMAGE_HOSTS = ["scontent.cdninstagram.com", "lookaside.instagram.com"];
10
+ export const INSTAGRAM_ASSET_HOST = "static.cdninstagram.com";
11
+ const CHARSET_RE = /@charset\s[^;]+;/gi;
12
+ const IMPORT_RE = /@import\s[^;]+;/gi;
13
+ const WHITESPACE_RE = /\s/;
14
+ const AT_RULE_NAME_RE = /@([\w-]+)/;
15
+ const MULTI_SPACE_RE = /\s+/g;
16
+ const SRCSET_SPLIT_RE = /\s+/;
17
+ export function proxyImageUrl(url) {
18
+ return `/_scripts/embed/instagram-image?url=${encodeURIComponent(url.replace(AMP_RE, "&"))}`;
19
+ }
20
+ export function proxyAssetUrl(url) {
21
+ return `/_scripts/embed/instagram-asset?url=${encodeURIComponent(url.replace(AMP_RE, "&"))}`;
22
+ }
23
+ export function rewriteUrl(url) {
24
+ try {
25
+ const parsed = new URL(url);
26
+ if (parsed.hostname === INSTAGRAM_ASSET_HOST)
27
+ return proxyAssetUrl(url);
28
+ if (INSTAGRAM_IMAGE_HOSTS.some((h) => parsed.hostname === h || parsed.hostname.endsWith(`.cdninstagram.com`)))
29
+ return proxyImageUrl(url);
30
+ } catch {
31
+ }
32
+ return url;
33
+ }
34
+ export function rewriteUrlsInText(text) {
35
+ return text.replace(SCONTENT_RE, (m) => proxyImageUrl(m)).replace(STATIC_CDN_RE, (m) => proxyAssetUrl(m)).replace(LOOKASIDE_RE, (m) => proxyImageUrl(m));
36
+ }
37
+ function removeNode(node) {
38
+ node.type = TEXT_NODE;
39
+ node.value = "";
40
+ node.name = void 0;
41
+ node.attributes = {};
42
+ node.children = [];
43
+ }
44
+ export function scopeCss(css, scopeSelector) {
45
+ let result = css.replace(CHARSET_RE, "");
46
+ result = result.replace(IMPORT_RE, "");
47
+ return processRules(result, scopeSelector);
48
+ }
49
+ function processRules(css, scopeSelector) {
50
+ const output = [];
51
+ let i = 0;
52
+ while (i < css.length) {
53
+ while (i < css.length && WHITESPACE_RE.test(css[i])) i++;
54
+ if (i >= css.length)
55
+ break;
56
+ if (css[i] === "@") {
57
+ const atRule = extractAtRule(css, i);
58
+ if (atRule) {
59
+ const atName = atRule.content.match(AT_RULE_NAME_RE)?.[1]?.toLowerCase();
60
+ if (atName === "media" || atName === "supports" || atName === "layer") {
61
+ const braceStart = atRule.content.indexOf("{");
62
+ const innerCss = atRule.content.slice(braceStart + 1, -1);
63
+ const scopedInner = processRules(innerCss, scopeSelector);
64
+ output.push(`${atRule.content.slice(0, braceStart + 1) + scopedInner}}`);
65
+ } else if (atName === "keyframes" || atName === "-webkit-keyframes" || atName === "font-face") {
66
+ output.push(atRule.content);
67
+ }
68
+ i = atRule.end;
69
+ continue;
70
+ }
71
+ }
72
+ const bracePos = css.indexOf("{", i);
73
+ if (bracePos === -1)
74
+ break;
75
+ const selector = css.slice(i, bracePos).trim();
76
+ const block = extractBlock(css, bracePos);
77
+ if (!block)
78
+ break;
79
+ i = block.end;
80
+ if (!selector)
81
+ continue;
82
+ const selectors = selector.split(",").map((s) => s.trim());
83
+ const filteredSelectors = selectors.filter((s) => {
84
+ const normalized = s.replace(MULTI_SPACE_RE, " ").trim().toLowerCase();
85
+ return normalized !== ":root" && normalized !== "html" && normalized !== "body" && !normalized.startsWith(":root ") && !normalized.startsWith("html ") && !normalized.startsWith("body ") && normalized !== "html, body";
86
+ });
87
+ if (filteredSelectors.length === 0)
88
+ continue;
89
+ const scopedSelectors = filteredSelectors.map((s) => {
90
+ return `${scopeSelector} ${s}`;
91
+ });
92
+ output.push(`${scopedSelectors.join(", ")} ${block.content}`);
93
+ }
94
+ return output.join("\n");
95
+ }
96
+ function extractAtRule(css, start) {
97
+ const bracePos = css.indexOf("{", start);
98
+ const semiPos = css.indexOf(";", start);
99
+ if (semiPos !== -1 && (bracePos === -1 || semiPos < bracePos)) {
100
+ return { content: css.slice(start, semiPos + 1), end: semiPos + 1 };
101
+ }
102
+ if (bracePos === -1)
103
+ return null;
104
+ const block = extractBlock(css, bracePos);
105
+ if (!block)
106
+ return null;
107
+ return {
108
+ content: css.slice(start, bracePos) + block.content,
109
+ end: block.end
110
+ };
111
+ }
112
+ function extractBlock(css, openBrace) {
113
+ let depth = 0;
114
+ for (let j = openBrace; j < css.length; j++) {
115
+ if (css[j] === "{") {
116
+ depth++;
117
+ } else if (css[j] === "}") {
118
+ depth--;
119
+ if (depth === 0) {
120
+ return { content: css.slice(openBrace, j + 1), end: j + 1 };
121
+ }
122
+ }
123
+ }
124
+ return null;
125
+ }
3
126
  export default defineEventHandler(async (event) => {
4
127
  const query = getQuery(event);
5
128
  const postUrl = query.url;
@@ -27,11 +150,10 @@ export default defineEventHandler(async (event) => {
27
150
  }
28
151
  const pathname = parsedUrl.pathname.endsWith("/") ? parsedUrl.pathname : `${parsedUrl.pathname}/`;
29
152
  const cleanUrl = parsedUrl.origin + pathname;
30
- const embedUrl = cleanUrl + "embed/" + (captions ? "captioned/" : "");
153
+ const embedUrl = `${cleanUrl}embed/${captions ? "captioned/" : ""}`;
31
154
  const html = await $fetch(embedUrl, {
32
155
  headers: {
33
156
  "Accept": "text/html",
34
- // Use simple UA - full Chrome UA triggers JS-heavy version without static content
35
157
  "User-Agent": "Mozilla/5.0 (compatible; Googlebot/2.1; +http://www.google.com/bot.html)"
36
158
  }
37
159
  }).catch((error) => {
@@ -40,18 +162,45 @@ export default defineEventHandler(async (event) => {
40
162
  statusMessage: error.statusMessage || "Failed to fetch Instagram embed"
41
163
  });
42
164
  });
165
+ const ast = parse(html);
43
166
  const cssUrls = [];
44
- const linkRegex = /<link[^>]+rel=["']stylesheet["'][^>]+href=["']([^"']+)["'][^>]*>/gi;
45
- let match;
46
- while ((match = linkRegex.exec(html)) !== null) {
47
- if (match[1])
48
- cssUrls.push(match[1]);
49
- }
50
- const linkRegex2 = /<link[^>]+href=["']([^"']+)["'][^>]+rel=["']stylesheet["'][^>]*>/gi;
51
- while ((match = linkRegex2.exec(html)) !== null) {
52
- if (match[1])
53
- cssUrls.push(match[1]);
54
- }
167
+ walkSync(ast, (node) => {
168
+ if (node.type !== ELEMENT_NODE)
169
+ return;
170
+ if (node.name === "link" && node.attributes.rel === "stylesheet" && node.attributes.href) {
171
+ cssUrls.push(node.attributes.href);
172
+ removeNode(node);
173
+ return;
174
+ }
175
+ if (node.name === "script" || node.name === "noscript" || node.name === "style") {
176
+ removeNode(node);
177
+ return;
178
+ }
179
+ for (const attr of ["src", "poster"]) {
180
+ if (node.attributes[attr])
181
+ node.attributes[attr] = rewriteUrl(node.attributes[attr]);
182
+ }
183
+ if (node.attributes.srcset) {
184
+ node.attributes.srcset = node.attributes.srcset.split(",").map((entry) => {
185
+ const parts = entry.trim().split(SRCSET_SPLIT_RE);
186
+ const url = parts[0];
187
+ const descriptor = parts.slice(1).join(" ");
188
+ return url ? `${rewriteUrl(url)}${descriptor ? ` ${descriptor}` : ""}` : entry;
189
+ }).join(", ");
190
+ }
191
+ if (node.attributes.style)
192
+ node.attributes.style = rewriteUrlsInText(node.attributes.style);
193
+ });
194
+ walkSync(ast, (node) => {
195
+ if (node.type === TEXT_NODE && node.value)
196
+ node.value = rewriteUrlsInText(node.value);
197
+ });
198
+ let bodyNode = null;
199
+ walkSync(ast, (node) => {
200
+ if (node.type === ELEMENT_NODE && node.name === "body")
201
+ bodyNode = node;
202
+ });
203
+ const bodyHtml = bodyNode ? bodyNode.children.map((child) => renderSync(child)).join("") : renderSync(ast);
55
204
  const cssContents = await Promise.all(
56
205
  cssUrls.map(
57
206
  (url) => $fetch(url, {
@@ -61,31 +210,20 @@ export default defineEventHandler(async (event) => {
61
210
  );
62
211
  let combinedCss = cssContents.join("\n");
63
212
  combinedCss = combinedCss.replace(
64
- /url\(\/rsrc\.php([^)]+)\)/g,
65
- (_m, path) => `url(/api/_scripts/instagram-embed-asset?url=${encodeURIComponent(`https://static.cdninstagram.com/rsrc.php${path}`)})`
213
+ RSRC_RE,
214
+ (_m, path) => `url(/_scripts/embed/instagram-asset?url=${encodeURIComponent(`https://static.cdninstagram.com/rsrc.php${path}`)})`
66
215
  );
216
+ combinedCss = rewriteUrlsInText(combinedCss);
217
+ combinedCss = scopeCss(combinedCss, ".instagram-embed-root");
67
218
  const baseStyles = `
68
- html { background: white; max-width: 540px; width: calc(100% - 2px); border-radius: 3px; border: 1px solid rgb(219, 219, 219); display: block; margin: 0px 0px 12px; min-width: 326px; padding: 0px; }
69
- #splash-screen { display: none !important; }
70
- .Embed { opacity: 1 !important; visibility: visible !important; }
71
- .EmbeddedMedia, .EmbeddedMediaImage { display: block !important; visibility: visible !important; }
219
+ .instagram-embed-root { background: white; max-width: 540px; width: calc(100% - 2px); border-radius: 3px; border: 1px solid rgb(219, 219, 219); display: block; margin: 0px 0px 12px; min-width: 326px; padding: 0px; }
220
+ .instagram-embed-root #splash-screen { display: none !important; }
221
+ .instagram-embed-root .Embed { opacity: 1 !important; visibility: visible !important; }
222
+ .instagram-embed-root .EmbeddedMedia, .instagram-embed-root .EmbeddedMediaImage { display: block !important; visibility: visible !important; }
72
223
  `;
73
- let rewrittenHtml = html.replace(/<script[\s\S]*?<\/script>/gi, "").replace(/<link[^>]+rel=["']stylesheet["'][^>]*>/gi, "").replace(/<link[^>]+href=["'][^"']+\.css[^"']*["'][^>]*>/gi, "").replace(/<noscript>[\s\S]*?<\/noscript>/gi, "").replace(
74
- /https:\/\/scontent[^"'\s),]+\.cdninstagram\.com[^"'\s),]+/g,
75
- (m) => `/api/_scripts/instagram-embed-image?url=${encodeURIComponent(m.replace(/&amp;/g, "&"))}`
76
- ).replace(
77
- /https:\/\/static\.cdninstagram\.com[^"'\s),]+/g,
78
- (m) => `/api/_scripts/instagram-embed-asset?url=${encodeURIComponent(m.replace(/&amp;/g, "&"))}`
79
- ).replace(
80
- /https:\/\/lookaside\.instagram\.com[^"'\s),]+/g,
81
- (m) => `/api/_scripts/instagram-embed-image?url=${encodeURIComponent(m.replace(/&amp;/g, "&"))}`
82
- );
83
- rewrittenHtml = rewrittenHtml.replace(
84
- "</head>",
85
- `<style>${baseStyles}
86
- ${combinedCss}</style></head>`
87
- );
224
+ const result = `<div class="instagram-embed-root"><style>${baseStyles}
225
+ ${combinedCss}</style>${bodyHtml}</div>`;
88
226
  setHeader(event, "Content-Type", "text/html");
89
227
  setHeader(event, "Cache-Control", "public, max-age=600, s-maxage=600");
90
- return rewrittenHtml;
228
+ return result;
91
229
  });