/* eslint-disable no-param-reassign */ const { statSync, existsSync } = require('fs'); const path = require('path'); const glob = require('glob'); const sass = require('sass'); const webpack = require('webpack'); const { red } = require('chalk'); const IS_EE = require('../../config/helpers/is_ee_env'); const IS_JH = require('../../config/helpers/is_jh_env'); const gitlabWebpackConfig = require('../../config/webpack.config'); const ROOT_PATH = path.resolve(__dirname, '..', '..'); const EMPTY_VUE_COMPONENT_PATH = path.join( ROOT_PATH, 'app/assets/javascripts/vue_shared/components/empty_component.js', ); const buildIncludePaths = (nodeSassIncludePaths, previouslyResolvedPath) => { const includePaths = []; if (path.isAbsolute(previouslyResolvedPath)) { includePaths.push(path.dirname(previouslyResolvedPath)); } return [...new Set([...includePaths, ...nodeSassIncludePaths.split(path.delimiter)])]; }; const resolveGlobUrl = (url, includePaths = []) => { const filePaths = new Set(); if (glob.hasMagic(url)) { includePaths.forEach((includePath) => { const globPaths = glob.sync(url, { cwd: includePath }); globPaths.forEach((relativePath) => { filePaths.add( path .resolve(includePath, relativePath) // This fixes a problem with importing absolute paths on windows. .split(`\\`) .join(`/`), ); }); }); return [...filePaths]; } return null; }; const ROOT = path.resolve(__dirname, '../../'); const TRANSPARENT_1X1_PNG = 'url()'; const SASS_INCLUDE_PATHS = [ 'app/assets/stylesheets', 'app/assets/stylesheets/_ee', 'app/assets/stylesheets/_jh', 'ee/app/assets/stylesheets', 'ee/app/assets/stylesheets/_ee', 'node_modules', ].map((p) => path.resolve(ROOT, p)); if (IS_JH) { SASS_INCLUDE_PATHS.push( ...['jh/app/assets/stylesheets', 'jh/app/assets/stylesheets/_jh'].map((p) => path.resolve(ROOT, p), ), ); } /** * Custom importer for node-sass, used when LibSass encounters the `@import` directive. * Doc source: https://github.com/sass/node-sass#importer--v200---experimental * @param {*} url the path in import as-is, which LibSass encountered. * @param {*} prev the previously resolved path. * @returns {Object | null} the new import string. */ function sassSmartImporter(url, prev) { const nodeSassOptions = this.options; const includePaths = buildIncludePaths(nodeSassOptions.includePaths, prev).filter( (includePath) => !includePath.includes('node_modules'), ); // GitLab extensively uses glob-style import paths, but // Sass doesn't support glob-style URLs out of the box. // Here, we try and resolve the glob URL. // If it resolves, we update the @import statement with the resolved path. const filePaths = resolveGlobUrl(url, includePaths); if (filePaths) { const contents = filePaths .filter((file) => statSync(file).isFile()) .map((x) => `@import '${x}';`) .join(`\n`); return { contents }; } return null; } const sassLoaderOptions = { functions: { 'image-url($url)': function sassImageUrlStub() { return new sass.types.String(TRANSPARENT_1X1_PNG); }, 'asset_path($url)': function sassAssetPathStub() { return new sass.types.String(TRANSPARENT_1X1_PNG); }, 'asset_url($url)': function sassAssetUrlStub() { return new sass.types.String(TRANSPARENT_1X1_PNG); }, 'url($url)': function sassUrlStub() { return new sass.types.String(TRANSPARENT_1X1_PNG); }, }, includePaths: SASS_INCLUDE_PATHS, importer: sassSmartImporter, }; module.exports = function storybookWebpackConfig({ config }) { // Add any missing extensions from the main GitLab webpack config config.resolve.extensions = Array.from( new Set([...config.resolve.extensions, ...gitlabWebpackConfig.resolve.extensions]), ); config.resolve.alias = { ...config.resolve.alias, gridstack: require.resolve('gridstack/dist/es5/gridstack.js'), '@cubejs-client/core': require.resolve('@cubejs-client/core/dist/cubejs-client-core.js'), }; // Replace any Storybook-defined CSS loaders with our custom one. config.module.rules = [ ...config.module.rules.filter((r) => !r.test.test('.css')), { test: /\.s?css$/, exclude: /typescale\/\w+_demo\.scss$/, // skip typescale demo stylesheets loaders: [ 'style-loader', 'css-loader', { loader: 'sass-loader', options: sassLoaderOptions, }, ], }, { test: /\.(graphql|gql)$/, exclude: /node_modules/, loader: 'graphql-tag/loader', }, { test: /\.(zip)$/, loader: 'file-loader', options: { esModule: false, }, }, ]; // Silence webpack warnings about moment/pikaday not being able to resolve. config.plugins.push(new webpack.IgnorePlugin(/moment/, /pikaday/)); if (!IS_EE) { config.plugins.push( new webpack.NormalModuleReplacementPlugin(/^ee_component\/(.*)\.vue/, (resource) => { resource.request = EMPTY_VUE_COMPONENT_PATH; }), ); } if (!IS_JH) { config.plugins.push( new webpack.NormalModuleReplacementPlugin(/^jh_component\/(.*)\.vue/, (resource) => { resource.request = EMPTY_VUE_COMPONENT_PATH; }), ); } const baseIntegrationTestHelpersPath = 'spec/frontend_integration/test_helpers'; // Add any missing aliases from the main GitLab webpack config Object.assign(config.resolve.alias, gitlabWebpackConfig.resolve.alias, { test_helpers: path.resolve(ROOT, baseIntegrationTestHelpersPath), ee_else_ce_test_helpers: path.resolve(ROOT, IS_EE ? 'ee' : '', baseIntegrationTestHelpersPath), test_fixtures: path.resolve(ROOT, 'tmp/tests/frontend', IS_EE ? 'fixtures-ee' : 'fixtures'), }); // The main GitLab project aliases this `icons.svg` file to app/assets/javascripts/lib/utils/icons_path.js, // which depends on the existence of a global `gon` variable. // By deleting the alias, imports of this path will resolve as expected. delete config.resolve.alias['@gitlab/svgs/dist/icons.svg']; // Fail soft if a story requires a fixture, and the fixture file is absent. // Without this, webpack fails at build phase, with a hard to read error. // This rewrite rule pushes the error to be runtime. config.plugins.push( new webpack.NormalModuleReplacementPlugin(/^test_fixtures/, (resource) => { const filename = resource.request.replace( /^test_fixtures/, config.resolve.alias.test_fixtures, ); if (!existsSync(filename)) { console.error(red(`\nFixture '${filename}' wasn't found.\n`)); resource.request = path.join(ROOT, 'storybook', 'fixture_stub.js'); } }), ); return config; };