babel-plugin-sfcc-modules
Version:
Babel plugin to handle non-standard module paths used by Salesforce Commerce Cloud (SFCC)
107 lines (92 loc) • 3.89 kB
JavaScript
const fs = require('node:fs')
const path = require('node:path')
const importsVisitor = require('imports-visitor')
const t = require('@babel/types')
const SUPPORTED_EXTENSIONS = ['js', 'ds', 'json']
const getModulePath = (moduleName, basePath, cartridge, target) => {
const relativePath = path.relative(
path.dirname(moduleName),
`${basePath}/${cartridge}${target}`,
)
return (relativePath.includes('.') ? '' : './') + relativePath
}
const plugin = (babel, { cartridgePath, basePath }) => ({
visitor: {
Program(thePath, state) {
const imports = []
thePath.traverse(importsVisitor, { imports })
// eslint-disable-next-line no-restricted-syntax
for (const imp of imports) {
/**
* Finds and sets the rewrittten module path.
*
* @param findCartridge a function to find the cartridge
*/
const resolve = (findCartridge) => {
const target = imp.source.slice(1)
const foundCartridge = findCartridge(target)
if (foundCartridge) {
imp.source = getModulePath(state.file.opts.filename, basePath, foundCartridge, target)
}
}
// Handle
//
// require('*/cartridge/scripts/foo')
//
// Find the first cartridge that matches the requested module name
//
if (imp.source.indexOf('*/') === 0) {
resolve((target) => cartridgePath
.find((cartridge) => SUPPORTED_EXTENSIONS.find(
// eslint-disable-next-line sonarjs/no-nested-functions
(extension) => fs.existsSync(`${basePath}/${cartridge}${target}.${extension}`),
)))
}
// Handle
//
// require('~/cartridge/scripts/foo')
//
// Own cartridge - rewrites the module path to a relative URL.
//
if (imp.source.indexOf('~/') === 0) {
resolve(() => path.relative(basePath, path.dirname(state.file.opts.filename)).split(path.sep)[0])
}
}
},
MemberExpression(thePath, state) {
// Find "module.superModule"
if (thePath.node.object.type === 'Identifier'
&& thePath.node.object.name === 'module' && thePath.node.property.name === 'superModule') {
// path to module relative to the cartridge base path
const pathToModule = path.relative(basePath, state.file.opts.filename)
// Path without cartridge name
let shortendedPathToModule = pathToModule
.split(path.sep)
.slice(1)
// Remove extension
shortendedPathToModule[shortendedPathToModule.length - 1] = shortendedPathToModule.at(
-1,
)
.split('.')
shortendedPathToModule.at(-1).pop()
shortendedPathToModule.at(-1).join('.')
shortendedPathToModule = `/${shortendedPathToModule.join(path.sep)}`
const cartridge = pathToModule.split(path.sep)[0] // the own cartridge name
// all cartridges after the found cartridges in cartridge path
const newCartridgePath = cartridgePath.slice(cartridgePath.indexOf(cartridge) + 1)
// Find the the cartridge which contains the next match for the module path
const foundCartridge = newCartridgePath.find((theCartridge) => SUPPORTED_EXTENSIONS
.find((extension) => fs.existsSync(`${basePath}/${theCartridge}${shortendedPathToModule}.${extension}`)))
let foundRequire
if (foundCartridge) {
foundRequire = getModulePath(state.file.opts.filename, basePath, foundCartridge, shortendedPathToModule)
}
// Replace "module.superModule" with a require() or undefined
thePath.replaceWith(foundRequire
? t.callExpression(t.identifier('require'), [t.stringLiteral(foundRequire)])
: t.identifier('undefined'))
}
},
},
})
module.exports = plugin