|
"use strict"; |
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { |
|
if (k2 === undefined) k2 = k; |
|
var desc = Object.getOwnPropertyDescriptor(m, k); |
|
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { |
|
desc = { enumerable: true, get: function() { return m[k]; } }; |
|
} |
|
Object.defineProperty(o, k2, desc); |
|
}) : (function(o, m, k, k2) { |
|
if (k2 === undefined) k2 = k; |
|
o[k2] = m[k]; |
|
})); |
|
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { |
|
Object.defineProperty(o, "default", { enumerable: true, value: v }); |
|
}) : function(o, v) { |
|
o["default"] = v; |
|
}); |
|
var __importStar = (this && this.__importStar) || function (mod) { |
|
if (mod && mod.__esModule) return mod; |
|
var result = {}; |
|
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); |
|
__setModuleDefault(result, mod); |
|
return result; |
|
}; |
|
var __importDefault = (this && this.__importDefault) || function (mod) { |
|
return (mod && mod.__esModule) ? mod : { "default": mod }; |
|
}; |
|
Object.defineProperty(exports, "__esModule", { value: true }); |
|
exports.parseSync = exports.parse = exports.parseFromFilesSync = exports.parseFromFiles = exports.parseString = exports.parseBuffer = void 0; |
|
const fs = __importStar(require("fs")); |
|
const path = __importStar(require("path")); |
|
const semver = __importStar(require("semver")); |
|
const minimatch_1 = require("minimatch"); |
|
const wasm_1 = require("@one-ini/wasm"); |
|
|
|
|
|
const package_json_1 = __importDefault(require("../package.json")); |
|
const escapedSep = new RegExp(path.sep.replace(/\\/g, '\\\\'), 'g'); |
|
const matchOptions = { matchBase: true, dot: true, noext: true }; |
|
|
|
|
|
const knownProps = { |
|
end_of_line: true, |
|
indent_style: true, |
|
indent_size: true, |
|
insert_final_newline: true, |
|
trim_trailing_whitespace: true, |
|
charset: true, |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function parseBuffer(data) { |
|
const parsed = (0, wasm_1.parse_to_uint32array)(data); |
|
let cur = {}; |
|
const res = [[null, cur]]; |
|
let key = null; |
|
for (let i = 0; i < parsed.length; i += 3) { |
|
switch (parsed[i]) { |
|
case wasm_1.TokenTypes.Section: { |
|
cur = {}; |
|
res.push([ |
|
data.toString('utf8', parsed[i + 1], parsed[i + 2]), |
|
cur, |
|
]); |
|
break; |
|
} |
|
case wasm_1.TokenTypes.Key: |
|
key = data.toString('utf8', parsed[i + 1], parsed[i + 2]); |
|
break; |
|
case wasm_1.TokenTypes.Value: { |
|
cur[key] = data.toString('utf8', parsed[i + 1], parsed[i + 2]); |
|
break; |
|
} |
|
default: |
|
break; |
|
} |
|
} |
|
return res; |
|
} |
|
exports.parseBuffer = parseBuffer; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function parseString(data) { |
|
return parseBuffer(Buffer.from(data)); |
|
} |
|
exports.parseString = parseString; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function getConfigFileNames(filepath, options) { |
|
const paths = []; |
|
do { |
|
filepath = path.dirname(filepath); |
|
paths.push(path.join(filepath, options.config)); |
|
} while (filepath !== options.root); |
|
return paths; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function processMatches(matches, version) { |
|
|
|
|
|
if ('indent_style' in matches |
|
&& matches.indent_style === 'tab' |
|
&& !('indent_size' in matches) |
|
&& semver.gte(version, '0.10.0')) { |
|
matches.indent_size = 'tab'; |
|
} |
|
|
|
|
|
if ('indent_size' in matches |
|
&& !('tab_width' in matches) |
|
&& matches.indent_size !== 'tab') { |
|
matches.tab_width = matches.indent_size; |
|
} |
|
|
|
if ('indent_size' in matches |
|
&& 'tab_width' in matches |
|
&& matches.indent_size === 'tab') { |
|
matches.indent_size = matches.tab_width; |
|
} |
|
return matches; |
|
} |
|
function buildFullGlob(pathPrefix, glob) { |
|
switch (glob.indexOf('/')) { |
|
case -1: |
|
glob = '**/' + glob; |
|
break; |
|
case 0: |
|
glob = glob.substring(1); |
|
break; |
|
default: |
|
break; |
|
} |
|
|
|
|
|
glob = glob.replace(/\\\\/g, '\\\\\\\\'); |
|
|
|
glob = glob.replace(/\*\*/g, '{*,**/**/**}'); |
|
|
|
return new minimatch_1.Minimatch(`${pathPrefix}/${glob}`, matchOptions); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function normalizeProps(options) { |
|
const props = {}; |
|
for (const key in options) { |
|
if (options.hasOwnProperty(key)) { |
|
const value = options[key]; |
|
const key2 = key.toLowerCase(); |
|
let value2 = value; |
|
|
|
if (knownProps[key2]) { |
|
|
|
value2 = String(value).toLowerCase(); |
|
} |
|
try { |
|
value2 = JSON.parse(String(value)); |
|
} |
|
catch (e) { } |
|
if (typeof value2 === 'undefined' || value2 === null) { |
|
|
|
|
|
value2 = String(value); |
|
} |
|
|
|
props[key2] = value2; |
|
} |
|
} |
|
return props; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function processFileContents(filepath, contents, options) { |
|
let res; |
|
if (!contents) { |
|
|
|
res = { |
|
root: false, |
|
notfound: true, |
|
name: filepath, |
|
config: [[null, {}, null]], |
|
}; |
|
} |
|
else { |
|
let pathPrefix = path.dirname(filepath); |
|
if (path.sep !== '/') { |
|
|
|
pathPrefix = pathPrefix.replace(escapedSep, '/'); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
pathPrefix = pathPrefix.replace(/[?*+@!()|[\]{}]/g, '\\$&'); |
|
|
|
|
|
pathPrefix = pathPrefix.replace(/^#/, '\\#'); |
|
const globbed = parseBuffer(contents).map(([name, body]) => [ |
|
name, |
|
normalizeProps(body), |
|
name ? buildFullGlob(pathPrefix, name) : null, |
|
]); |
|
res = { |
|
root: !!globbed[0][1].root, |
|
name: filepath, |
|
config: globbed, |
|
}; |
|
} |
|
if (options.cache) { |
|
options.cache.set(filepath, res); |
|
} |
|
return res; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async function getConfig(filepath, options) { |
|
if (options.cache) { |
|
const cached = options.cache.get(filepath); |
|
if (cached) { |
|
return cached; |
|
} |
|
} |
|
const contents = await new Promise(resolve => { |
|
fs.readFile(filepath, (_, buf) => { |
|
|
|
|
|
resolve(buf); |
|
}); |
|
}); |
|
return processFileContents(filepath, contents, options); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function getConfigSync(filepath, options) { |
|
if (options.cache) { |
|
const cached = options.cache.get(filepath); |
|
if (cached) { |
|
return cached; |
|
} |
|
} |
|
let contents; |
|
try { |
|
contents = fs.readFileSync(filepath); |
|
} |
|
catch (_) { |
|
|
|
|
|
} |
|
return processFileContents(filepath, contents, options); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async function getAllConfigs(files, options) { |
|
const configs = []; |
|
for (const file of files) { |
|
const config = await getConfig(file, options); |
|
if (!config.notfound) { |
|
configs.push(config); |
|
if (config.root) { |
|
break; |
|
} |
|
} |
|
} |
|
return configs; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function getAllConfigsSync(files, options) { |
|
const configs = []; |
|
for (const file of files) { |
|
const config = getConfigSync(file, options); |
|
if (!config.notfound) { |
|
configs.push(config); |
|
if (config.root) { |
|
break; |
|
} |
|
} |
|
} |
|
return configs; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function opts(filepath, options = {}) { |
|
const resolvedFilePath = path.resolve(filepath); |
|
return [ |
|
resolvedFilePath, |
|
{ |
|
config: options.config || '.editorconfig', |
|
version: options.version || package_json_1.default.version, |
|
root: path.resolve(options.root || path.parse(resolvedFilePath).root), |
|
files: options.files, |
|
cache: options.cache, |
|
}, |
|
]; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async function parseFromFiles(filepath, files, options = {}) { |
|
return parseFromFilesSync(filepath, await files, options); |
|
} |
|
exports.parseFromFiles = parseFromFiles; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function parseFromFilesSync(filepath, files, options = {}) { |
|
const [resolvedFilePath, processedOptions] = opts(filepath, options); |
|
const configs = []; |
|
for (const ecf of files) { |
|
let cfg; |
|
if (!options.cache || !(cfg = options.cache.get(ecf.name))) { |
|
cfg = processFileContents(ecf.name, ecf.contents, processedOptions); |
|
} |
|
if (!cfg.notfound) { |
|
configs.push(cfg); |
|
} |
|
if (cfg.root) { |
|
break; |
|
} |
|
} |
|
return combine(resolvedFilePath, configs, processedOptions); |
|
} |
|
exports.parseFromFilesSync = parseFromFilesSync; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function combine(filepath, configs, options) { |
|
const ret = configs.reverse().reduce((props, processed) => { |
|
for (const [name, body, glob] of processed.config) { |
|
if (glob && glob.match(filepath)) { |
|
Object.assign(props, body); |
|
if (options.files) { |
|
options.files.push({ |
|
fileName: processed.name, |
|
glob: name, |
|
}); |
|
} |
|
} |
|
} |
|
return props; |
|
}, {}); |
|
return processMatches(ret, options.version); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async function parse(filepath, options = {}) { |
|
const [resolvedFilePath, processedOptions] = opts(filepath, options); |
|
const filepaths = getConfigFileNames(resolvedFilePath, processedOptions); |
|
const configs = await getAllConfigs(filepaths, processedOptions); |
|
return combine(resolvedFilePath, configs, processedOptions); |
|
} |
|
exports.parse = parse; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function parseSync(filepath, options = {}) { |
|
const [resolvedFilePath, processedOptions] = opts(filepath, options); |
|
const filepaths = getConfigFileNames(resolvedFilePath, processedOptions); |
|
const configs = getAllConfigsSync(filepaths, processedOptions); |
|
return combine(resolvedFilePath, configs, processedOptions); |
|
} |
|
exports.parseSync = parseSync; |
|
|