From fac49db8e4881e7f9c3f5cbd451410f423b875f6 Mon Sep 17 00:00:00 2001 From: Haroen Viaene Date: Tue, 6 Aug 2019 22:18:34 +0200 Subject: [PATCH 1/3] feat(data): add flagging of type=module I've looked around, and it seems like this is a sufficient heuristic to detect esm. I have explicitly left the most common case as unknown, so we can later find out which type those are. References: - https://nodejs.org/api/esm.html#esm_code_package_json_code_code_type_code_field - https://github.com/rollup/rollup/wiki/pkg.module There's also "jsnext:main", but that doesn't seem to be used anymore --- README.md | 1 + .../__snapshots__/config.test.js.snap | 1 + .../__snapshots__/formatPkg.test.js.snap | 5 ++ src/__tests__/formatPkg.test.js | 61 +++++++++++++++++++ src/config.js | 1 + src/formatPkg.js | 17 ++++++ 6 files changed, 86 insertions(+) diff --git a/README.md b/README.md index 75675c38e..c48ea2dfc 100644 --- a/README.md +++ b/README.md @@ -121,6 +121,7 @@ For every single NPM package, we create a record in the Algolia index. The resul ts: 'definitely-typed', // definitely-typed | included | false definitelyTyped: '@types/babel__core', }, + moduleType: 'cjs', // esm | cjs | unknown humanDependents: '3.3k', changelogFilename: null, // if babel-core had a changelog, it would be the raw GitHub url here objectID: 'babel-core', diff --git a/src/__tests__/__snapshots__/config.test.js.snap b/src/__tests__/__snapshots__/config.test.js.snap index b5a978d66..536a806e9 100644 --- a/src/__tests__/__snapshots__/config.test.js.snap +++ b/src/__tests__/__snapshots__/config.test.js.snap @@ -96,6 +96,7 @@ Object { "searchable(owner.name)", "deprecated", "types.ts", + "moduleType", ], "customRanking": Array [ "desc(_searchInternal.downloadsMagnitude)", diff --git a/src/__tests__/__snapshots__/formatPkg.test.js.snap b/src/__tests__/__snapshots__/formatPkg.test.js.snap index 08c2e4d8c..8ae45ce15 100644 --- a/src/__tests__/__snapshots__/formatPkg.test.js.snap +++ b/src/__tests__/__snapshots__/formatPkg.test.js.snap @@ -46,6 +46,7 @@ Object { }, "license": "Apache-2.0", "modified": 1510790341980, + "moduleType": "unknown", "name": "@atlaskit/input", "objectID": "@atlaskit/input", "originalAuthor": undefined, @@ -261,6 +262,7 @@ Object { }, "license": "MIT", "modified": 1498153976616, + "moduleType": "unknown", "name": "@atomic-package/tab", "objectID": "@atomic-package/tab", "originalAuthor": undefined, @@ -383,6 +385,7 @@ Object { }, "license": "MIT", "modified": 1554896045753, + "moduleType": "unknown", "name": "create-instantsearch-app", "objectID": "create-instantsearch-app", "originalAuthor": undefined, @@ -457,6 +460,7 @@ Object { "lastPublisher": null, "license": null, "modified": 1424001480609, + "moduleType": "unknown", "name": "indexof", "objectID": "indexof", "originalAuthor": undefined, @@ -540,6 +544,7 @@ Object { }, "license": null, "modified": NaN, + "moduleType": "unknown", "name": "long-boy", "objectID": "long-boy", "originalAuthor": undefined, diff --git a/src/__tests__/formatPkg.test.js b/src/__tests__/formatPkg.test.js index 0bbcb634d..7a332fb32 100644 --- a/src/__tests__/formatPkg.test.js +++ b/src/__tests__/formatPkg.test.js @@ -441,3 +441,64 @@ describe('alternative names', () => { `); }); }); + +describe('moduleType', () => { + test('type=module', () => { + expect( + formatPkg({ + name: 'irrelevant', + lastPublisher: { name: 'unknown' }, + type: 'module', + }).moduleType + ).toEqual('esm'); + }); + + test('type=commonjs', () => { + expect( + formatPkg({ + name: 'irrelevant', + lastPublisher: { name: 'unknown' }, + type: 'commonjs', + }).moduleType + ).toEqual('cjs'); + }); + + test('module=xxx.js', () => { + expect( + formatPkg({ + name: 'irrelevant', + lastPublisher: { name: 'unknown' }, + module: 'index.js', + }).moduleType + ).toEqual('esm'); + }); + + test('main: index.mjs', () => { + expect( + formatPkg({ + name: 'irrelevant', + lastPublisher: { name: 'unknown' }, + main: 'index.mjs', + }).moduleType + ).toEqual('esm'); + }); + + test('main: index.cjs', () => { + expect( + formatPkg({ + name: 'irrelevant', + lastPublisher: { name: 'unknown' }, + main: 'index.cjs', + }).moduleType + ).toEqual('cjs'); + }); + + test('unknown', () => { + expect( + formatPkg({ + name: 'irrelevant', + lastPublisher: { name: 'unknown' }, + }).moduleType + ).toEqual('unknown'); + }); +}); diff --git a/src/config.js b/src/config.js index 2e3eeb67d..3edd6b6d5 100644 --- a/src/config.js +++ b/src/config.js @@ -35,6 +35,7 @@ const defaultConfig = { 'searchable(owner.name)', 'deprecated', 'types.ts', + 'moduleType', ], customRanking: [ 'desc(_searchInternal.downloadsMagnitude)', diff --git a/src/formatPkg.js b/src/formatPkg.js index a5962cfba..b4355f160 100644 --- a/src/formatPkg.js +++ b/src/formatPkg.js @@ -60,6 +60,7 @@ export default function formatPkg(pkg) { const dependencies = cleaned.dependencies || {}; const devDependencies = cleaned.devDependencies || {}; const alternativeNames = getAlternativeNames(cleaned.name); + const moduleType = getModuleType(cleaned); const tags = pkg['dist-tags']; @@ -94,6 +95,7 @@ export default function formatPkg(pkg) { owners: (cleaned.owners || []).map(formatUser), bin: cleaned.bin, types, + moduleType, lastCrawl: new Date().toISOString(), _searchInternal: { alternativeNames, @@ -431,3 +433,18 @@ function getAlternativeNames(name) { new Set([concatenatedName, splitName, dotJSName, normalName]) ); } + +function getModuleType(cleaned) { + const main = cleaned.main || ''; + if ( + typeof cleaned.module === 'string' || + cleaned.type === 'module' || + main.endsWith('.mjs') + ) { + return 'esm'; + } + if (cleaned.type === 'commonjs' || main.endsWith('.cjs')) { + return 'cjs'; + } + return 'unknown'; +} From 7b693db93048a1987f6ac2425a97f9c6177cd5e8 Mon Sep 17 00:00:00 2001 From: Haroen Viaene Date: Wed, 7 Aug 2019 10:16:05 +0200 Subject: [PATCH 2/3] turn into array --- README.md | 2 +- .../__snapshots__/formatPkg.test.js.snap | 20 +- src/__tests__/formatPkg.test.js | 31 +- src/__tests__/preact-simplified.json | 379 ++++++++++++++++++ src/formatPkg.js | 18 +- 5 files changed, 425 insertions(+), 25 deletions(-) create mode 100644 src/__tests__/preact-simplified.json diff --git a/README.md b/README.md index c48ea2dfc..00a0f278d 100644 --- a/README.md +++ b/README.md @@ -121,7 +121,7 @@ For every single NPM package, we create a record in the Algolia index. The resul ts: 'definitely-typed', // definitely-typed | included | false definitelyTyped: '@types/babel__core', }, - moduleType: 'cjs', // esm | cjs | unknown + moduleTypes: ['unknown'], // esm | cjs | unknown humanDependents: '3.3k', changelogFilename: null, // if babel-core had a changelog, it would be the raw GitHub url here objectID: 'babel-core', diff --git a/src/__tests__/__snapshots__/formatPkg.test.js.snap b/src/__tests__/__snapshots__/formatPkg.test.js.snap index 8ae45ce15..419a2e3b3 100644 --- a/src/__tests__/__snapshots__/formatPkg.test.js.snap +++ b/src/__tests__/__snapshots__/formatPkg.test.js.snap @@ -46,7 +46,9 @@ Object { }, "license": "Apache-2.0", "modified": 1510790341980, - "moduleType": "unknown", + "moduleTypes": Array [ + "unknown", + ], "name": "@atlaskit/input", "objectID": "@atlaskit/input", "originalAuthor": undefined, @@ -262,7 +264,9 @@ Object { }, "license": "MIT", "modified": 1498153976616, - "moduleType": "unknown", + "moduleTypes": Array [ + "unknown", + ], "name": "@atomic-package/tab", "objectID": "@atomic-package/tab", "originalAuthor": undefined, @@ -385,7 +389,9 @@ Object { }, "license": "MIT", "modified": 1554896045753, - "moduleType": "unknown", + "moduleTypes": Array [ + "unknown", + ], "name": "create-instantsearch-app", "objectID": "create-instantsearch-app", "originalAuthor": undefined, @@ -460,7 +466,9 @@ Object { "lastPublisher": null, "license": null, "modified": 1424001480609, - "moduleType": "unknown", + "moduleTypes": Array [ + "unknown", + ], "name": "indexof", "objectID": "indexof", "originalAuthor": undefined, @@ -544,7 +552,9 @@ Object { }, "license": null, "modified": NaN, - "moduleType": "unknown", + "moduleTypes": Array [ + "unknown", + ], "name": "long-boy", "objectID": "long-boy", "originalAuthor": undefined, diff --git a/src/__tests__/formatPkg.test.js b/src/__tests__/formatPkg.test.js index 7a332fb32..3fa1dacdd 100644 --- a/src/__tests__/formatPkg.test.js +++ b/src/__tests__/formatPkg.test.js @@ -1,5 +1,6 @@ import formatPkg from '../formatPkg.js'; import rawPackages from './rawPackages.json'; +import preact from './preact-simplified.json'; import isISO8601 from 'validator/lib/isISO8601.js'; it('transforms correctly', () => { @@ -442,15 +443,15 @@ describe('alternative names', () => { }); }); -describe('moduleType', () => { +describe('moduleTypes', () => { test('type=module', () => { expect( formatPkg({ name: 'irrelevant', lastPublisher: { name: 'unknown' }, type: 'module', - }).moduleType - ).toEqual('esm'); + }).moduleTypes + ).toEqual(['esm']); }); test('type=commonjs', () => { @@ -459,8 +460,8 @@ describe('moduleType', () => { name: 'irrelevant', lastPublisher: { name: 'unknown' }, type: 'commonjs', - }).moduleType - ).toEqual('cjs'); + }).moduleTypes + ).toEqual(['cjs']); }); test('module=xxx.js', () => { @@ -469,8 +470,8 @@ describe('moduleType', () => { name: 'irrelevant', lastPublisher: { name: 'unknown' }, module: 'index.js', - }).moduleType - ).toEqual('esm'); + }).moduleTypes + ).toEqual(['esm']); }); test('main: index.mjs', () => { @@ -479,8 +480,8 @@ describe('moduleType', () => { name: 'irrelevant', lastPublisher: { name: 'unknown' }, main: 'index.mjs', - }).moduleType - ).toEqual('esm'); + }).moduleTypes + ).toEqual(['esm']); }); test('main: index.cjs', () => { @@ -489,8 +490,8 @@ describe('moduleType', () => { name: 'irrelevant', lastPublisher: { name: 'unknown' }, main: 'index.cjs', - }).moduleType - ).toEqual('cjs'); + }).moduleTypes + ).toEqual(['cjs']); }); test('unknown', () => { @@ -498,7 +499,11 @@ describe('moduleType', () => { formatPkg({ name: 'irrelevant', lastPublisher: { name: 'unknown' }, - }).moduleType - ).toEqual('unknown'); + }).moduleTypes + ).toEqual(['unknown']); + }); + + test('preact (esm & umd)', () => { + expect(formatPkg(preact).moduleTypes).toEqual(['esm']); }); }); diff --git a/src/__tests__/preact-simplified.json b/src/__tests__/preact-simplified.json new file mode 100644 index 000000000..d691e3dc0 --- /dev/null +++ b/src/__tests__/preact-simplified.json @@ -0,0 +1,379 @@ +{ + "_id": "preact", + "_rev": "202-877707b490a7d5d39e9f83d4b528254c", + "name": "preact", + "description": "Fast 3kb React alternative with the same modern API. Components & Virtual DOM.", + "dist-tags": { "latest": "8.5.0", "next": "10.0.0-rc.1" }, + "versions": { + "8.5.0": { + "name": "preact", + "version": "8.5.0", + "description": "Fast 3kb React alternative with the same modern API. Components & Virtual DOM.", + "main": "dist/preact.js", + "jsnext:main": "dist/preact.mjs", + "module": "dist/preact.mjs", + "dev:main": "dist/preact.dev.js", + "minified:main": "dist/preact.min.js", + "unpkg": "dist/preact.min.js", + "types": "dist/preact.d.ts", + "browser": "dist/preact.umd.js", + "scripts": { + "clean": "rimraf dist/ devtools.js devtools.js.map debug.js debug.js.map test/ts/**/*.js", + "copy-flow-definition": "copyfiles -f src/preact.js.flow dist", + "copy-typescript-definition": "copyfiles -f src/preact.d.ts dist", + "build": "npm-run-all --silent clean transpile copy-flow-definition copy-typescript-definition strip optimize minify size", + "flow": "flow", + "transpile:main": "rollup -c config/rollup.config.js", + "transpile:devtools": "rollup -c config/rollup.config.devtools.js", + "transpile:esm": "rollup -c config/rollup.config.module.js", + "transpile:umd": "rollup -c config/rollup.config.umd.js", + "transpile:debug": "babel debug/ -o debug.js -s", + "transpile": "npm-run-all transpile:main transpile:esm transpile:umd transpile:devtools transpile:debug", + "optimize": "uglifyjs dist/preact.dev.js -c conditionals=false,sequences=false,loops=false,join_vars=false,collapse_vars=false --pure-funcs=Object.defineProperty --mangle-props --mangle-regex=\"/^(_|normalizedNodeName|nextBase|prev[CPS]|_parentC)/\" --name-cache config/properties.json -b width=120,quote_style=3 -o dist/preact.js -p relative --in-source-map dist/preact.dev.js.map --source-map dist/preact.js.map", + "minify": "uglifyjs dist/preact.js -c collapse_vars,evaluate,screw_ie8,unsafe,loops=false,keep_fargs=false,pure_getters,unused,dead_code -m -o dist/preact.min.js -p relative --in-source-map dist/preact.js.map --source-map dist/preact.min.js.map", + "prepare": "npm run build", + "strip:main": "jscodeshift --run-in-band -s -t config/codemod-strip-tdz.js dist/preact.dev.js && jscodeshift --run-in-band -s -t config/codemod-const.js dist/preact.dev.js && jscodeshift --run-in-band -s -t config/codemod-let-name.js dist/preact.dev.js", + "strip:esm": "jscodeshift --run-in-band -s -t config/codemod-strip-tdz.js dist/preact.mjs && jscodeshift --run-in-band -s -t config/codemod-const.js dist/preact.mjs && jscodeshift --run-in-band -s -t config/codemod-let-name.js dist/preact.mjs", + "strip": "npm-run-all strip:main strip:esm", + "size": "node -e \"process.stdout.write('gzip size: ')\" && gzip-size --raw dist/preact.min.js", + "test": "npm-run-all lint --parallel test:mocha test:karma test:ts test:flow test:size", + "test:flow": "flow check", + "test:ts": "tsc -p test/ts/ && mocha --require babel-register test/ts/**/*-test.js", + "test:mocha": "mocha --recursive --require babel-register test/shared test/node", + "test:karma": "karma start test/karma.conf.js --single-run", + "test:mocha:watch": "npm run test:mocha -- --watch", + "test:karma:watch": "npm run test:karma -- no-single-run", + "test:size": "bundlesize", + "lint": "eslint debug devtools src test", + "prepublishOnly": "npm run build", + "smart-release": "npm run build && npm test && git commit -am $npm_package_version && git tag $npm_package_version && git push && git push --tags && npm publish", + "release": "cross-env npm run smart-release", + "postinstall": "node -e \"console.log('\\u001b[35m\\u001b[1mLove Preact? You can now donate to our open collective:\\u001b[22m\\u001b[39m\\n > \\u001b[34mhttps://opencollective.com/preact/donate\\u001b[0m')\"" + }, + "eslintConfig": { "extends": "./config/eslint-config.js" }, + "typings": "./dist/preact.d.ts", + "repository": { + "type": "git", + "url": "https://github.com/developit/preact.git" + }, + "keywords": [ + "preact", + "react", + "virtual dom", + "vdom", + "components", + "virtual", + "dom" + ], + "author": { "name": "Jason Miller", "email": "jason@developit.ca" }, + "license": "MIT", + "bugs": { "url": "https://github.com/developit/preact/issues" }, + "homepage": "https://github.com/developit/preact", + "devDependencies": { + "@types/chai": "^4.1.7", + "@types/mocha": "^5.2.5", + "@types/node": "^9.6.40", + "babel-cli": "^6.24.1", + "babel-core": "^6.24.1", + "babel-eslint": "^8.2.6", + "babel-loader": "^7.0.0", + "babel-plugin-transform-object-rest-spread": "^6.23.0", + "babel-plugin-transform-react-jsx": "^6.24.1", + "babel-preset-env": "^1.6.1", + "bundlesize": "^0.17.0", + "chai": "^4.2.0", + "copyfiles": "^2.1.0", + "core-js": "^2.6.0", + "coveralls": "^3.0.0", + "cross-env": "^5.1.4", + "diff": "^3.0.0", + "eslint": "^4.18.2", + "eslint-plugin-react": "^7.11.1", + "flow-bin": "^0.89.0", + "gzip-size-cli": "^2.0.0", + "istanbul-instrumenter-loader": "^3.0.0", + "jscodeshift": "^0.5.0", + "karma": "^3.1.3", + "karma-babel-preprocessor": "^7.0.0", + "karma-chai-sinon": "^0.1.5", + "karma-chrome-launcher": "^2.2.0", + "karma-coverage": "^1.1.2", + "karma-mocha": "^1.3.0", + "karma-mocha-reporter": "^2.2.5", + "karma-sauce-launcher": "^1.2.0", + "karma-sinon": "^1.0.5", + "karma-source-map-support": "^1.3.0", + "karma-sourcemap-loader": "^0.3.6", + "karma-webpack": "^3.0.5", + "mocha": "^5.0.4", + "npm-run-all": "^4.1.5", + "puppeteer": "^1.11.0", + "rimraf": "^2.5.3", + "rollup": "^0.57.1", + "rollup-plugin-babel": "^3.0.2", + "rollup-plugin-memory": "^3.0.0", + "rollup-plugin-node-resolve": "^3.4.0", + "sinon": "^4.4.2", + "sinon-chai": "^3.3.0", + "typescript": "^3.0.1", + "uglify-js": "^2.7.5", + "webpack": "^4.27.1" + }, + "greenkeeper": { + "ignore": [ + "babel-cli", + "babel-core", + "babel-eslint", + "babel-loader", + "jscodeshift", + "rollup-plugin-babel" + ] + }, + "bundlesize": [{ "path": "./dist/preact.min.js", "threshold": "4Kb" }], + "licenseText": "The MIT License (MIT)\n\nCopyright (c) 2015-present Jason Miller\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n", + "_id": "preact@8.5.0", + "dist": { + "shasum": "05690de3af035cd8ad393e8b4057b8ab29aedee1", + "integrity": "sha512-S2OOz+lRjfbqDbV5LOIcJ6yfYvshDtKvvsMwaFW84wuw4HtFUYH67T+hnWhdCDYtW8BfmyIg9utz16s6U80HbA==", + "tarball": "https://registry.npmjs.org/preact/-/preact-8.5.0.tgz", + "fileCount": 56, + "unpackedSize": 978252, + "npm-signature": "-----BEGIN PGP SIGNATURE-----\r\nVersion: OpenPGP.js v3.0.4\r\nComment: https://openpgpjs.org\r\n\r\nwsFcBAEBCAAQBQJdRIIwCRA9TVsSAnZWagAAFiUP/j+wGH4kdXgyELOhS5qd\nHP2hzGFDVm6oViJKM4AboDQJtoqirCQHpfjB+QkeiPn97HHEGfsDxbxzdMMw\nGiaQKum0A6crCuEPxohR7bsV3xN6ANLnFWCBsQ80aLZ50lVUTkRQRMAjgEcT\n5zl1CaS9GSSmE1B5fMVTWWFSVU/mbGPcAkXZMjwdqci88RHf3mtgn8IJ/5oC\n4xiZSobSvIBUlv9cBqIVaflvxHl9xZpdqzV1iddF4C7Ou71edxVjWBwXdEoa\nL8tNmJmtJPEUR4h12ZAjQFDp1wiM/iBbT/5yXGE8rqlDxtDf1WLtDI2sU0SK\nYykUj3yCNwCT9Li7A6O6I0cTQu8CKIQ8B7JtrBjMR/PzLQFSsxXBYVfc2IB9\n5VcuUpmGvqcnn4l36dPzBQEZlkRQSZ92N0L2WZtkwuN6uTqAOP85XSQ+FjyD\nk1kuNrwXimBKuL4g7LiClTYMRylwcA8WR8T93Lc5DbXMQRcUlAqhQyE/2UBf\niTI2jBeJmjGyUJgyhmt56qaWk2Y+5ltXaq4evb5/JpAgrO0vMEsGbQ0B/RYo\nCt/OHE/LAHhZ+/OCTuDsJS1PsWoT81tXqizE9/wQ5KJTeew7b7tZOn6Au5S/\nyv0eQmO6mqk0voLz60hLauHhFFPP4MkNFGhCr4Dt7QT+G2QRuQmwNXOeLuJg\nQE+S\r\n=36vW\r\n-----END PGP SIGNATURE-----\r\n" + }, + "maintainers": [ + { "email": "jason@developit.ca", "name": "developit" }, + { "email": "ulliftw@gmail.com", "name": "harmony" }, + { "email": "luke@lukeed.com", "name": "lukeed" }, + { "email": "marvin@marvinhagemeister.de", "name": "marvinhagemeister" }, + { "email": "prateek89born@gmail.com", "name": "prateekbh" }, + { "email": "hello@preactjs.com", "name": "preactjs" }, + { "email": "allamsetty.anup@gmail.com", "name": "reznord" } + ], + "_npmUser": { "name": "harmony", "email": "npm.leah@hrmny.sh" }, + "directories": {}, + "_npmOperationalInternal": { + "host": "s3://npm-registry-packages", + "tmp": "tmp/preact_8.5.0_1564770863260_0.5179813670573445" + }, + "_hasShrinkwrap": false + }, + "10.0.0-rc.1": { + "name": "preact", + "amdName": "preact", + "version": "10.0.0-rc.1", + "private": false, + "description": "Fast 3kb React-compatible Virtual DOM library.", + "main": "dist/preact.js", + "module": "dist/preact.module.js", + "umd:main": "dist/preact.umd.js", + "source": "src/index.js", + "license": "MIT", + "types": "src/index.d.ts", + "scripts": { + "build": "npm-run-all --parallel build:*", + "build:core": "microbundle build --raw", + "build:debug": "microbundle build --raw --cwd debug", + "build:hooks": "microbundle build --raw --cwd hooks", + "build:test-utils": "microbundle build --raw --cwd test-utils", + "build:compat": "microbundle build --raw --cwd compat --globals 'preact/hooks=preactHooks'", + "dev": "microbundle watch --raw --format cjs", + "dev:hooks": "microbundle watch --raw --format cjs --cwd hooks", + "dev:compat": "microbundle watch --raw --format cjs --cwd compat --globals 'preact/hooks=preactHooks'", + "test": "npm-run-all lint build --parallel test:mocha test:karma test:ts", + "test:flow": "flow check", + "test:ts": "tsc -p test/ts/ && mocha --require babel-register test/ts/**/*-test.js", + "test:mocha": "mocha --recursive --require babel-register test/shared test/node", + "test:karma": "cross-env COVERAGE=true karma start karma.conf.js --single-run", + "test:mocha:watch": "npm run test:mocha -- --watch", + "test:karma:watch": "karma start karma.conf.js --no-single-run", + "test:karma:hooks": "cross-env COVERAGE=false karma start karma.conf.js --grep=hooks/test/browser/**.js --no-single-run", + "test:karma:test-utils": "cross-env PERFORMANCE=false COVERAGE=false karma start karma.conf.js --grep=test-utils/test/shared/**.js --no-single-run", + "test:karma:bench": "cross-env PERFORMANCE=true COVERAGE=false karma start karma.conf.js --grep=test/benchmarks/**.js --single-run", + "benchmark": "npm run test:karma:bench -- no-single-run", + "lint": "eslint src test debug compat hooks test-utils", + "postinstall": "node -e \"console.log('\\u001b[35m\\u001b[1mLove Preact? You can now donate to our open collective:\\u001b[22m\\u001b[39m\\n > \\u001b[34mhttps://opencollective.com/preact/donate\\u001b[0m')\"" + }, + "eslintConfig": { + "extends": "developit", + "settings": { "react": { "pragma": "createElement" } }, + "rules": { + "camelcase": [ + 1, + { "allow": ["__test__*", "unstable_*", "UNSAFE_*"] } + ], + "prefer-rest-params": 0, + "prefer-spread": 0, + "no-cond-assign": 0, + "react/jsx-no-bind": 0, + "react/prefer-stateless-function": 0, + "react/sort-comp": 0, + "jest/valid-expect": 0, + "jest/no-disabled-tests": 0, + "react/no-find-dom-node": 0 + } + }, + "eslintIgnore": ["test/fixtures", "test/ts/", "*.ts", "dist"], + "keywords": [ + "preact", + "react", + "virtual dom", + "vdom", + "components", + "virtual", + "dom" + ], + "authors": ["Jason Miller "], + "repository": { + "type": "git", + "url": "git+https://github.com/preactjs/preact.git" + }, + "bugs": { "url": "https://github.com/preactjs/preact/issues" }, + "homepage": "https://github.com/preactjs/preact", + "devDependencies": { + "@types/chai": "^4.1.2", + "@types/mocha": "^5.0.0", + "@types/node": "^10.5.2", + "babel-cli": "6.26.0", + "babel-core": "6.26.3", + "babel-loader": "7.1.5", + "babel-plugin-istanbul": "5.0.1", + "babel-plugin-transform-object-rest-spread": "^6.23.0", + "babel-plugin-transform-react-jsx": "^6.24.1", + "babel-preset-env": "^1.6.1", + "benchmark": "^2.1.4", + "chai": "^4.1.2", + "coveralls": "^3.0.0", + "cross-env": "^5.2.0", + "diff": "^3.5.0", + "eslint": "5.15.1", + "eslint-config-developit": "^1.1.1", + "eslint-plugin-react": "7.12.4", + "flow-bin": "^0.79.1", + "karma": "^3.0.0", + "karma-babel-preprocessor": "^7.0.0", + "karma-chai-sinon": "^0.1.5", + "karma-chrome-launcher": "^2.2.0", + "karma-coverage": "^1.1.2", + "karma-mocha": "^1.3.0", + "karma-mocha-reporter": "^2.2.5", + "karma-sauce-launcher": "^1.2.0", + "karma-sinon": "^1.0.5", + "karma-source-map-support": "^1.3.0", + "karma-sourcemap-loader": "^0.3.7", + "karma-webpack": "^3.0.5", + "lodash": "^4.17.10", + "microbundle": "^0.11.0", + "mocha": "^5.2.0", + "npm-run-all": "^4.0.0", + "prop-types": "^15.7.2", + "sinon": "^6.1.3", + "sinon-chai": "^3.0.0", + "travis-size-report": "^1.0.1", + "typescript": "^3.0.1", + "webpack": "^4.3.0" + }, + "gitHead": "e7c39e8528e5f407f5accb91efd49aeefb9776d7", + "readme": "

\n\n\t\n![Preact](https://raw.githubusercontent.com/preactjs/preact/8b0bcc927995c188eca83cba30fbc83491cc0b2f/logo.svg?sanitize=true \"Preact\")\n\n\n

\n

Fast 3kB alternative to React with the same modern API.

\n\n**All the power of Virtual DOM components, without the overhead:**\n\n- Familiar React API & patterns: [ES6 Class] and [Functional Components]\n- Extensive React compatibility via a simple [preact-compat] alias\n- Everything you need: JSX, VDOM, React DevTools, HMR, SSR..\n- A highly optimized diff algorithm and seamless Server Side Rendering\n- Transparent asynchronous rendering with a pluggable scheduler\n- 🆕💥 **Instant no-config app bundling with [Preact CLI](https://github.com/preactjs/preact-cli)**\n\n### 💁 More information at the [Preact Website ➞](https://preactjs.com)\n\n\n---\n\n\n\n- [Demos](#demos)\n- [Libraries & Add-ons](#libraries--add-ons)\n- [Getting Started](#getting-started)\n\t- [Import what you need](#import-what-you-need)\n\t- [Rendering JSX](#rendering-jsx)\n\t- [Components](#components)\n\t- [Props & State](#props--state)\n- [Linked State](#linked-state)\n- [Examples](#examples)\n- [Extensions](#extensions)\n- [Debug Mode](#debug-mode)\n- [Backers](#backers)\n- [Sponsors](#sponsors)\n- [License](#license)\n\n\n\n\n# Preact\n\n[![npm](https://img.shields.io/npm/v/preact.svg)](http://npm.im/preact)\n[![CDNJS](https://img.shields.io/cdnjs/v/preact.svg)](https://cdnjs.com/libraries/preact)\n[![Preact Slack Community](https://preact-slack.now.sh/badge.svg)](https://preact-slack.now.sh)\n[![OpenCollective Backers](https://opencollective.com/preact/backers/badge.svg)](#backers)\n[![OpenCollective Sponsors](https://opencollective.com/preact/sponsors/badge.svg)](#sponsors)\n[![travis](https://travis-ci.org/preactjs/preact.svg?branch=master)](https://travis-ci.org/preactjs/preact)\n[![coveralls](https://img.shields.io/coveralls/preactjs/preact/master.svg)](https://coveralls.io/github/preactjs/preact)\n[![gzip size](http://img.badgesize.io/https://unpkg.com/preact/dist/preact.min.js?compression=gzip)](https://unpkg.com/preact/dist/preact.min.js)\n[![install size](https://packagephobia.now.sh/badge?p=preact)](https://packagephobia.now.sh/result?p=preact)\n\nPreact supports modern browsers and IE9+:\n\n[![Browsers](https://saucelabs.com/browser-matrix/preact.svg)](https://saucelabs.com/u/preact)\n\n\n---\n\n\n## Demos\n\n#### Real-World Apps\n\n- [**Preact Hacker News**](https://hn.kristoferbaxter.com) _([GitHub Project](https://github.com/kristoferbaxter/preact-hn))_\n- [**Play.cash**](https://play.cash) :notes: _([GitHub Project](https://github.com/feross/play.cash))_\n- [**BitMidi**](https://bitmidi.com/) 🎹 Wayback machine for free MIDI files _([GitHub Project](https://github.com/feross/bitmidi.com))_\n- [**Ultimate Guitar**](https://www.ultimate-guitar.com) 🎸speed boosted by Preact.\n- [**ESBench**](http://esbench.com) is built using Preact.\n- [**BigWebQuiz**](https://bigwebquiz.com) _([GitHub Project](https://github.com/jakearchibald/big-web-quiz))_\n- [**Nectarine.rocks**](http://nectarine.rocks) _([GitHub Project](https://github.com/developit/nectarine))_ :peach:\n- [**TodoMVC**](https://preact-todomvc.surge.sh) _([GitHub Project](https://github.com/developit/preact-todomvc))_\n- [**OSS.Ninja**](https://oss.ninja) _([GitHub Project](https://github.com/developit/oss.ninja))_\n- [**GuriVR**](https://gurivr.com) _([GitHub Project](https://github.com/opennewslabs/guri-vr))_\n- [**Color Picker**](https://colors.now.sh) _([GitHub Project](https://github.com/lukeed/colors-app))_ :art:\n- [**Offline Gallery**](https://use-the-platform.com/offline-gallery/) _([GitHub Project](https://github.com/vaneenige/offline-gallery/))_ :balloon:\n- [**Periodic Weather**](https://use-the-platform.com/periodic-weather/) _([GitHub Project](https://github.com/vaneenige/periodic-weather/))_ :sunny:\n- [**Rugby News Board**](http://nbrugby.com) _[(GitHub Project)](https://github.com/rugby-board/rugby-board-node)_\n- [**Preact Gallery**](https://preact.gallery/) an 8KB photo gallery PWA built using Preact.\n- [**Rainbow Explorer**](https://use-the-platform.com/rainbow-explorer/) Preact app to translate real life color to digital color _([Github project](https://github.com/vaneenige/rainbow-explorer))_.\n- [**YASCC**](https://carlosqsilva.github.io/YASCC/#/) Yet Another SoundCloud Client _([Github project](https://github.com/carlosqsilva/YASCC))_.\n- [**Journalize**](https://preact-journal.herokuapp.com/) 14k offline-capable journaling PWA using preact. _([Github project](https://github.com/jpodwys/preact-journal))_.\n- [**Proxx**](https://proxx.app) A game of proximity by GoogleChromeLabs using preact. _([Github project](https://github.com/GoogleChromeLabs/proxx))_.\n- [**Web Maker**](https://webmaker.app) An offline and blazing fast frontend playground built using Preact. _([Github project](https://github.com/chinchang/web-maker))_.\n\n\n#### Runnable Examples\n\n- [**Flickr Browser**](http://codepen.io/developit/full/VvMZwK/) (@ CodePen)\n- [**Animating Text**](http://codepen.io/developit/full/LpNOdm/) (@ CodePen)\n- [**60FPS Rainbow Spiral**](http://codepen.io/developit/full/xGoagz/) (@ CodePen)\n- [**Simple Clock**](http://jsfiddle.net/developit/u9m5x0L7/embedded/result,js/) (@ JSFiddle)\n- [**3D + ThreeJS**](http://codepen.io/developit/pen/PPMNjd?editors=0010) (@ CodePen)\n- [**Stock Ticker**](http://codepen.io/developit/pen/wMYoBb?editors=0010) (@ CodePen)\n- [*Create your Own!*](https://jsfiddle.net/developit/rs6zrh5f/embedded/result/) (@ JSFiddle)\n\n### Starter Projects\n\n- [**Preact Boilerplate**](https://preact-boilerplate.surge.sh) _([GitHub Project](https://github.com/developit/preact-boilerplate))_ :zap:\n- [**Preact Offline Starter**](https://preact-starter.now.sh) _([GitHub Project](https://github.com/lukeed/preact-starter))_ :100:\n- [**Preact PWA**](https://preact-pwa-yfxiijbzit.now.sh/) _([GitHub Project](https://github.com/ezekielchentnik/preact-pwa))_ :hamburger:\n- [**Parcel + Preact + Unistore Starter**](https://github.com/hwclass/parcel-preact-unistore-starter)\n- [**Preact Mobx Starter**](https://awaw00.github.io/preact-mobx-starter/) _([GitHub Project](https://github.com/awaw00/preact-mobx-starter))_ :sunny:\n- [**Preact Redux Example**](https://github.com/developit/preact-redux-example) :star:\n- [**Preact Redux/RxJS/Reselect Example**](https://github.com/continuata/preact-seed)\n- [**V2EX Preact**](https://github.com/yanni4night/v2ex-preact)\n- [**Preact Coffeescript**](https://github.com/crisward/preact-coffee)\n- [**Preact + TypeScript + Webpack**](https://github.com/k1r0s/bleeding-preact-starter)\n- [**0 config => Preact + Poi**](https://github.com/k1r0s/preact-poi-starter)\n- [**Zero configuration => Preact + Typescript + Parcel**](https://github.com/aalises/preact-typescript-parcel-starter)\n\n---\n\n## Libraries & Add-ons\n\n- :raised_hands: [**preact-compat**](https://git.io/preact-compat): use any React library with Preact *([full example](http://git.io/preact-compat-example))*\n- :twisted_rightwards_arrows: [**preact-context**](https://github.com/valotas/preact-context): React's `createContext` api for Preact\n- :page_facing_up: [**preact-render-to-string**](https://git.io/preact-render-to-string): Universal rendering.\n- :eyes: [**preact-render-spy**](https://github.com/mzgoddard/preact-render-spy): Enzyme-lite: Renderer with access to the produced virtual dom for testing.\n- :loop: [**preact-render-to-json**](https://git.io/preact-render-to-json): Render for Jest Snapshot testing.\n- :earth_americas: [**preact-router**](https://git.io/preact-router): URL routing for your components\n- :bookmark_tabs: [**preact-markup**](https://git.io/preact-markup): Render HTML & Custom Elements as JSX & Components\n- :satellite: [**preact-portal**](https://git.io/preact-portal): Render Preact components into (a) SPACE :milky_way:\n- :pencil: [**preact-richtextarea**](https://git.io/preact-richtextarea): Simple HTML editor component\n- :bookmark: [**preact-token-input**](https://github.com/developit/preact-token-input): Text field that tokenizes input, for things like tags\n- :card_index: [**preact-virtual-list**](https://github.com/developit/preact-virtual-list): Easily render lists with millions of rows ([demo](https://jsfiddle.net/developit/qqan9pdo/))\n- :repeat: [**preact-cycle**](https://git.io/preact-cycle): Functional-reactive paradigm for Preact\n- :triangular_ruler: [**preact-layout**](https://download.github.io/preact-layout/): Small and simple layout library\n- :thought_balloon: [**preact-socrates**](https://github.com/matthewmueller/preact-socrates): Preact plugin for [Socrates](http://github.com/matthewmueller/socrates)\n- :rowboat: [**preact-flyd**](https://github.com/xialvjun/preact-flyd): Use [flyd](https://github.com/paldepind/flyd) FRP streams in Preact + JSX\n- :speech_balloon: [**preact-i18nline**](https://github.com/download/preact-i18nline): Integrates the ecosystem around [i18n-js](https://github.com/everydayhero/i18n-js) with Preact via [i18nline](https://github.com/download/i18nline).\n- :microscope: [**preact-jsx-chai**](https://git.io/preact-jsx-chai): JSX assertion testing _(no DOM, right in Node)_\n- :tophat: [**preact-classless-component**](https://github.com/ld0rman/preact-classless-component): create preact components without the class keyword\n- :hammer: [**preact-hyperscript**](https://github.com/queckezz/preact-hyperscript): Hyperscript-like syntax for creating elements\n- :white_check_mark: [**shallow-compare**](https://github.com/tkh44/shallow-compare): simplified `shouldComponentUpdate` helper.\n- :shaved_ice: [**preact-codemod**](https://github.com/vutran/preact-codemod): Transform your React code to Preact.\n- :construction_worker: [**preact-helmet**](https://github.com/download/preact-helmet): A document head manager for Preact\n- :necktie: [**preact-delegate**](https://github.com/NekR/preact-delegate): Delegate DOM events\n- :art: [**preact-stylesheet-decorator**](https://github.com/k1r0s/preact-stylesheet-decorator): Add Scoped Stylesheets to your Preact Components\n- :electric_plug: [**preact-routlet**](https://github.com/k1r0s/preact-routlet): Simple `Component Driven` Routing for Preact using ES7 Decorators\n- :fax: [**preact-bind-group**](https://github.com/k1r0s/preact-bind-group): Preact Forms made easy, Group Events into a Single Callback\n- :hatching_chick: [**preact-habitat**](https://github.com/zouhir/preact-habitat): Declarative Preact widgets renderer in any CMS or DOM host ([demo](https://codepen.io/zouhir/pen/brrOPB)).\n- :tada: [**proppy-preact**](https://github.com/fahad19/proppy): Functional props composition for Preact components\n\n#### UI Component Libraries\n\n> Want to prototype something or speed up your development? Try one of these toolkits:\n\n- [**preact-material-components**](https://github.com/prateekbh/preact-material-components): Material Design Components for Preact ([website](https://material.preactjs.com/))\n- [**preact-mdc**](https://github.com/BerndWessels/preact-mdc): Material Design Components for Preact ([demo](https://github.com/BerndWessels/preact-mdc-demo))\n- [**preact-mui**](https://git.io/v1aVO): The MUI CSS Preact library.\n- [**preact-photon**](https://git.io/preact-photon): build beautiful desktop UI with [photon](http://photonkit.com)\n- [**preact-mdl**](https://git.io/preact-mdl): [Material Design Lite](https://getmdl.io) for Preact\n- [**preact-weui**](https://github.com/afeiship/preact-weui): [Weui](https://github.com/afeiship/preact-weui) for Preact\n- [**preact-charts**](https://github.com/pmkroeker/preact-charts): Charts for Preact\n\n\n---\n\n## Getting Started\n\n> 💁 _**Note:** You [don't need ES2015 to use Preact](https://github.com/developit/preact-in-es3)... but give it a try!_\n\nThe easiest way to get started with Preact is to install [Preact CLI](https://github.com/preactjs/preact-cli). This simple command-line tool wraps up the best possible Webpack and Babel setup for you, and even keeps you up-to-date as the underlying tools change. Best of all, it's easy to understand! It builds your app in a single command (`preact build`), doesn't need any configuration, and bakes in best-practises 🙌.\n\nThe following guide assumes you have some sort of ES2015 build set up using babel and/or webpack/browserify/gulp/grunt/etc.\n\nYou can also start with [preact-boilerplate] or a [CodePen Template](http://codepen.io/developit/pen/pgaROe?editors=0010).\n\n\n### Import what you need\n\nThe `preact` module provides both named and default exports, so you can either import everything under a namespace of your choosing, or just what you need as locals:\n\n##### Named:\n\n```js\nimport { h, render, Component } from 'preact';\n\n// Tell Babel to transform JSX into h() calls:\n/** @jsx h */\n```\n\n##### Default:\n\n```js\nimport preact from 'preact';\n\n// Tell Babel to transform JSX into preact.h() calls:\n/** @jsx preact.h */\n```\n\n> Named imports work well for highly structured applications, whereas the default import is quick and never needs to be updated when using different parts of the library.\n>\n> Instead of declaring the `@jsx` pragma in your code, it's best to configure it globally in a `.babelrc`:\n>\n> **For Babel 5 and prior:**\n>\n> ```json\n> { \"jsxPragma\": \"h\" }\n> ```\n>\n> **For Babel 6:**\n>\n> ```json\n> {\n> \"plugins\": [\n> [\"transform-react-jsx\", { \"pragma\":\"h\" }]\n> ]\n> }\n> ```\n>\n> **For Babel 7:**\n>\n> ```json\n> {\n> \"plugins\": [\n> [\"@babel/plugin-transform-react-jsx\", { \"pragma\":\"h\" }]\n> ]\n> }\n> ```\n> **For using Preact along with TypeScript add to `tsconfig.json`:**\n>\n> ```json\n> {\n> \"jsx\": \"react\",\n> \"jsxFactory\": \"h\",\n> }\n> ```\n\n\n### Rendering JSX\n\nOut of the box, Preact provides an `h()` function that turns your JSX into Virtual DOM elements _([here's how](http://jasonformat.com/wtf-is-jsx))_. It also provides a `render()` function that creates a DOM tree from that Virtual DOM.\n\nTo render some JSX, just import those two functions and use them like so:\n\n```js\nimport { h, render } from 'preact';\n\nrender((\n\t
\n\t\tHello, world!\n\t\t\n\t
\n), document.body);\n```\n\nThis should seem pretty straightforward if you've used hyperscript or one of its many friends. If you're not, the short of it is that the `h()` function import gets used in the final, transpiled code as a drop in replacement for `React.createElement()`, and so needs to be imported even if you don't explicitly use it in the code you write. Also note that if you're the kind of person who likes writing your React code in \"pure JavaScript\" (you know who you are) you will need to use `h()` wherever you would otherwise use `React.createElement()`.\n\nRendering hyperscript with a virtual DOM is pointless, though. We want to render components and have them updated when data changes - that's where the power of virtual DOM diffing shines. :star2:\n\n\n### Components\n\nPreact exports a generic `Component` class, which can be extended to build encapsulated, self-updating pieces of a User Interface. Components support all of the standard React [lifecycle methods], like `shouldComponentUpdate()` and `componentWillReceiveProps()`. Providing specific implementations of these methods is the preferred mechanism for controlling _when_ and _how_ components update.\n\nComponents also have a `render()` method, but unlike React this method is passed `(props, state)` as arguments. This provides an ergonomic means to destructure `props` and `state` into local variables to be referenced from JSX.\n\nLet's take a look at a very simple `Clock` component, which shows the current time.\n\n```js\nimport { h, render, Component } from 'preact';\n\nclass Clock extends Component {\n\trender() {\n\t\tlet time = new Date();\n\t\treturn ;\n\t}\n}\n\n// render an instance of Clock into :\nrender(, document.body);\n```\n\n\nThat's great. Running this produces the following HTML DOM structure:\n\n```html\n10:28:57 PM\n```\n\nIn order to have the clock's time update every second, we need to know when `` gets mounted to the DOM. _If you've used HTML5 Custom Elements, this is similar to the `attachedCallback` and `detachedCallback` lifecycle methods._ Preact invokes the following lifecycle methods if they are defined for a Component:\n\n| Lifecycle method | When it gets called |\n|-----------------------------|--------------------------------------------------|\n| `componentWillMount` | before the component gets mounted to the DOM |\n| `componentDidMount` | after the component gets mounted to the DOM |\n| `componentWillUnmount` | prior to removal from the DOM |\n| `componentWillReceiveProps` | before new props get accepted |\n| `shouldComponentUpdate` | before `render()`. Return `false` to skip render |\n| `componentWillUpdate` | before `render()` |\n| `componentDidUpdate` | after `render()` |\n\n\n\nSo, we want to have a 1-second timer start once the Component gets added to the DOM, and stop if it is removed. We'll create the timer and store a reference to it in `componentDidMount()`, and stop the timer in `componentWillUnmount()`. On each timer tick, we'll update the component's `state` object with a new time value. Doing this will automatically re-render the component.\n\n```js\nimport { h, render, Component } from 'preact';\n\nclass Clock extends Component {\n\tconstructor() {\n\t\tsuper();\n\t\t// set initial time:\n\t\tthis.state = {\n\t\t\ttime: Date.now()\n\t\t};\n\t}\n\n\tcomponentDidMount() {\n\t\t// update time every second\n\t\tthis.timer = setInterval(() => {\n\t\t\tthis.setState({ time: Date.now() });\n\t\t}, 1000);\n\t}\n\n\tcomponentWillUnmount() {\n\t\t// stop when not renderable\n\t\tclearInterval(this.timer);\n\t}\n\n\trender(props, state) {\n\t\tlet time = new Date(state.time).toLocaleTimeString();\n\t\treturn { time };\n\t}\n}\n\n// render an instance of Clock into :\nrender(, document.body);\n```\n\nNow we have [a ticking clock](http://jsfiddle.net/developit/u9m5x0L7/embedded/result,js/)!\n\n\n### Props & State\n\nThe concept (and nomenclature) for `props` and `state` is the same as in React. `props` are passed to a component by defining attributes in JSX, `state` is internal state. Changing either triggers a re-render, though by default Preact re-renders Components asynchronously for `state` changes and synchronously for `props` changes. You can tell Preact to render `prop` changes asynchronously by setting `options.syncComponentUpdates` to `false`.\n\n\n---\n\n\n## Linked State\n\nOne area Preact takes a little further than React is in optimizing state changes. A common pattern in ES2015 React code is to use Arrow functions within a `render()` method in order to update state in response to events. Creating functions enclosed in a scope on every render is inefficient and forces the garbage collector to do more work than is necessary.\n\nOne solution to this is to bind component methods declaratively.\nHere is an example using [decko](http://git.io/decko):\n\n```js\nclass Foo extends Component {\n\t@bind\n\tupdateText(e) {\n\t\tthis.setState({ text: e.target.value });\n\t}\n\trender({ }, { text }) {\n\t\treturn ;\n\t}\n}\n```\n\nWhile this achieves much better runtime performance, it's still a lot of unnecessary code to wire up state to UI.\n\nFortunately there is a solution, in the form of a module called [linkstate](https://github.com/developit/linkstate). Calling `linkState(component, 'text')` returns a function that accepts an Event and uses its associated value to update the given property in your component's state. Calls to `linkState()` with the same arguments are cached, so there is no performance penalty. Here is the previous example rewritten using _Linked State_:\n\n```js\nimport linkState from 'linkstate';\n\nclass Foo extends Component {\n\trender({ }, { text }) {\n\t\treturn ;\n\t}\n}\n```\n\nSimple and effective. It handles linking state from any input type, or an optional second parameter can be used to explicitly provide a keypath to the new state value.\n\n> **Note:** In Preact 7 and prior, `linkState()` was built right into Component. In 8.0, it was moved to a separate module. You can restore the 7.x behavior by using linkstate as a polyfill - see [the linkstate docs](https://github.com/developit/linkstate#usage).\n\n\n\n## Examples\n\nHere is a somewhat verbose Preact `` component:\n\n```js\nclass Link extends Component {\n\trender(props, state) {\n\t\treturn {props.children};\n\t}\n}\n```\n\nSince this is ES6/ES2015, we can further simplify:\n\n```js\nclass Link extends Component {\n render({ href, children }) {\n return ;\n }\n}\n\n// or, for wide-open props support:\nclass Link extends Component {\n render(props) {\n return ;\n }\n}\n\n// or, as a stateless functional component:\nconst Link = ({ children, ...props }) => (\n { children }\n);\n```\n\n\n## Extensions\n\nIt is likely that some projects based on Preact would wish to extend Component with great new functionality.\n\nPerhaps automatic connection to stores for a Flux-like architecture, or mixed-in context bindings to make it feel more like `React.createClass()`. Just use ES2015 inheritance:\n\n```js\nclass BoundComponent extends Component {\n\tconstructor(props) {\n\t\tsuper(props);\n\t\tthis.bind();\n\t}\n\tbind() {\n\t\tthis.binds = {};\n\t\tfor (let i in this) {\n\t\t\tthis.binds[i] = this[i].bind(this);\n\t\t}\n\t}\n}\n\n// example usage\nclass Link extends BoundComponent {\n\tclick() {\n\t\topen(this.href);\n\t}\n\trender() {\n\t\tlet { click } = this.binds;\n\t\treturn { children };\n\t}\n}\n```\n\n\nThe possibilities are pretty endless here. You could even add support for rudimentary mixins:\n\n```js\nclass MixedComponent extends Component {\n\tconstructor() {\n\t\tsuper();\n\t\t(this.mixins || []).forEach( m => Object.assign(this, m) );\n\t}\n}\n```\n\n## Debug Mode\n\nYou can inspect and modify the state of your Preact UI components at runtime using the\n[React Developer Tools](https://github.com/facebook/react-devtools) browser extension.\n\n1. Install the [React Developer Tools](https://github.com/facebook/react-devtools) extension\n2. Import the \"preact/debug\" module in your app\n3. Set `process.env.NODE_ENV` to 'development'\n4. Reload and go to the 'React' tab in the browser's development tools\n\n\n```js\nimport { h, Component, render } from 'preact';\n\n// Enable debug mode. You can reduce the size of your app by only including this\n// module in development builds. eg. In Webpack, wrap this with an `if (module.hot) {...}`\n// check.\nrequire('preact/debug');\n```\n\n### Runtime Error Checking\n\nTo enable debug mode, you need to set `process.env.NODE_ENV=development`. You can do this\nwith webpack via a builtin plugin.\n\n```js\n// webpack.config.js\n\n// Set NODE_ENV=development to enable error checking\nnew webpack.DefinePlugin({\n 'process.env': {\n 'NODE_ENV': JSON.stringify('development')\n }\n});\n```\n\nWhen enabled, warnings are logged to the console when undefined components or string refs\nare detected.\n\n### Developer Tools\n\nIf you only want to include devtool integration, without runtime error checking, you can\nreplace `preact/debug` in the above example with `preact/devtools`. This option doesn't\nrequire setting `NODE_ENV=development`.\n\n\n\n## Backers\nSupport us with a monthly donation and help us continue our activities. [[Become a backer](https://opencollective.com/preact#backer)]\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n## Sponsors\nBecome a sponsor and get your logo on our README on GitHub with a link to your site. [[Become a sponsor](https://opencollective.com/preact#sponsor)]\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n## License\n\nMIT\n\n\n\n[![Preact](http://i.imgur.com/YqCHvEW.gif)](https://preactjs.com)\n\n\n[preact-compat]: https://github.com/developit/preact-compat\n[ES6 Class]: https://facebook.github.io/react/docs/reusable-components.html#es6-classes\n[Functional Components]: https://facebook.github.io/react/blog/2015/10/07/react-v0.14.html#stateless-functional-components\n[hyperscript]: https://github.com/dominictarr/hyperscript\n[preact-boilerplate]: https://github.com/developit/preact-boilerplate\n[lifecycle methods]: https://facebook.github.io/react/docs/component-specs.html\n", + "readmeFilename": "README.md", + "_id": "preact@10.0.0-rc.1", + "_nodeVersion": "11.15.0", + "_npmVersion": "6.10.1", + "dist": { + "integrity": "sha512-DyG8ySCbrQZ8w4cSyxnofM6yH/hzyYLHmIrE3xL1FjSCX4DpvVmP7DTpkV535FSQX0d2zOscb7DCIY+crBtxow==", + "shasum": "bae4419f8e289c327504d4ee6e734c377983548c", + "tarball": "https://registry.npmjs.org/preact/-/preact-10.0.0-rc.1.tgz", + "fileCount": 72, + "unpackedSize": 679393, + "npm-signature": "-----BEGIN PGP SIGNATURE-----\r\nVersion: OpenPGP.js v3.0.4\r\nComment: https://openpgpjs.org\r\n\r\nwsFcBAEBCAAQBQJdRJ5lCRA9TVsSAnZWagAAdW0P/1LRcC3kme3I27EBoRS3\nL58mBrQ/MxxMH687VNc2KOq8/oQnbT+PfxNQMvrNJPNl5t1flutUnjZWoezX\nZnbmmLj4tlA3PD37X6w8upDt6XW+dRIGUCYPioUT+vaR+U3m4/10wuUon5yO\nWg/IjspZ0HiauScdPSAev6s5hGeKKPZsjcAMWHT1ahSCU94Hnny9aZvzDV+J\nde7THknOGlkxdkOteJ4AL46qanFTZTDoebm4mzIZOHgk51RVnne+CPLbMkLv\nJEXgJP0TWyVSzkwRmnEgKbd80f6IE0CA4aoV1ye0O7WocKNy4fNq65OqVN/v\n4TbPMsxqQV8QvxRXQ8cAfW64Z3fQwfeJj40DOQPa7E7pfac3wvxlvXEXdiYc\nfDAw2VHPrkWDNUHJG3amBq5ZA6LqHEtD/Ewd/+LgP2VOAXMucCjWnMWqdI99\n3+gDp27jZBPYBqClYzpjO9pRouG0ESbnWEFHpF2f41/9oOJGQBsvoewsCU1J\nBhJMV+iPv3fK6N1u+StIqGDtssJ2iA8riswjMo7VqQJ232sTUPW+4zeKMhjP\n8dBg8jt/tDM7CdCHUvYsrsxuTMjeoc83nEhI13DfnJzxheN1s76pUaGzIdUs\n388SNBoIbeseE2QR/3vTbNZTcQW82r0gHhr6Z/cJqKwQSo6n3+Dbc8IOgNkR\nEaZt\r\n=0QW8\r\n-----END PGP SIGNATURE-----\r\n" + }, + "maintainers": [ + { "email": "jason@developit.ca", "name": "developit" }, + { "email": "ulliftw@gmail.com", "name": "harmony" }, + { "email": "luke@lukeed.com", "name": "lukeed" }, + { "email": "marvin@marvinhagemeister.de", "name": "marvinhagemeister" }, + { "email": "prateek89born@gmail.com", "name": "prateekbh" }, + { "email": "hello@preactjs.com", "name": "preactjs" }, + { "email": "allamsetty.anup@gmail.com", "name": "reznord" } + ], + "_npmUser": { + "name": "marvinhagemeister", + "email": "marvin@marvinhagemeister.de" + }, + "directories": {}, + "_npmOperationalInternal": { + "host": "s3://npm-registry-packages", + "tmp": "tmp/preact_10.0.0-rc.1_1564778084833_0.9974358457635735" + }, + "_hasShrinkwrap": false + } + }, + "readme": "", + "maintainers": [ + { "email": "jason@developit.ca", "name": "developit" }, + { "email": "ulliftw@gmail.com", "name": "harmony" }, + { "email": "luke@lukeed.com", "name": "lukeed" }, + { "email": "marvin@marvinhagemeister.de", "name": "marvinhagemeister" }, + { "email": "prateek89born@gmail.com", "name": "prateekbh" }, + { "email": "hello@preactjs.com", "name": "preactjs" }, + { "email": "allamsetty.anup@gmail.com", "name": "reznord" } + ], + "time": { + "modified": "2019-08-02T20:34:48.321Z", + "created": "2015-09-11T02:41:33.521Z", + "8.5.0": "2019-08-02T18:34:23.572Z", + "10.0.0-rc.1": "2019-08-02T20:34:45.123Z" + }, + "homepage": "https://github.com/developit/preact", + "repository": { + "type": "git", + "url": "https://github.com/developit/preact.git" + }, + "bugs": { "url": "https://github.com/developit/preact/issues" }, + "license": "MIT", + "readmeFilename": "", + "users": { + "developit": true, + "pje": true, + "kratyk": true, + "rexpan": true, + "abhisekp": true, + "billneff79": true, + "pixel67": true, + "shanewholloway": true, + "princetoad": true, + "charlespeters": true, + "gcwelborn": true, + "lassevolkmann": true, + "erikvold": true, + "iamale": true, + "xueboren": true, + "grahm": true, + "d-band": true, + "daniellink": true, + "youtwo": true, + "rethinkflash": true, + "kkho595": true, + "sangdth": true, + "sternelee": true, + "tztz": true, + "alexparish": true, + "petershev": true, + "wayn": true, + "sshrike": true, + "vpzomtrrfrt": true, + "severen": true, + "mdedirudianto": true, + "huiyifyj": true, + "karzanosman984": true + }, + "keywords": [ + "preact", + "react", + "virtual dom", + "vdom", + "components", + "virtual", + "dom" + ], + "author": { "name": "Jason Miller", "email": "jason@developit.ca" } +} diff --git a/src/formatPkg.js b/src/formatPkg.js index b4355f160..dbe479078 100644 --- a/src/formatPkg.js +++ b/src/formatPkg.js @@ -60,7 +60,7 @@ export default function formatPkg(pkg) { const dependencies = cleaned.dependencies || {}; const devDependencies = cleaned.devDependencies || {}; const alternativeNames = getAlternativeNames(cleaned.name); - const moduleType = getModuleType(cleaned); + const moduleTypes = getModuleTypes(cleaned); const tags = pkg['dist-tags']; @@ -95,7 +95,7 @@ export default function formatPkg(pkg) { owners: (cleaned.owners || []).map(formatUser), bin: cleaned.bin, types, - moduleType, + moduleTypes, lastCrawl: new Date().toISOString(), _searchInternal: { alternativeNames, @@ -434,17 +434,23 @@ function getAlternativeNames(name) { ); } -function getModuleType(cleaned) { +function getModuleTypes(cleaned) { const main = cleaned.main || ''; + const moduleTypes = []; if ( typeof cleaned.module === 'string' || cleaned.type === 'module' || main.endsWith('.mjs') ) { - return 'esm'; + moduleTypes.push('esm'); } if (cleaned.type === 'commonjs' || main.endsWith('.cjs')) { - return 'cjs'; + moduleTypes.push('cjs'); } - return 'unknown'; + + if (moduleTypes.length === 0) { + moduleTypes.push('unknown'); + } + + return moduleTypes; } From 83447e6d958c2ef4c2f0ae4ed8a3ee77f6e595d1 Mon Sep 17 00:00:00 2001 From: Haroen Viaene Date: Wed, 7 Aug 2019 10:21:57 +0200 Subject: [PATCH 3/3] finish plural migration --- src/__tests__/__snapshots__/config.test.js.snap | 2 +- src/config.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/__tests__/__snapshots__/config.test.js.snap b/src/__tests__/__snapshots__/config.test.js.snap index 536a806e9..253f159b8 100644 --- a/src/__tests__/__snapshots__/config.test.js.snap +++ b/src/__tests__/__snapshots__/config.test.js.snap @@ -96,7 +96,7 @@ Object { "searchable(owner.name)", "deprecated", "types.ts", - "moduleType", + "moduleTypes", ], "customRanking": Array [ "desc(_searchInternal.downloadsMagnitude)", diff --git a/src/config.js b/src/config.js index 3edd6b6d5..0a55a9c7c 100644 --- a/src/config.js +++ b/src/config.js @@ -35,7 +35,7 @@ const defaultConfig = { 'searchable(owner.name)', 'deprecated', 'types.ts', - 'moduleType', + 'moduleTypes', ], customRanking: [ 'desc(_searchInternal.downloadsMagnitude)',