source: imaps-frontend/node_modules/loader-runner/lib/LoaderRunner.js@ 79a0317

main
Last change on this file since 79a0317 was 79a0317, checked in by stefan toskovski <stefantoska84@…>, 3 days ago

F4 Finalna Verzija

  • Property mode set to 100644
File size: 12.4 KB
Line 
1/*
2 MIT License http://www.opensource.org/licenses/mit-license.php
3 Author Tobias Koppers @sokra
4*/
5var fs = require("fs");
6var readFile = fs.readFile.bind(fs);
7var loadLoader = require("./loadLoader");
8
9function utf8BufferToString(buf) {
10 var str = buf.toString("utf-8");
11 if(str.charCodeAt(0) === 0xFEFF) {
12 return str.substr(1);
13 } else {
14 return str;
15 }
16}
17
18const PATH_QUERY_FRAGMENT_REGEXP = /^((?:\0.|[^?#\0])*)(\?(?:\0.|[^#\0])*)?(#.*)?$/;
19
20/**
21 * @param {string} str the path with query and fragment
22 * @returns {{ path: string, query: string, fragment: string }} parsed parts
23 */
24function parsePathQueryFragment(str) {
25 var match = PATH_QUERY_FRAGMENT_REGEXP.exec(str);
26 return {
27 path: match[1].replace(/\0(.)/g, "$1"),
28 query: match[2] ? match[2].replace(/\0(.)/g, "$1") : "",
29 fragment: match[3] || ""
30 };
31}
32
33function dirname(path) {
34 if(path === "/") return "/";
35 var i = path.lastIndexOf("/");
36 var j = path.lastIndexOf("\\");
37 var i2 = path.indexOf("/");
38 var j2 = path.indexOf("\\");
39 var idx = i > j ? i : j;
40 var idx2 = i > j ? i2 : j2;
41 if(idx < 0) return path;
42 if(idx === idx2) return path.substr(0, idx + 1);
43 return path.substr(0, idx);
44}
45
46function createLoaderObject(loader) {
47 var obj = {
48 path: null,
49 query: null,
50 fragment: null,
51 options: null,
52 ident: null,
53 normal: null,
54 pitch: null,
55 raw: null,
56 data: null,
57 pitchExecuted: false,
58 normalExecuted: false
59 };
60 Object.defineProperty(obj, "request", {
61 enumerable: true,
62 get: function() {
63 return obj.path.replace(/#/g, "\0#") + obj.query.replace(/#/g, "\0#") + obj.fragment;
64 },
65 set: function(value) {
66 if(typeof value === "string") {
67 var splittedRequest = parsePathQueryFragment(value);
68 obj.path = splittedRequest.path;
69 obj.query = splittedRequest.query;
70 obj.fragment = splittedRequest.fragment;
71 obj.options = undefined;
72 obj.ident = undefined;
73 } else {
74 if(!value.loader)
75 throw new Error("request should be a string or object with loader and options (" + JSON.stringify(value) + ")");
76 obj.path = value.loader;
77 obj.fragment = value.fragment || "";
78 obj.type = value.type;
79 obj.options = value.options;
80 obj.ident = value.ident;
81 if(obj.options === null)
82 obj.query = "";
83 else if(obj.options === undefined)
84 obj.query = "";
85 else if(typeof obj.options === "string")
86 obj.query = "?" + obj.options;
87 else if(obj.ident)
88 obj.query = "??" + obj.ident;
89 else if(typeof obj.options === "object" && obj.options.ident)
90 obj.query = "??" + obj.options.ident;
91 else
92 obj.query = "?" + JSON.stringify(obj.options);
93 }
94 }
95 });
96 obj.request = loader;
97 if(Object.preventExtensions) {
98 Object.preventExtensions(obj);
99 }
100 return obj;
101}
102
103function runSyncOrAsync(fn, context, args, callback) {
104 var isSync = true;
105 var isDone = false;
106 var isError = false; // internal error
107 var reportedError = false;
108 context.async = function async() {
109 if(isDone) {
110 if(reportedError) return; // ignore
111 throw new Error("async(): The callback was already called.");
112 }
113 isSync = false;
114 return innerCallback;
115 };
116 var innerCallback = context.callback = function() {
117 if(isDone) {
118 if(reportedError) return; // ignore
119 throw new Error("callback(): The callback was already called.");
120 }
121 isDone = true;
122 isSync = false;
123 try {
124 callback.apply(null, arguments);
125 } catch(e) {
126 isError = true;
127 throw e;
128 }
129 };
130 try {
131 var result = (function LOADER_EXECUTION() {
132 return fn.apply(context, args);
133 }());
134 if(isSync) {
135 isDone = true;
136 if(result === undefined)
137 return callback();
138 if(result && typeof result === "object" && typeof result.then === "function") {
139 return result.then(function(r) {
140 callback(null, r);
141 }, callback);
142 }
143 return callback(null, result);
144 }
145 } catch(e) {
146 if(isError) throw e;
147 if(isDone) {
148 // loader is already "done", so we cannot use the callback function
149 // for better debugging we print the error on the console
150 if(typeof e === "object" && e.stack) console.error(e.stack);
151 else console.error(e);
152 return;
153 }
154 isDone = true;
155 reportedError = true;
156 callback(e);
157 }
158
159}
160
161function convertArgs(args, raw) {
162 if(!raw && Buffer.isBuffer(args[0]))
163 args[0] = utf8BufferToString(args[0]);
164 else if(raw && typeof args[0] === "string")
165 args[0] = Buffer.from(args[0], "utf-8");
166}
167
168function iteratePitchingLoaders(options, loaderContext, callback) {
169 // abort after last loader
170 if(loaderContext.loaderIndex >= loaderContext.loaders.length)
171 return processResource(options, loaderContext, callback);
172
173 var currentLoaderObject = loaderContext.loaders[loaderContext.loaderIndex];
174
175 // iterate
176 if(currentLoaderObject.pitchExecuted) {
177 loaderContext.loaderIndex++;
178 return iteratePitchingLoaders(options, loaderContext, callback);
179 }
180
181 // load loader module
182 loadLoader(currentLoaderObject, function(err) {
183 if(err) {
184 loaderContext.cacheable(false);
185 return callback(err);
186 }
187 var fn = currentLoaderObject.pitch;
188 currentLoaderObject.pitchExecuted = true;
189 if(!fn) return iteratePitchingLoaders(options, loaderContext, callback);
190
191 runSyncOrAsync(
192 fn,
193 loaderContext, [loaderContext.remainingRequest, loaderContext.previousRequest, currentLoaderObject.data = {}],
194 function(err) {
195 if(err) return callback(err);
196 var args = Array.prototype.slice.call(arguments, 1);
197 // Determine whether to continue the pitching process based on
198 // argument values (as opposed to argument presence) in order
199 // to support synchronous and asynchronous usages.
200 var hasArg = args.some(function(value) {
201 return value !== undefined;
202 });
203 if(hasArg) {
204 loaderContext.loaderIndex--;
205 iterateNormalLoaders(options, loaderContext, args, callback);
206 } else {
207 iteratePitchingLoaders(options, loaderContext, callback);
208 }
209 }
210 );
211 });
212}
213
214function processResource(options, loaderContext, callback) {
215 // set loader index to last loader
216 loaderContext.loaderIndex = loaderContext.loaders.length - 1;
217
218 var resourcePath = loaderContext.resourcePath;
219 if(resourcePath) {
220 options.processResource(loaderContext, resourcePath, function(err) {
221 if(err) return callback(err);
222 var args = Array.prototype.slice.call(arguments, 1);
223 options.resourceBuffer = args[0];
224 iterateNormalLoaders(options, loaderContext, args, callback);
225 });
226 } else {
227 iterateNormalLoaders(options, loaderContext, [null], callback);
228 }
229}
230
231function iterateNormalLoaders(options, loaderContext, args, callback) {
232 if(loaderContext.loaderIndex < 0)
233 return callback(null, args);
234
235 var currentLoaderObject = loaderContext.loaders[loaderContext.loaderIndex];
236
237 // iterate
238 if(currentLoaderObject.normalExecuted) {
239 loaderContext.loaderIndex--;
240 return iterateNormalLoaders(options, loaderContext, args, callback);
241 }
242
243 var fn = currentLoaderObject.normal;
244 currentLoaderObject.normalExecuted = true;
245 if(!fn) {
246 return iterateNormalLoaders(options, loaderContext, args, callback);
247 }
248
249 convertArgs(args, currentLoaderObject.raw);
250
251 runSyncOrAsync(fn, loaderContext, args, function(err) {
252 if(err) return callback(err);
253
254 var args = Array.prototype.slice.call(arguments, 1);
255 iterateNormalLoaders(options, loaderContext, args, callback);
256 });
257}
258
259exports.getContext = function getContext(resource) {
260 var path = parsePathQueryFragment(resource).path;
261 return dirname(path);
262};
263
264exports.runLoaders = function runLoaders(options, callback) {
265 // read options
266 var resource = options.resource || "";
267 var loaders = options.loaders || [];
268 var loaderContext = options.context || {};
269 var processResource = options.processResource || ((readResource, context, resource, callback) => {
270 context.addDependency(resource);
271 readResource(resource, callback);
272 }).bind(null, options.readResource || readFile);
273
274 //
275 var splittedResource = resource && parsePathQueryFragment(resource);
276 var resourcePath = splittedResource ? splittedResource.path : undefined;
277 var resourceQuery = splittedResource ? splittedResource.query : undefined;
278 var resourceFragment = splittedResource ? splittedResource.fragment : undefined;
279 var contextDirectory = resourcePath ? dirname(resourcePath) : null;
280
281 // execution state
282 var requestCacheable = true;
283 var fileDependencies = [];
284 var contextDependencies = [];
285 var missingDependencies = [];
286
287 // prepare loader objects
288 loaders = loaders.map(createLoaderObject);
289
290 loaderContext.context = contextDirectory;
291 loaderContext.loaderIndex = 0;
292 loaderContext.loaders = loaders;
293 loaderContext.resourcePath = resourcePath;
294 loaderContext.resourceQuery = resourceQuery;
295 loaderContext.resourceFragment = resourceFragment;
296 loaderContext.async = null;
297 loaderContext.callback = null;
298 loaderContext.cacheable = function cacheable(flag) {
299 if(flag === false) {
300 requestCacheable = false;
301 }
302 };
303 loaderContext.dependency = loaderContext.addDependency = function addDependency(file) {
304 fileDependencies.push(file);
305 };
306 loaderContext.addContextDependency = function addContextDependency(context) {
307 contextDependencies.push(context);
308 };
309 loaderContext.addMissingDependency = function addMissingDependency(context) {
310 missingDependencies.push(context);
311 };
312 loaderContext.getDependencies = function getDependencies() {
313 return fileDependencies.slice();
314 };
315 loaderContext.getContextDependencies = function getContextDependencies() {
316 return contextDependencies.slice();
317 };
318 loaderContext.getMissingDependencies = function getMissingDependencies() {
319 return missingDependencies.slice();
320 };
321 loaderContext.clearDependencies = function clearDependencies() {
322 fileDependencies.length = 0;
323 contextDependencies.length = 0;
324 missingDependencies.length = 0;
325 requestCacheable = true;
326 };
327 Object.defineProperty(loaderContext, "resource", {
328 enumerable: true,
329 get: function() {
330 if(loaderContext.resourcePath === undefined)
331 return undefined;
332 return loaderContext.resourcePath.replace(/#/g, "\0#") + loaderContext.resourceQuery.replace(/#/g, "\0#") + loaderContext.resourceFragment;
333 },
334 set: function(value) {
335 var splittedResource = value && parsePathQueryFragment(value);
336 loaderContext.resourcePath = splittedResource ? splittedResource.path : undefined;
337 loaderContext.resourceQuery = splittedResource ? splittedResource.query : undefined;
338 loaderContext.resourceFragment = splittedResource ? splittedResource.fragment : undefined;
339 }
340 });
341 Object.defineProperty(loaderContext, "request", {
342 enumerable: true,
343 get: function() {
344 return loaderContext.loaders.map(function(o) {
345 return o.request;
346 }).concat(loaderContext.resource || "").join("!");
347 }
348 });
349 Object.defineProperty(loaderContext, "remainingRequest", {
350 enumerable: true,
351 get: function() {
352 if(loaderContext.loaderIndex >= loaderContext.loaders.length - 1 && !loaderContext.resource)
353 return "";
354 return loaderContext.loaders.slice(loaderContext.loaderIndex + 1).map(function(o) {
355 return o.request;
356 }).concat(loaderContext.resource || "").join("!");
357 }
358 });
359 Object.defineProperty(loaderContext, "currentRequest", {
360 enumerable: true,
361 get: function() {
362 return loaderContext.loaders.slice(loaderContext.loaderIndex).map(function(o) {
363 return o.request;
364 }).concat(loaderContext.resource || "").join("!");
365 }
366 });
367 Object.defineProperty(loaderContext, "previousRequest", {
368 enumerable: true,
369 get: function() {
370 return loaderContext.loaders.slice(0, loaderContext.loaderIndex).map(function(o) {
371 return o.request;
372 }).join("!");
373 }
374 });
375 Object.defineProperty(loaderContext, "query", {
376 enumerable: true,
377 get: function() {
378 var entry = loaderContext.loaders[loaderContext.loaderIndex];
379 return entry.options && typeof entry.options === "object" ? entry.options : entry.query;
380 }
381 });
382 Object.defineProperty(loaderContext, "data", {
383 enumerable: true,
384 get: function() {
385 return loaderContext.loaders[loaderContext.loaderIndex].data;
386 }
387 });
388
389 // finish loader context
390 if(Object.preventExtensions) {
391 Object.preventExtensions(loaderContext);
392 }
393
394 var processOptions = {
395 resourceBuffer: null,
396 processResource: processResource
397 };
398 iteratePitchingLoaders(processOptions, loaderContext, function(err, result) {
399 if(err) {
400 return callback(err, {
401 cacheable: requestCacheable,
402 fileDependencies: fileDependencies,
403 contextDependencies: contextDependencies,
404 missingDependencies: missingDependencies
405 });
406 }
407 callback(null, {
408 result: result,
409 resourceBuffer: processOptions.resourceBuffer,
410 cacheable: requestCacheable,
411 fileDependencies: fileDependencies,
412 contextDependencies: contextDependencies,
413 missingDependencies: missingDependencies
414 });
415 });
416};
Note: See TracBrowser for help on using the repository browser.