forked from egoist/rollup-plugin-postcss
-
Notifications
You must be signed in to change notification settings - Fork 0
/
sass-loader.js
92 lines (83 loc) · 2.93 KB
/
sass-loader.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
import path from 'path'
import pify from 'pify'
import resolve from 'resolve'
import importCwd from 'import-cwd'
import PQueue from 'p-queue'
// This queue makes sure node-sass leaves one thread available for executing fs tasks
// See: https://github.com/sass/node-sass/issues/857
const threadPoolSize = process.env.UV_THREADPOOL_SIZE || 4
const workQueue = new PQueue({ concurrency: threadPoolSize - 1 })
const moduleRe = /^~([a-z0-9]|@).+/i
const getUrlOfPartial = url => {
const parsedUrl = path.parse(url)
return `${parsedUrl.dir}${path.sep}_${parsedUrl.base}`
}
const resolvePromise = pify(resolve)
export default {
name: 'sass',
test: /\.s[ac]ss$/,
process({ code }) {
return new Promise((resolve, reject) => {
const sass = importCwd.silent('node-sass') || importCwd.silent('sass')
if (!sass) {
throw new Error(`You need to install either node-sass or sass in order to process Sass files`)
}
const render = pify(sass.render.bind(sass))
return workQueue.add(() =>
render({
...this.options,
file: this.id,
data: code,
indentedSyntax: /\.sass$/.test(this.id),
sourceMap: this.sourceMap,
importer: [
(url, importer, done) => {
if (!moduleRe.test(url)) return done({ file: url })
const moduleUrl = url.slice(1)
const partialUrl = getUrlOfPartial(moduleUrl)
const options = {
basedir: path.dirname(importer),
extensions: ['.scss', '.sass', '.css']
}
const finishImport = id => {
done({
// Do not add `.css` extension in order to inline the file
file: id.endsWith('.css') ? id.replace(/\.css$/, '') : id
})
}
const next = () => {
// Catch all resolving errors, return the original file and pass responsibility back to other custom importers
done({ file: url })
}
// Give precedence to importing a partial
resolvePromise(partialUrl, options)
.then(finishImport)
.catch(err => {
if (
err.code === 'MODULE_NOT_FOUND' ||
err.code === 'ENOENT'
) {
resolvePromise(moduleUrl, options)
.then(finishImport)
.catch(next)
} else {
next()
}
})
}
].concat(this.options.importer || [])
})
.then(res => {
for (const file of res.stats.includedFiles) {
this.dependencies.add(file)
}
resolve({
code: res.css.toString(),
map: res.map && res.map.toString()
})
})
.catch(reject)
)
})
}
}