logo

pleroma-fe

My custom branche(s) on git.pleroma.social/pleroma/pleroma-fe git clone https://anongit.hacktivis.me/git/pleroma-fe.git/

sw_plugin.js (4421B)


  1. import { fileURLToPath } from 'node:url'
  2. import { dirname, resolve } from 'node:path'
  3. import { readFile } from 'node:fs/promises'
  4. import { build } from 'vite'
  5. import * as esbuild from 'esbuild'
  6. import { generateServiceWorkerMessages, i18nFiles } from './service_worker_messages.js'
  7. const getSWMessagesAsText = async () => {
  8. const messages = await generateServiceWorkerMessages()
  9. return `export default ${JSON.stringify(messages, undefined, 2)}`
  10. }
  11. const projectRoot = dirname(dirname(fileURLToPath(import.meta.url)))
  12. export const devSwPlugin = ({
  13. swSrc,
  14. swDest,
  15. transformSW,
  16. alias
  17. }) => {
  18. const swFullSrc = resolve(projectRoot, swSrc)
  19. const esbuildAlias = {}
  20. Object.entries(alias).forEach(([source, dest]) => {
  21. esbuildAlias[source] = dest.startsWith('/') ? projectRoot + dest : dest
  22. })
  23. return {
  24. name: 'dev-sw-plugin',
  25. apply: 'serve',
  26. configResolved (conf) {
  27. },
  28. resolveId (id) {
  29. const name = id.startsWith('/') ? id.slice(1) : id
  30. if (name === swDest) {
  31. return swFullSrc
  32. }
  33. return null
  34. },
  35. async load (id) {
  36. if (id === swFullSrc) {
  37. return readFile(swFullSrc, 'utf-8')
  38. }
  39. return null
  40. },
  41. /**
  42. * vite does not bundle the service worker
  43. * during dev, and firefox does not support ESM as service worker
  44. * https://bugzilla.mozilla.org/show_bug.cgi?id=1360870
  45. */
  46. async transform (code, id) {
  47. if (id === swFullSrc && transformSW) {
  48. const res = await esbuild.build({
  49. entryPoints: [swSrc],
  50. bundle: true,
  51. write: false,
  52. outfile: 'sw-pleroma.js',
  53. alias: esbuildAlias,
  54. plugins: [{
  55. name: 'vite-like-root-resolve',
  56. setup (b) {
  57. b.onResolve(
  58. { filter: new RegExp(/^\//) },
  59. args => ({
  60. path: resolve(projectRoot, args.path.slice(1))
  61. })
  62. )
  63. }
  64. }, {
  65. name: 'sw-messages',
  66. setup (b) {
  67. b.onResolve(
  68. { filter: new RegExp('^' + swMessagesName + '$') },
  69. args => ({
  70. path: args.path,
  71. namespace: 'sw-messages'
  72. }))
  73. b.onLoad(
  74. { filter: /.*/, namespace: 'sw-messages' },
  75. async () => ({
  76. contents: await getSWMessagesAsText()
  77. }))
  78. }
  79. }]
  80. })
  81. const text = res.outputFiles[0].text
  82. return text
  83. }
  84. }
  85. }
  86. }
  87. // Idea taken from
  88. // https://github.com/vite-pwa/vite-plugin-pwa/blob/main/src/plugins/build.ts
  89. // rollup does not support compiling to iife if we want to code-split;
  90. // however, we must compile the service worker to iife because of browser support.
  91. // Run another vite build just for the service worker targeting iife at
  92. // the end of the build.
  93. export const buildSwPlugin = ({
  94. swSrc,
  95. swDest,
  96. }) => {
  97. let config
  98. return {
  99. name: 'build-sw-plugin',
  100. enforce: 'post',
  101. apply: 'build',
  102. configResolved (resolvedConfig) {
  103. config = {
  104. define: resolvedConfig.define,
  105. resolve: resolvedConfig.resolve,
  106. plugins: [swMessagesPlugin()],
  107. publicDir: false,
  108. build: {
  109. ...resolvedConfig.build,
  110. lib: {
  111. entry: swSrc,
  112. formats: ['iife'],
  113. name: 'sw_pleroma'
  114. },
  115. emptyOutDir: false,
  116. rollupOptions: {
  117. output: {
  118. entryFileNames: swDest
  119. }
  120. }
  121. },
  122. configFile: false
  123. }
  124. },
  125. closeBundle: {
  126. order: 'post',
  127. sequential: true,
  128. async handler () {
  129. console.log('Building service worker for production')
  130. await build(config)
  131. }
  132. }
  133. }
  134. }
  135. const swMessagesName = 'virtual:pleroma-fe/service_worker_messages'
  136. const swMessagesNameResolved = '\0' + swMessagesName
  137. export const swMessagesPlugin = () => {
  138. return {
  139. name: 'sw-messages-plugin',
  140. resolveId (id) {
  141. if (id === swMessagesName) {
  142. Object.values(i18nFiles).forEach(f => {
  143. this.addWatchFile(f)
  144. })
  145. return swMessagesNameResolved
  146. } else {
  147. return null
  148. }
  149. },
  150. async load (id) {
  151. if (id === swMessagesNameResolved) {
  152. return await getSWMessagesAsText()
  153. }
  154. return null
  155. }
  156. }
  157. }