3 module.exports = function( grunt ) {
4 function readOptionalJSON( filepath ) {
5 const stripJSONComments = require( "strip-json-comments" );
8 data = JSON.parse( stripJSONComments(
9 fs.readFileSync( filepath, { encoding: "utf8" } )
15 const fs = require( "fs" );
16 const gzip = require( "gzip-js" );
17 const nodeV14OrNewer = !/^v1[0-3]\./.test( process.version );
18 const nodeV17OrNewer = !/^v1[0-6]\./.test( process.version );
19 const customBrowsers = process.env.BROWSERS && process.env.BROWSERS.split( "," );
21 // Support: Node.js <14
22 // Skip running tasks that dropped support for Node.js 10 or 12
23 // in this Node version.
24 function runIfNewNode( task ) {
25 return nodeV14OrNewer ? task : "print_old_node_message:" + task;
28 if ( nodeV14OrNewer ) {
29 const playwright = require( "playwright-webkit" );
30 process.env.WEBKIT_HEADLESS_BIN = playwright.webkit.executablePath();
33 if ( !grunt.option( "filename" ) ) {
34 grunt.option( "filename", "jquery.js" );
38 pkg: grunt.file.readJSON( "package.json" ),
39 dst: readOptionalJSON( "dist/.destination.json" ),
41 files: [ "dist/jquery.js", "dist/jquery.min.js" ],
44 gz: function( contents ) {
45 return gzip.zip( contents, {} ).length;
48 cache: "build/.sizecache.json"
55 plugins: [ "@babel/transform-for-of" ]
59 "test/data/core/jquery-iterability-transpiled.js":
60 "test/data/core/jquery-iterability-transpiled-es6.js"
66 dest: "dist/jquery.js",
71 // Exclude specified modules if the module matching the key is removed
73 ajax: [ "manipulation/_evalUrl", "deprecated/ajax-event-alias" ],
74 callbacks: [ "deferred" ],
75 css: [ "effects", "dimensions", "offset" ],
76 "css/showHide": [ "effects" ],
78 remove: [ "ajax", "effects", "queue", "core/ready" ],
79 include: [ "core/ready-no-deferred" ]
81 event: [ "deprecated/ajax-event-alias", "deprecated/event" ],
82 selector: [ "css/hiddenVisibleSelectors", "effects/animatedSelector" ]
89 destPrefix: "external"
92 "bootstrap/bootstrap.css": "bootstrap/dist/css/bootstrap.css",
93 "bootstrap/bootstrap.min.css": "bootstrap/dist/css/bootstrap.min.css",
94 "bootstrap/bootstrap.min.css.map": "bootstrap/dist/css/bootstrap.min.css.map",
96 "core-js-bundle/core-js-bundle.js": "core-js-bundle/minified.js",
97 "core-js-bundle/LICENSE": "core-js-bundle/LICENSE",
99 "npo/npo.js": "native-promise-only/lib/npo.src.js",
101 "qunit/qunit.js": "qunit/qunit/qunit.js",
102 "qunit/qunit.css": "qunit/qunit/qunit.css",
103 "qunit/LICENSE.txt": "qunit/LICENSE.txt",
105 "requirejs/require.js": "requirejs/require.js",
107 "sinon/sinon.js": "sinon/pkg/sinon.js",
108 "sinon/LICENSE.txt": "sinon/LICENSE"
114 src: [ "package.json" ]
122 // We have to explicitly declare "src" property otherwise "newer"
123 // task wouldn't work properly :/
125 src: [ "dist/jquery.js", "dist/jquery.min.js" ]
134 // Ignore files from .eslintignore
135 // See https://github.com/sindresorhus/grunt-eslint/issues/119
137 .readFileSync( `${ __dirname }/.eslintignore`, "utf-8" )
139 .filter( filePath => filePath )
140 .map( filePath => filePath[ 0 ] === "!" ?
141 filePath.slice( 1 ) :
145 // Explicitly ignore `dist/` as it could be unignored by
146 // the above `.eslintignore` parsing.
154 // A special module with basic tests, meant for not fully
155 // supported environments like jsdom. We run it everywhere,
156 // though, to make sure tests are not broken.
183 customContextFile: "test/karma.context.html",
184 customDebugFile: "test/karma.debug.html",
186 ChromeHeadlessNoSandbox: {
187 base: "ChromeHeadless",
188 flags: [ "--no-sandbox" ]
191 frameworks: [ "qunit" ],
192 middleware: [ "mockserver" ],
196 "middleware:mockserver": [
198 require( "./test/middleware-mockserver.js" )
205 // We're running `QUnit.start()` ourselves via `loadTests()`
211 "test/data/jquery-1.9.1.js",
212 "external/sinon/sinon.js",
213 "external/npo/npo.js",
214 "external/requirejs/require.js",
215 "test/data/testinit.js",
220 pattern: "dist/jquery.*",
232 { pattern: "external/**", included: false, served: true },
234 pattern: "test/**/*.@(js|css|jpg|html|xml|svg)",
240 reporters: [ "dots" ],
243 // 2 minutes; has to be longer than QUnit.config.testTimeout
244 browserNoActivityTimeout: 120e3,
247 captureTimeout: 20 * 1000,
251 browsers: customBrowsers ||
252 [ "ChromeHeadless", "FirefoxHeadless", "WebkitHeadless" ]
255 browsers: customBrowsers || [ "ChromeHeadless" ],
260 // We're running `QUnit.start()` ourselves via `loadTests()`
273 "test/data/jquery-1.9.1.js",
274 "test/data/testinit-jsdom.js",
276 // We don't support various loading methods like esmodules,
277 // choosing a version etc. for jsdom.
280 // A partial replacement for testinit.js#loadTests()
281 "test/data/testrunner.js",
283 // jsdom only runs basic tests
284 "test/unit/basic.js",
287 pattern: "test/**/*.@(js|css|jpg|html|xml|svg)",
293 browsers: [ "jsdom" ]
296 // To debug tests with Karma:
297 // 1. Run 'grunt karma:chrome-debug' or 'grunt karma:firefox-debug'
298 // (any karma subtask that has singleRun=false)
299 // 2. Press "Debug" in the opened browser window to start
300 // the tests. Unlike the other karma tasks, the debug task will
301 // keep the browser window open.
303 browsers: [ "Chrome" ],
307 browsers: [ "Firefox" ],
316 files: [ "<%= eslint.dev.src %>" ],
322 "dist/<%= grunt.option('filename').replace('.js', '.min.js') %>":
323 "dist/<%= grunt.option('filename') %>"
327 filename: "dist/<%= grunt.option('filename').replace('.js', '.min.map') %>",
329 // The map's `files` & `sources` property are set incorrectly, fix
330 // them via overrides from the task config.
331 // See https://github.com/swc-project/swc/issues/7588#issuecomment-1624345254
333 file: "jquery.min.js",
344 preamble: "/*! jQuery v4.0.0-pre | " +
345 "(c) OpenJS Foundation and other contributors | " +
346 "jquery.org/license */\n"
360 // Load grunt tasks from NPM packages
361 require( "load-grunt-tasks" )( grunt, {
362 pattern: nodeV14OrNewer ? [ "grunt-*" ] : [ "grunt-*", "!grunt-eslint" ]
365 // Integrate jQuery specific tasks
366 grunt.loadTasks( "build/tasks" );
368 grunt.registerTask( "print_old_node_message", ( ...args ) => {
369 var task = args.join( ":" );
370 grunt.log.writeln( "Old Node.js detected, running the task \"" + task + "\" skipped..." );
373 grunt.registerTask( "print_jsdom_message", () => {
374 grunt.log.writeln( "Node.js 17 or newer detected, skipping jsdom tests..." );
377 grunt.registerTask( "lint", [
380 // Running the full eslint task without breaking it down to targets
381 // would run the dist target first which would point to errors in the built
382 // file, making it harder to fix them. We want to check the built file only
383 // if we already know the source files pass the linter.
384 runIfNewNode( "eslint:dev" ),
385 runIfNewNode( "eslint:dist" )
388 grunt.registerTask( "lint:newer", [
391 // Don't replace it with just the task; see the above comment.
392 runIfNewNode( "newer:eslint:dev" ),
393 runIfNewNode( "newer:eslint:dist" )
396 grunt.registerTask( "test:fast", runIfNewNode( "node_smoke_tests" ) );
397 grunt.registerTask( "test:slow", [
398 runIfNewNode( "promises_aplus_tests" ),
400 // Support: Node.js 17+
401 // jsdom fails to connect to the Karma server in Node 17+.
402 // Until we figure out a fix, skip jsdom tests there.
403 nodeV17OrNewer ? "print_jsdom_message" : runIfNewNode( "karma:jsdom" )
406 grunt.registerTask( "test:prepare", [
412 grunt.registerTask( "test", [
418 grunt.registerTask( "dev", [
420 runIfNewNode( "newer:eslint:dev" ),
422 "remove_map_comment",
428 grunt.registerTask( "default", [
429 runIfNewNode( "eslint:dev" ),
432 "remove_map_comment",
435 runIfNewNode( "eslint:dist" ),