|  | // Copyright (c) 2017, Compiler Explorer Authors | 
|  | // All rights reserved. | 
|  | // | 
|  | // Redistribution and use in source and binary forms, with or without | 
|  | // modification, are permitted provided that the following conditions are met: | 
|  | // | 
|  | //     * Redistributions of source code must retain the above copyright notice, | 
|  | //       this list of conditions and the following disclaimer. | 
|  | //     * Redistributions in binary form must reproduce the above copyright | 
|  | //       notice, this list of conditions and the following disclaimer in the | 
|  | //       documentation and/or other materials provided with the distribution. | 
|  | // | 
|  | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
|  | // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
|  | // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | 
|  | // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE | 
|  | // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | 
|  | // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | 
|  | // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | 
|  | // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | 
|  | // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | 
|  | // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | 
|  | // POSSIBILITY OF SUCH DAMAGE. | 
|  |  | 
|  | 'use strict'; | 
|  | var Sentry = require('@sentry/browser'); | 
|  | var $ = require('jquery'); | 
|  | var _ = require('underscore'); | 
|  | var LruCache = require('lru-cache'); | 
|  | var options = require('./options').options; | 
|  | var Promise = require('es6-promise').Promise; | 
|  |  | 
|  | function CompilerService(eventHub) { | 
|  | this.base = window.httpRoot; | 
|  | this.allowStoreCodeDebug = true; | 
|  | this.cache = new LruCache({ | 
|  | max: 200 * 1024, | 
|  | length: function (n) { | 
|  | return JSON.stringify(n).length; | 
|  | }, | 
|  | }); | 
|  | this.compilersByLang = {}; | 
|  | _.each( | 
|  | options.compilers, | 
|  | function (compiler) { | 
|  | if (!this.compilersByLang[compiler.lang]) this.compilersByLang[compiler.lang] = {}; | 
|  | this.compilersByLang[compiler.lang][compiler.id] = compiler; | 
|  | }, | 
|  | this | 
|  | ); | 
|  | // settingsChange is triggered on page load | 
|  | eventHub.on( | 
|  | 'settingsChange', | 
|  | function (newSettings) { | 
|  | this.allowStoreCodeDebug = newSettings.allowStoreCodeDebug; | 
|  | }, | 
|  | this | 
|  | ); | 
|  | } | 
|  |  | 
|  | CompilerService.prototype.getDefaultCompilerForLang = function (langId) { | 
|  | return options.defaultCompiler[langId]; | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.processFromLangAndCompiler = function (languageId, compilerId) { | 
|  | var langId = languageId; | 
|  | var foundCompiler = null; | 
|  | try { | 
|  | if (langId) { | 
|  | if (!compilerId) { | 
|  | compilerId = this.getDefaultCompilerForLang(langId); | 
|  | } | 
|  |  | 
|  | foundCompiler = this.findCompiler(langId, compilerId); | 
|  | if (!foundCompiler) { | 
|  | var compilers = this.getCompilersForLang(langId); | 
|  | foundCompiler = compilers[_.first(_.keys(compilers))]; | 
|  | } | 
|  | } else if (compilerId) { | 
|  | var matchingCompilers = _.map( | 
|  | options.languages, | 
|  | function (lang) { | 
|  | var compiler = this.findCompiler(lang.id, compilerId); | 
|  | if (compiler) { | 
|  | return { | 
|  | langId: lang.id, | 
|  | compiler: compiler, | 
|  | }; | 
|  | } | 
|  | return null; | 
|  | }, | 
|  | this | 
|  | ); | 
|  |  | 
|  | return _.find(matchingCompilers, function (match) { | 
|  | return match !== null; | 
|  | }); | 
|  | } else { | 
|  | var firstLang = _.first(_.values(options.languages)); | 
|  | if (firstLang) { | 
|  | return this.processFromLangAndCompiler(firstLang.id, compilerId); | 
|  | } | 
|  | } | 
|  | } catch (e) { | 
|  | Sentry.captureException(e); | 
|  | } | 
|  |  | 
|  | return { | 
|  | langId: langId, | 
|  | compiler: foundCompiler, | 
|  | }; | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.getGroupsInUse = function (langId) { | 
|  | return _.chain(this.getCompilersForLang(langId)) | 
|  | .map() | 
|  | .uniq(false, function (compiler) { | 
|  | return compiler.group; | 
|  | }) | 
|  | .map(function (compiler) { | 
|  | return {value: compiler.group, label: compiler.groupName || compiler.group}; | 
|  | }) | 
|  | .sort(function (a, b) { | 
|  | return a.label.localeCompare(b.label, undefined /* Ignore language */, {sensitivity: 'base'}); | 
|  | }) | 
|  | .value(); | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.getCompilersForLang = function (langId) { | 
|  | return this.compilersByLang[langId] || {}; | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.findCompilerInList = function (compilers, compilerId) { | 
|  | if (compilers && compilers[compilerId]) { | 
|  | return compilers[compilerId]; | 
|  | } | 
|  | return _.find(compilers, function (compiler) { | 
|  | return compiler.alias.includes(compilerId); | 
|  | }); | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.findCompiler = function (langId, compilerId) { | 
|  | if (!compilerId) return null; | 
|  | var compilers = this.getCompilersForLang(langId); | 
|  | return this.findCompilerInList(compilers, compilerId); | 
|  | }; | 
|  |  | 
|  | function handleRequestError(request, reject, xhr, textStatus, errorThrown) { | 
|  | var error = errorThrown; | 
|  | if (!error) { | 
|  | switch (textStatus) { | 
|  | case 'timeout': | 
|  | error = 'Request timed out'; | 
|  | break; | 
|  | case 'abort': | 
|  | error = 'Request was aborted'; | 
|  | break; | 
|  | case 'error': | 
|  | switch (xhr.status) { | 
|  | case 500: | 
|  | error = 'Request failed: internal server error'; | 
|  | break; | 
|  | case 504: | 
|  | error = 'Request failed: gateway timeout'; | 
|  | break; | 
|  | default: | 
|  | error = 'Request failed: HTTP error code ' + xhr.status; | 
|  | break; | 
|  | } | 
|  | break; | 
|  | default: | 
|  | error = 'Error sending request'; | 
|  | break; | 
|  | } | 
|  | } | 
|  | reject({ | 
|  | request: request, | 
|  | error: error, | 
|  | }); | 
|  | } | 
|  |  | 
|  | CompilerService.prototype.submit = function (request) { | 
|  | request.allowStoreCodeDebug = this.allowStoreCodeDebug; | 
|  | var jsonRequest = JSON.stringify(request); | 
|  | if (options.doCache) { | 
|  | var cachedResult = this.cache.get(jsonRequest); | 
|  | if (cachedResult) { | 
|  | return Promise.resolve({ | 
|  | request: request, | 
|  | result: cachedResult, | 
|  | localCacheHit: true, | 
|  | }); | 
|  | } | 
|  | } | 
|  | return new Promise( | 
|  | _.bind(function (resolve, reject) { | 
|  | var bindHandler = _.partial(handleRequestError, request, reject); | 
|  | var compilerId = encodeURIComponent(request.compiler); | 
|  | $.ajax({ | 
|  | type: 'POST', | 
|  | url: window.location.origin + this.base + 'api/compiler/' + compilerId + '/compile', | 
|  | dataType: 'json', | 
|  | contentType: 'application/json', | 
|  | data: jsonRequest, | 
|  | success: _.bind(function (result) { | 
|  | if (result && result.okToCache && options.doCache) { | 
|  | this.cache.set(jsonRequest, result); | 
|  | } | 
|  | resolve({ | 
|  | request: request, | 
|  | result: result, | 
|  | localCacheHit: false, | 
|  | }); | 
|  | }, this), | 
|  | error: bindHandler, | 
|  | }); | 
|  | }, this) | 
|  | ); | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.submitCMake = function (request) { | 
|  | request.allowStoreCodeDebug = this.allowStoreCodeDebug; | 
|  | var jsonRequest = JSON.stringify(request); | 
|  | if (options.doCache) { | 
|  | var cachedResult = this.cache.get(jsonRequest); | 
|  | if (cachedResult) { | 
|  | return Promise.resolve({ | 
|  | request: request, | 
|  | result: cachedResult, | 
|  | localCacheHit: true, | 
|  | }); | 
|  | } | 
|  | } | 
|  | return new Promise( | 
|  | _.bind(function (resolve, reject) { | 
|  | var bindHandler = _.partial(handleRequestError, request, reject); | 
|  | var compilerId = encodeURIComponent(request.compiler); | 
|  | $.ajax({ | 
|  | type: 'POST', | 
|  | url: window.location.origin + this.base + 'api/compiler/' + compilerId + '/cmake', | 
|  | dataType: 'json', | 
|  | contentType: 'application/json', | 
|  | data: jsonRequest, | 
|  | success: _.bind(function (result) { | 
|  | if (result && result.okToCache && options.doCache) { | 
|  | this.cache.set(jsonRequest, result); | 
|  | } | 
|  | resolve({ | 
|  | request: request, | 
|  | result: result, | 
|  | localCacheHit: false, | 
|  | }); | 
|  | }, this), | 
|  | error: bindHandler, | 
|  | }); | 
|  | }, this) | 
|  | ); | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.requestPopularArguments = function (compilerId, options) { | 
|  | return new Promise( | 
|  | _.bind(function (resolve, reject) { | 
|  | var bindHandler = _.partial(handleRequestError, compilerId, reject); | 
|  | $.ajax({ | 
|  | type: 'POST', | 
|  | url: window.location.origin + this.base + 'api/popularArguments/' + compilerId, | 
|  | dataType: 'json', | 
|  | data: JSON.stringify({ | 
|  | usedOptions: options, | 
|  | presplit: false, | 
|  | }), | 
|  | success: _.bind(function (result) { | 
|  | resolve({ | 
|  | request: compilerId, | 
|  | result: result, | 
|  | localCacheHit: false, | 
|  | }); | 
|  | }, this), | 
|  | error: bindHandler, | 
|  | }); | 
|  | }, this) | 
|  | ); | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.expand = function (source) { | 
|  | var includeFind = /^\s*#\s*include\s*["<](https?:\/\/[^>"]+)[>"]/; | 
|  | var lines = source.split('\n'); | 
|  | var promises = []; | 
|  | _.each(lines, function (line, lineNumZeroBased) { | 
|  | var match = line.match(includeFind); | 
|  | if (match) { | 
|  | promises.push( | 
|  | new Promise(function (resolve) { | 
|  | var req = $.get(match[1], function (data) { | 
|  | data = | 
|  | '#line 1 "' + | 
|  | match[1] + | 
|  | '"\n' + | 
|  | data + | 
|  | '\n\n#line ' + | 
|  | (lineNumZeroBased + 1) + | 
|  | ' "<stdin>"\n'; | 
|  |  | 
|  | lines[lineNumZeroBased] = data; | 
|  | resolve(); | 
|  | }); | 
|  | req.fail(function () { | 
|  | resolve(); | 
|  | }); | 
|  | }) | 
|  | ); | 
|  | } | 
|  | }); | 
|  | return Promise.all(promises).then(function () { | 
|  | return lines.join('\n'); | 
|  | }); | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.getSelectizerOrder = function () { | 
|  | return [{field: '$order'}, {field: '$score'}, {field: 'name'}]; | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.doesCompilationResultHaveWarnings = function (result) { | 
|  | var stdout = result.stdout || []; | 
|  | var stderr = result.stderr || []; | 
|  | // TODO: Pass what compiler did this and check if it it's actually skippable | 
|  | // Right now we're ignoring outputs that match the input filename | 
|  | // Compiler & Executor are capable of giving us the info, but conformance view is not | 
|  | if (stdout.length === 1 && stderr.length === 0) { | 
|  | // We could also move this calculation to the server at some point | 
|  | var lastSlashPos = _.findLastIndex(result.inputFilename, function (ch) { | 
|  | return ch === '\\'; | 
|  | }); | 
|  | return result.inputFilename.substr(lastSlashPos + 1) !== stdout[0].text; | 
|  | } | 
|  | return stdout.length > 0 || stderr.length > 0; | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.calculateStatusIcon = function (result) { | 
|  | var code = 1; | 
|  | if (result.code !== 0) { | 
|  | code = 3; | 
|  | } else if (this.doesCompilationResultHaveWarnings(result)) { | 
|  | code = 2; | 
|  | } | 
|  | return {code: code, compilerOut: result.code}; | 
|  | }; | 
|  |  | 
|  | function ariaLabel(status) { | 
|  | // Compiling... | 
|  | if (status.code === 4) return 'Compiling'; | 
|  | if (status.compilerOut === 0) { | 
|  | // StdErr.length > 0 | 
|  | if (status.code === 3) return 'Compilation succeeded with errors'; | 
|  | // StdOut.length > 0 | 
|  | if (status.code === 2) return 'Compilation succeeded with warnings'; | 
|  | return 'Compilation succeeded'; | 
|  | } else { | 
|  | // StdErr.length > 0 | 
|  | if (status.code === 3) return 'Compilation failed with errors'; | 
|  | // StdOut.length > 0 | 
|  | if (status.code === 2) return 'Compilation failed with warnings'; | 
|  | return 'Compilation failed'; | 
|  | } | 
|  | } | 
|  |  | 
|  | function color(status) { | 
|  | // Compiling... | 
|  | if (status.code === 4) return '#888888'; | 
|  | if (status.compilerOut === 0) { | 
|  | // StdErr.length > 0 | 
|  | if (status.code === 3) return '#FF6645'; | 
|  | // StdOut.length > 0 | 
|  | if (status.code === 2) return '#FF6500'; | 
|  | return '#12BB12'; | 
|  | } else { | 
|  | // StdErr.length > 0 | 
|  | if (status.code === 3) return '#FF1212'; | 
|  | // StdOut.length > 0 | 
|  | if (status.code === 2) return '#BB8700'; | 
|  | return '#FF6645'; | 
|  | } | 
|  | } | 
|  |  | 
|  | CompilerService.prototype.handleCompilationStatus = function (statusLabel, statusIcon, status) { | 
|  | if (statusLabel != null) { | 
|  | statusLabel.toggleClass('error', status.code === 3).toggleClass('warning', status.code === 2); | 
|  | } | 
|  |  | 
|  | if (statusIcon != null) { | 
|  | statusIcon | 
|  | .removeClass() | 
|  | .addClass('status-icon fas') | 
|  | .css('color', color(status)) | 
|  | .toggle(status.code !== 0) | 
|  | .prop('aria-label', ariaLabel(status)) | 
|  | .prop('data-status', status.code) | 
|  | .toggleClass('fa-spinner fa-spin', status.code === 4) | 
|  | .toggleClass('fa-times-circle', status.code === 3) | 
|  | .toggleClass('fa-check-circle', status.code === 1 || status.code === 2); | 
|  | } | 
|  | }; | 
|  |  | 
|  | CompilerService.prototype.handleOutputButtonTitle = function (element, result) { | 
|  | var stdout = result.stdout || []; | 
|  | var stderr = result.stderr || []; | 
|  |  | 
|  | var asciiColorsRe = RegExp(/\x1b\[[0-9;]*m(.\[K)?/g); | 
|  |  | 
|  | function filterAsciiColors(line) { | 
|  | return line.text.replace(asciiColorsRe, ''); | 
|  | } | 
|  |  | 
|  | var output = _.map(stdout, filterAsciiColors).concat(_.map(stderr, filterAsciiColors)).join('\n'); | 
|  |  | 
|  | element.prop('title', output); | 
|  | }; | 
|  |  | 
|  | module.exports = { | 
|  | CompilerService: CompilerService, | 
|  | }; |