/** *------------------------------------------------------------------------------ * Odoo Web Boostrap Code *------------------------------------------------------------------------------ * * Each module can return a promise. In that case, the module is marked as loaded * only when the promise is resolved, and its value is equal to the resolved value. * The module can be rejected (unloaded). This will be logged in the console as info. * * logs: * Missing dependencies: * These modules do not appear in the page. It is possible that the * JavaScript file is not in the page or that the module name is wrong * Failed modules: * A javascript error is detected * Rejected modules: * The module returns a rejected promise. It (and its dependent modules) * is not loaded. * Rejected linked modules: * Modules who depend on a rejected module * Non loaded modules: * Modules who depend on a missing or a failed module * Debug: * Non loaded or failed module informations for debugging */ (function () { "use strict"; var jobUID = Date.now(); var jobs = []; var factories = Object.create(null); var jobDeps = []; var jobPromises = []; var services = Object.create({}); var commentRegExp = /(\/\*([\s\S]*?)\*\/|([^:]|^)\/\/(.*)$)/mg; var cjsRequireRegExp = /[^.]\s*require\s*\(\s*["']([^'"\s]+)["']\s*\)/g; if (!window.odoo) { window.odoo = {}; } var odoo = window.odoo; var didLogInfoResolve; var didLogInfoPromise = new Promise(function (resolve) { didLogInfoResolve = resolve; }); odoo.testing = typeof QUnit === 'object'; odoo.remainingJobs = jobs; odoo.__DEBUG__ = { didLogInfo: didLogInfoPromise, getDependencies: function (name, transitive) { var deps = name instanceof Array ? name : [name]; var changed; do { changed = false; jobDeps.forEach(function (dep) { if (deps.indexOf(dep.to) >= 0 && deps.indexOf(dep.from) < 0) { deps.push(dep.from); changed = true; } }); } while (changed && transitive); return deps; }, getDependents: function (name) { return jobDeps.filter(function (dep) { return dep.from === name; }).map(function (dep) { return dep.to; }); }, getWaitedJobs: function () { return jobs.map(function (job) { return job.name; }).filter(function (item, index, self) { // uniq return self.indexOf(item) === index; }); }, getMissingJobs: function () { var self = this; var waited = this.getWaitedJobs(); var missing = []; waited.forEach(function (job) { self.getDependencies(job).forEach(function (job) { if (!(job in self.services)) { missing.push(job); } }); }); return missing.filter(function (item, index, self) { return self.indexOf(item) === index; }).filter(function (item) { return waited.indexOf(item) < 0; }).filter(function (job) { return !job.error; }); }, getFailedJobs: function () { return jobs.filter(function (job) { return !!job.error; }); }, factories: factories, services: services, }; odoo.define = function () { var args = Array.prototype.slice.call(arguments); var name = typeof args[0] === 'string' ? args.shift() : ('__odoo_job' + (jobUID++)); var factory = args[args.length - 1]; var deps; if (args[0] instanceof Array) { deps = args[0]; } else { deps = []; factory.toString() .replace(commentRegExp, '') .replace(cjsRequireRegExp, function (match, dep) { deps.push(dep); }); } if (odoo.debug) { if (!(deps instanceof Array)) { throw new Error('Dependencies should be defined by an array', deps); } if (typeof factory !== 'function') { throw new Error('Factory should be defined by a function', factory); } if (typeof name !== 'string') { throw new Error("Invalid name definition (should be a string", name); } if (name in factories) { throw new Error("Service " + name + " already defined"); } } factory.deps = deps; factories[name] = factory; jobs.push({ name: name, factory: factory, deps: deps, }); deps.forEach(function (dep) { jobDeps.push({from: dep, to: name}); }); this.processJobs(jobs, services); }; odoo.log = function () { var missing = []; var failed = []; if (jobs.length) { var debugJobs = {}; var rejected = []; var rejectedLinked = []; var job; var jobdep; for (var k = 0; k < jobs.length; k++) { debugJobs[jobs[k].name] = job = { dependencies: jobs[k].deps, dependents: odoo.__DEBUG__.getDependents(jobs[k].name), name: jobs[k].name }; if (jobs[k].error) { job.error = jobs[k].error; } if (jobs[k].rejected) { job.rejected = jobs[k].rejected; rejected.push(job.name); } var deps = odoo.__DEBUG__.getDependencies(job.name); for (var i = 0; i < deps.length; i++) { if (job.name !== deps[i] && !(deps[i] in services)) { jobdep = debugJobs[deps[i]]; if (!jobdep && deps[i] in factories) { for (var j = 0; j < jobs.length; j++) { if (jobs[j].name === deps[i]) { jobdep = jobs[j]; break; } } } if (jobdep && jobdep.rejected) { if (!job.rejected) { job.rejected = []; rejectedLinked.push(job.name); } job.rejected.push(deps[i]); } else { if (!job.missing) { job.missing = []; } job.missing.push(deps[i]); } } } } missing = odoo.__DEBUG__.getMissingJobs(); failed = odoo.__DEBUG__.getFailedJobs(); var unloaded = Object.keys(debugJobs) // Object.values is not supported .map(function (key) { return debugJobs[key]; }).filter(function (job) { return job.missing; }); if (odoo.debug || failed.length || unloaded.length) { var log = window.console[!failed.length || !unloaded.length ? 'info' : 'error'].bind(window.console); log((failed.length ? 'error' : (unloaded.length ? 'warning' : 'info')) + ': Some modules could not be started'); if (missing.length) { log('Missing dependencies: ', missing); } if (failed.length) { log('Failed modules: ', failed.map(function (fail) { return fail.name; })); } if (rejected.length) { log('Rejected modules: ', rejected); } if (rejectedLinked.length) { log('Rejected linked modules: ', rejectedLinked); } if (unloaded.length) { log('Non loaded modules: ', unloaded.map(function (unload) { return unload.name; })); } if (odoo.debug && Object.keys(debugJobs).length) { log('Debug: ', debugJobs); } } } odoo.__DEBUG__.jsModules = { missing: missing, failed: failed.map(function (fail) { return fail.name; }), }; didLogInfoResolve(); }; odoo.processJobs = function (jobs, services) { var job; function processJob(job) { var require = makeRequire(job); var jobExec; var def = new Promise(function (resolve) { try { jobExec = job.factory.call(null, require); jobs.splice(jobs.indexOf(job), 1); } catch (e) { job.error = e; console.error('Error while loading ' + job.name + ': '+ e.stack); } if (!job.error) { Promise.resolve(jobExec).then( function (data) { services[job.name] = data; resolve(); odoo.processJobs(jobs, services); }).guardedCatch(function (e) { job.rejected = e || true; jobs.push(job); resolve(); } ); } }); jobPromises.push(def); } function isReady(job) { return !job.error && !job.rejected && job.factory.deps.every(function (name) { return name in services; }); } function makeRequire(job) { var deps = {}; Object.keys(services).filter(function (item) { return job.deps.indexOf(item) >= 0; }).forEach(function (key) { deps[key] = services[key]; }); return function require(name) { if (!(name in deps)) { console.error('Undefined dependency: ', name); } return deps[name]; }; } while (jobs.length) { job = undefined; for (var i = 0; i < jobs.length; i++) { if (isReady(jobs[i])) { job = jobs[i]; break; } } if (!job) { break; } processJob(job); } return services; }; // Automatically log errors detected when loading modules window.addEventListener('load', function logWhenLoaded() { setTimeout(function () { var len = jobPromises.length; Promise.all(jobPromises).then(function () { if (len === jobPromises.length) { odoo.log(); } else { logWhenLoaded(); } }); }, 9999); }); })();