-
Notifications
You must be signed in to change notification settings - Fork 1.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(@formatjs/intl-utils): Add interpolateName
fix(@formatjs/ts-transformer): Fix interpolation issue not matching native fix(@formatjs/cli): Fix interpolation issue not matching native
- Loading branch information
Long Ho
committed
Jun 4, 2020
1 parent
d25a809
commit 4d09912
Showing
9 changed files
with
172 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,147 @@ | ||
import * as path from 'path'; | ||
import {createHash, HexBase64Latin1Encoding} from 'crypto'; | ||
export interface LoaderContext { | ||
resourceQuery?: string; | ||
resourcePath?: string; | ||
options?: { | ||
customInterpolateName( | ||
this: LoaderContext, | ||
url: string, | ||
name: string | NameFn, | ||
options: Options | ||
): string; | ||
}; | ||
} | ||
|
||
export interface Options { | ||
context?: string; | ||
content?: string; | ||
regExp?: RegExp; | ||
} | ||
|
||
export type NameFn = (resourcePath?: string, resourceQuery?: string) => string; | ||
|
||
function getHashDigest( | ||
content: string, | ||
hashType: string, | ||
digestType: HexBase64Latin1Encoding, | ||
length: number | ||
) { | ||
const hasher = createHash(hashType); | ||
hasher.update(content); | ||
return hasher.digest(digestType).slice(0, length); | ||
} | ||
|
||
export function interpolateName( | ||
loaderContext: LoaderContext, | ||
name: string | NameFn, | ||
options: Options | ||
) { | ||
let filename; | ||
|
||
const hasQuery = | ||
loaderContext.resourceQuery && loaderContext.resourceQuery.length > 1; | ||
|
||
if (typeof name === 'function') { | ||
filename = name( | ||
loaderContext.resourcePath, | ||
hasQuery ? loaderContext.resourceQuery : undefined | ||
); | ||
} else { | ||
filename = name || '[hash].[ext]'; | ||
} | ||
|
||
const context = options.context; | ||
const content = options.content; | ||
const regExp = options.regExp; | ||
|
||
let ext = 'bin'; | ||
let basename = 'file'; | ||
let directory = ''; | ||
let folder = ''; | ||
let query = ''; | ||
|
||
if (loaderContext.resourcePath) { | ||
const parsed = path.parse(loaderContext.resourcePath); | ||
let resourcePath = loaderContext.resourcePath; | ||
|
||
if (parsed.ext) { | ||
ext = parsed.ext.substr(1); | ||
} | ||
|
||
if (parsed.dir) { | ||
basename = parsed.name; | ||
resourcePath = parsed.dir + path.sep; | ||
} | ||
|
||
if (typeof context !== 'undefined') { | ||
directory = path | ||
.relative(context, resourcePath + '_') | ||
.replace(/\\/g, '/') | ||
.replace(/\.\.(\/)?/g, '_$1'); | ||
directory = directory.substr(0, directory.length - 1); | ||
} else { | ||
directory = resourcePath.replace(/\\/g, '/').replace(/\.\.(\/)?/g, '_$1'); | ||
} | ||
|
||
if (directory.length === 1) { | ||
directory = ''; | ||
} else if (directory.length > 1) { | ||
folder = path.basename(directory); | ||
} | ||
} | ||
|
||
if (loaderContext.resourceQuery && loaderContext.resourceQuery.length > 1) { | ||
query = loaderContext.resourceQuery; | ||
|
||
const hashIdx = query.indexOf('#'); | ||
|
||
if (hashIdx >= 0) { | ||
query = query.substr(0, hashIdx); | ||
} | ||
} | ||
|
||
let url = filename; | ||
|
||
if (content) { | ||
// Match hash template | ||
url = url | ||
// `hash` and `contenthash` are same in `loader-utils` context | ||
// let's keep `hash` for backward compatibility | ||
.replace( | ||
/\[(?:([^:\]]+):)?(?:hash|contenthash)(?::([a-z]+\d*))?(?::(\d+))?\]/gi, | ||
(_, hashType, digestType, maxLength) => | ||
getHashDigest(content, hashType, digestType, parseInt(maxLength, 10)) | ||
); | ||
} | ||
|
||
url = url | ||
.replace(/\[ext\]/gi, () => ext) | ||
.replace(/\[name\]/gi, () => basename) | ||
.replace(/\[path\]/gi, () => directory) | ||
.replace(/\[folder\]/gi, () => folder) | ||
.replace(/\[query\]/gi, () => query); | ||
|
||
if (regExp && loaderContext.resourcePath) { | ||
const match = loaderContext.resourcePath.match(new RegExp(regExp)); | ||
|
||
match && | ||
match.forEach((matched, i) => { | ||
url = url.replace(new RegExp('\\[' + i + '\\]', 'ig'), matched); | ||
}); | ||
} | ||
|
||
if ( | ||
typeof loaderContext.options === 'object' && | ||
typeof loaderContext.options.customInterpolateName === 'function' | ||
) { | ||
url = loaderContext.options.customInterpolateName.call( | ||
loaderContext, | ||
url, | ||
name, | ||
options | ||
); | ||
} | ||
|
||
return url; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
import {interpolateName} from '../src/interpolate-name'; | ||
import {createHash} from 'crypto'; | ||
describe('interpolateName', function () { | ||
it('should match native', function () { | ||
const hasher = createHash('sha1'); | ||
const content = 'foo#bar'; | ||
hasher.update(content); | ||
expect( | ||
interpolateName({}, '[sha1:contenthash:base64:6]', { | ||
content, | ||
}) | ||
).toBe(hasher.digest('base64').slice(0, 6)); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters