From 91e29d47a06b3f5c7dd48bd19447a639b3b7de4c Mon Sep 17 00:00:00 2001 From: Stas Vilchik Date: Wed, 12 Aug 2015 11:23:22 +0200 Subject: [PATCH] SONAR-6765 SONAR-6766 show multiple issue locations and execution flows --- .../js/apps/issues/component-viewer/main.js | 2 + .../main/js/components/issue/issue-view.js | 7 +- .../js/components/issue/templates/issue.hbs | 13 +++- .../code-with-issue-locations-helper.js | 32 +++++---- .../main/js/components/source-viewer/main.js | 65 ++++++++++++++++++- .../templates/source-viewer-flow-location.hbs | 1 + .../src/main/less/components/source.less | 45 +++++++++++-- .../sonar-web/src/main/less/init/icons.less | 9 +++ .../sonar-web/src/main/less/pages/issues.less | 5 ++ server/sonar-web/src/main/less/variables.less | 2 + .../issues-with-precise-location.json | 12 ++-- server/sonar-web/test/intern.js | 3 +- .../test/medium/source-viewer.spec.js | 44 ++++++++++++- .../code-with-issue-locations-helper.spec.js | 56 ++++++++++++++++ 14 files changed, 265 insertions(+), 31 deletions(-) create mode 100644 server/sonar-web/src/main/js/components/source-viewer/templates/source-viewer-flow-location.hbs create mode 100644 server/sonar-web/test/unit/code-with-issue-locations-helper.spec.js diff --git a/server/sonar-web/src/main/js/apps/issues/component-viewer/main.js b/server/sonar-web/src/main/js/apps/issues/component-viewer/main.js index 1e79d92e45d..8792113e213 100644 --- a/server/sonar-web/src/main/js/apps/issues/component-viewer/main.js +++ b/server/sonar-web/src/main/js/apps/issues/component-viewer/main.js @@ -24,6 +24,7 @@ define([ SourceViewer.prototype.onLoaded.apply(this, arguments); this.bindShortcuts(); if (this.baseIssue != null) { + this.baseIssue.trigger('locations', this.baseIssue); return this.scrollToLine(this.baseIssue.get('line')); } }, @@ -83,6 +84,7 @@ define([ var selected = this.options.app.state.get('selectedIndex'), selectedIssue = this.options.app.list.at(selected); if (selectedIssue.get('component') === this.model.get('key')) { + selectedIssue.trigger('locations', selectedIssue); return this.scrollToIssue(selectedIssue.get('key')); } else { this.unbindShortcuts(); diff --git a/server/sonar-web/src/main/js/components/issue/issue-view.js b/server/sonar-web/src/main/js/components/issue/issue-view.js index 14e83e689e4..aad34a2d04c 100644 --- a/server/sonar-web/src/main/js/components/issue/issue-view.js +++ b/server/sonar-web/src/main/js/components/issue/issue-view.js @@ -37,7 +37,8 @@ define([ 'click .js-issue-plan': 'plan', 'click .js-issue-show-changelog': 'showChangeLog', 'click .js-issue-rule': 'showRule', - 'click .js-issue-edit-tags': 'editTags' + 'click .js-issue-edit-tags': 'editTags', + 'click .js-issue-locations': 'showLocations' }; }, @@ -217,6 +218,10 @@ define([ this.popup.render(); }, + showLocations: function () { + this.model.trigger('locations', this.model); + }, + serializeData: function () { var issueKey = encodeURIComponent(this.model.get('key')); return _.extend(Marionette.ItemView.prototype.serializeData.apply(this, arguments), { diff --git a/server/sonar-web/src/main/js/components/issue/templates/issue.hbs b/server/sonar-web/src/main/js/components/issue/templates/issue.hbs index 930e5ca5693..06e878f7b8f 100644 --- a/server/sonar-web/src/main/js/components/issue/templates/issue.hbs +++ b/server/sonar-web/src/main/js/components/issue/templates/issue.hbs @@ -4,7 +4,8 @@
- {{message}}  + {{message}}  +
@@ -22,6 +23,14 @@ {{/if}} + {{#notEmpty secondaryLocations}} +
  • + +
  • + {{/notEmpty}} +
  • @@ -143,7 +152,7 @@ {{#if updatable}} + data-confirm-msg="{{t 'issue.comment.delete_confirm_message'}}"> {{/if}} diff --git a/server/sonar-web/src/main/js/components/source-viewer/helpers/code-with-issue-locations-helper.js b/server/sonar-web/src/main/js/components/source-viewer/helpers/code-with-issue-locations-helper.js index b95c6efd18c..fd5d56a3bde 100644 --- a/server/sonar-web/src/main/js/components/source-viewer/helpers/code-with-issue-locations-helper.js +++ b/server/sonar-web/src/main/js/components/source-viewer/helpers/code-with-issue-locations-helper.js @@ -22,7 +22,8 @@ define(function () { * @returns {string} */ function part (str, from, to, acc) { - return str.substr(from - acc, to - from); + // we do not want negative number as the first argument of `substr` + return from >= acc ? str.substr(from - acc, to - from) : str.substr(0, to - from); } @@ -53,9 +54,10 @@ define(function () { * Highlight issue locations in the list of tokens * @param {Array} tokens * @param {Array} issueLocations + * @param {string} className * @returns {Array} */ - function highlightIssueLocations (tokens, issueLocations) { + function highlightIssueLocations (tokens, issueLocations, className) { issueLocations.forEach(function (location) { var nextTokens = [], acc = 0; @@ -68,8 +70,8 @@ define(function () { nextTokens.push({ className: token.className, text: p1 }); } if (p2.length) { - var newClassName = token.className.indexOf('source-line-code-issue') === -1 ? - [token.className, 'source-line-code-issue'].join(' ') : token.className; + var newClassName = token.className.indexOf(className) === -1 ? + [token.className, className].join(' ') : token.className; nextTokens.push({ className: newClassName, text: p2 }); } if (p3.length) { @@ -100,20 +102,26 @@ define(function () { * highlight issues and generate result html * @param {string} code * @param {Array} issueLocations + * @param {string} [optionalClassName] * @returns {string} */ - function doTheStuff (code, issueLocations) { + function doTheStuff (code, issueLocations, optionalClassName) { var _code = code || ' '; var _issueLocations = issueLocations || []; - return generateHTML(highlightIssueLocations(splitByTokens(_code), _issueLocations)); + var _className = optionalClassName ? optionalClassName : 'source-line-code-issue'; + return generateHTML(highlightIssueLocations(splitByTokens(_code), _issueLocations, _className)); } - /** - * Handlebars helper to highlight issue locations in the source code - */ - Handlebars.registerHelper('codeWithIssueLocations', function (code, issueLocations) { - return doTheStuff(code, issueLocations); - }); + if (typeof Handlebars !== 'undefined') { + /** + * Handlebars helper to highlight issue locations in the source code + */ + Handlebars.registerHelper('codeWithIssueLocations', function (code, issueLocations) { + return doTheStuff(code, issueLocations); + }); + } + + return doTheStuff; }); diff --git a/server/sonar-web/src/main/js/components/source-viewer/main.js b/server/sonar-web/src/main/js/components/source-viewer/main.js index 828e9e53ffc..a73b56a315f 100644 --- a/server/sonar-web/src/main/js/components/source-viewer/main.js +++ b/server/sonar-web/src/main/js/components/source-viewer/main.js @@ -39,7 +39,8 @@ define([ SCMPopupView, CoveragePopupView, DuplicationPopupView, - LineActionsPopupView) { + LineActionsPopupView, + highlightLocations) { var $ = jQuery, HIGHLIGHTED_ROW_CLASS = 'source-line-highlighted'; @@ -47,6 +48,7 @@ define([ return Marionette.LayoutView.extend({ className: 'source-viewer', template: Templates['source-viewer'], + flowLocationTemplate: Templates['source-viewer-flow-location'], ISSUES_LIMIT: 3000, LINES_LIMIT: 1000, @@ -84,6 +86,7 @@ define([ } this.issues = new Issues(); this.listenTo(this.issues, 'change:severity', this.onIssuesSeverityChange); + this.listenTo(this.issues, 'locations', this.toggleFlowLocations); this.issueViews = []; this.loadSourceBeforeThrottled = _.throttle(this.loadSourceBefore, 1000); this.loadSourceAfterThrottled = _.throttle(this.loadSourceAfter, 1000); @@ -277,8 +280,8 @@ define([ data: { componentUuids: this.model.id, f: 'component,componentId,project,subProject,rule,status,resolution,author,reporter,assignee,debt,' + - 'line,message,severity,actionPlan,creationDate,updateDate,closeDate,tags,comments,attr,actions,' + - 'transitions,actionPlanName', + 'line,message,severity,actionPlan,creationDate,updateDate,closeDate,tags,comments,attr,actions,' + + 'transitions,actionPlanName', additionalFields: '_all', resolved: false, s: 'FILE_LINE', @@ -732,6 +735,62 @@ define([ hideFilteredTooltip: function (e) { $(e.currentTarget).tooltip('destroy'); + }, + + toggleFlowLocations: function (issue) { + if (this.locationsShowFor === issue) { + this.hideFlowLocations(); + } else { + this.hideFlowLocations(); + this.showFlowLocations(issue); + } + }, + + showFlowLocations: function (issue) { + this.locationsShowFor = issue; + var primaryLocation = { + msg: issue.get('message'), + textRange: issue.get('textRange') + }, + _locations = [primaryLocation].concat(issue.get('secondaryLocations')); + _locations.forEach(this.showFlowLocation, this); + }, + + showFlowLocation: function (location) { + if (location && location.textRange) { + var line = location.textRange.startLine, + row = this.$('.source-line-code[data-line-number="' + line + '"]'), + renderedFlowLocation = this.renderFlowLocation(location); + row.append(renderedFlowLocation); + this.highlightFlowLocationInCode(location); + } + }, + + renderFlowLocation: function (location) { + location.msg = location.msg ? location.msg : ' '; + return this.flowLocationTemplate(location); + }, + + highlightFlowLocationInCode: function (location) { + for (var line = location.textRange.startLine; line <= location.textRange.endLine; line++) { + var row = this.$('.source-line-code[data-line-number="' + line + '"]'); + + // get location for the current line + var from = line === location.textRange.startLine ? location.textRange.startOffset : 0, + to = line === location.textRange.endLine ? location.textRange.endOffset : 999999, + _location = { from: from, to: to }; + + // mark issue location in the source code + var code = row.find('pre').html(), + newCode = highlightLocations(code, [_location], 'source-line-code-secondary-issue'); + row.find('pre').html(newCode); + } + }, + + hideFlowLocations: function () { + this.locationsShowFor = null; + this.$('.source-viewer-flow-location').remove(); + this.$('.source-line-code-secondary-issue').removeClass('source-line-code-secondary-issue'); } }); diff --git a/server/sonar-web/src/main/js/components/source-viewer/templates/source-viewer-flow-location.hbs b/server/sonar-web/src/main/js/components/source-viewer/templates/source-viewer-flow-location.hbs new file mode 100644 index 00000000000..e11d2200798 --- /dev/null +++ b/server/sonar-web/src/main/js/components/source-viewer/templates/source-viewer-flow-location.hbs @@ -0,0 +1 @@ +
    {{limitString msg}}
    diff --git a/server/sonar-web/src/main/less/components/source.less b/server/sonar-web/src/main/less/components/source.less index cbe29981233..24f587f1ea5 100644 --- a/server/sonar-web/src/main/less/components/source.less +++ b/server/sonar-web/src/main/less/components/source.less @@ -21,7 +21,7 @@ @import (reference) "../variables"; @import (reference) "ui"; -@lineHeight: 18px; +@source-line-height: 18px; @lineWithIssuesBackground: #ffeaea; @duplicationColor: #f3ca8e; @@ -95,18 +95,19 @@ } .source-viewer pre { - height: @lineHeight; + height: @source-line-height; padding: 0; } .source-viewer pre, .source-meta { - line-height: @lineHeight; + line-height: @source-line-height; font-family: @monoFontFamily; font-size: 12px; } .source-line-code { + position: relative; padding: 0 10px; .issue-list { @@ -123,6 +124,12 @@ background-position: bottom; } +.source-line-code-secondary-issue { + display: inline-block; + background-color: @red; + color: #fff !important; +} + .source-meta { vertical-align: top; width: 1px; @@ -204,7 +211,7 @@ .source-line-bar { width: 5px; - height: @lineHeight; + height: @source-line-height; } .source-line-with-issues { @@ -444,3 +451,33 @@ .source-viewer-test-covered-lines { text-align: right; } + +.source-viewer-flow-location { + position: absolute; + top: 0; + right: 0; + line-height: @source-line-height - 2px; + margin: 1px 0; + padding: 0 10px; + background-color: @red; + color: #fff; + font-size: 12px; + z-index: @issue-flow-location-z-index; + + &:before { + @arrow-size: (@source-line-height - 2px) / 2; + content: " "; + position: absolute; + top: 0; + right: 100%; + display: block; + .square(0); + border-top: @arrow-size solid transparent; + border-bottom: @arrow-size solid transparent; + border-right: @arrow-size solid @red; + } +} + +.source-viewer-flow-location + .source-viewer-flow-location { + z-index: @issue-flow-location-z-index - 1; +} diff --git a/server/sonar-web/src/main/less/init/icons.less b/server/sonar-web/src/main/less/init/icons.less index ca4a8b2b49f..f0d20455d5e 100644 --- a/server/sonar-web/src/main/less/init/icons.less +++ b/server/sonar-web/src/main/less/init/icons.less @@ -536,6 +536,15 @@ a[class^="icon-"], a[class*=" icon-"] { background-image: url(); background-repeat: no-repeat; } +.icon-issue-flow { + position: relative; + top: 1px; + display: inline-block; + vertical-align: top; + .size(14px, 14px); + background-image: url(data:image/svg+xml,%3Csvg%20width%3D%2214%22%20height%3D%2214%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20stroke-linejoin%3D%22round%22%20stroke-miterlimit%3D%221.414%22%3E%3Cpath%20d%3D%22M2.977%2012.656c0%20.417-.142.745-.426.985-.283.24-.636.36-1.058.36-.552%200-1-.172-1.344-.516l.446-.687c.255.234.53.35.828.35.15%200%20.282-.036.394-.112.112-.075.168-.186.168-.332%200-.333-.273-.48-.82-.437l-.203-.438c.043-.052.127-.165.255-.34.127-.174.238-.315.332-.422.094-.106.19-.207.29-.3v-.008c-.084%200-.21.002-.38.008-.17.005-.296.007-.38.007v.415H.25V10h2.602v.688l-.743.898c.265.062.476.19.632.383.156.19.235.42.235.686zm.015-4.898V9H.164c-.03-.188-.047-.328-.047-.422%200-.265.06-.508.184-.726.123-.22.27-.396.442-.532.172-.135.344-.26.516-.37.172-.113.32-.226.44-.34.124-.115.185-.232.185-.352%200-.13-.038-.23-.113-.3-.076-.07-.18-.106-.31-.106-.24%200-.45.15-.632.453l-.664-.46c.125-.267.31-.474.56-.622.246-.15.52-.223.823-.223.38%200%20.7.108.96.324.26.216.39.51.39.88%200%20.26-.087.498-.264.714-.177.216-.373.384-.586.504-.214.12-.41.25-.59.394-.18.144-.272.28-.277.41h.992V7.76h.82zM14%2010.25v1.5c0%20.068-.025.126-.074.176-.05.05-.108.074-.176.074h-9.5c-.068%200-.126-.025-.176-.074-.05-.05-.074-.108-.074-.176v-1.5c0-.073.023-.133.07-.18.047-.047.107-.07.18-.07h9.5c.068%200%20.126.025.176.074.05.05.074.108.074.176zM3%203.227V4H.383v-.773h.836c0-.214%200-.532.003-.954l.004-.945v-.094H1.21c-.04.09-.17.23-.39.422l-.554-.593L1.328.07h.828v3.157H3zM14%206.25v1.5c0%20.068-.025.126-.074.176-.05.05-.108.074-.176.074h-9.5c-.068%200-.126-.025-.176-.074C4.024%207.876%204%207.818%204%207.75v-1.5c0-.073.023-.133.07-.18.047-.047.107-.07.18-.07h9.5c.068%200%20.126.025.176.074.05.05.074.108.074.176zm0-4v1.5c0%20.068-.025.126-.074.176-.05.05-.108.074-.176.074h-9.5c-.068%200-.126-.025-.176-.074C4.024%203.876%204%203.818%204%203.75v-1.5c0-.068.025-.126.074-.176.05-.05.108-.074.176-.074h9.5c.068%200%20.126.025.176.074.05.05.074.108.074.176z%22%20fill%3D%22%23236A97%22%20fill-rule%3D%22nonzero%22%2F%3E%3C%2Fsvg%3E); + background-repeat: no-repeat; +} /* diff --git a/server/sonar-web/src/main/less/pages/issues.less b/server/sonar-web/src/main/less/pages/issues.less index 0f512c1ba6c..cfa2cc22bba 100644 --- a/server/sonar-web/src/main/less/pages/issues.less +++ b/server/sonar-web/src/main/less/pages/issues.less @@ -35,6 +35,11 @@ } } + + .issue-meta-locations { + position: absolute; + visibility: hidden; + } } .issues-workspace-list-component { diff --git a/server/sonar-web/src/main/less/variables.less b/server/sonar-web/src/main/less/variables.less index 44a86acb13f..82366ca71be 100644 --- a/server/sonar-web/src/main/less/variables.less +++ b/server/sonar-web/src/main/less/variables.less @@ -216,6 +216,8 @@ @workspace-nav-z-index: 451; @workspace-viewer-z-index: 450; +@issue-flow-location-z-index: 505; + // ui elements @tooltip-z-index: 8000; @tip-z-index: 8000; diff --git a/server/sonar-web/src/test/json/source-viewer-spec/issues-with-precise-location.json b/server/sonar-web/src/test/json/source-viewer-spec/issues-with-precise-location.json index 706d0868793..abf37c14c8c 100644 --- a/server/sonar-web/src/test/json/source-viewer-spec/issues-with-precise-location.json +++ b/server/sonar-web/src/test/json/source-viewer-spec/issues-with-precise-location.json @@ -68,18 +68,18 @@ "componentId": 1875, "project": "com.sonarsource.samples:multiple-issue-locations", "subProject": "com.sonarsource.samples:multiple-issue-locations", - "line": 11, + "line": 9, "textRange": { - "startLine": 11, - "endLine": 11, + "startLine": 9, + "endLine": 9, "startOffset": 6, - "endOffset": 11 + "endOffset": 10 }, "secondaryLocations": [ { "textRange": { - "startLine": 10, - "endLine": 10, + "startLine": 8, + "endLine": 8, "startOffset": 6, "endOffset": 11 } diff --git a/server/sonar-web/test/intern.js b/server/sonar-web/test/intern.js index 42049ab87db..e17f7661681 100644 --- a/server/sonar-web/test/intern.js +++ b/server/sonar-web/test/intern.js @@ -17,7 +17,8 @@ define(['intern'], function (intern) { suites: [ 'test/unit/application.spec', 'test/unit/issue.spec', - 'test/unit/overview/card.spec' + 'test/unit/overview/card.spec', + 'test/unit/code-with-issue-locations-helper.spec' ], functionalSuites: [ diff --git a/server/sonar-web/test/medium/source-viewer.spec.js b/server/sonar-web/test/medium/source-viewer.spec.js index 27a5b96057f..95d33dbab18 100644 --- a/server/sonar-web/test/medium/source-viewer.spec.js +++ b/server/sonar-web/test/medium/source-viewer.spec.js @@ -19,8 +19,8 @@ define(function (require) { .checkElementExist('.source-line-code[data-line-number="3"] .source-line-code-issue') .checkElementInclude('.source-line-code[data-line-number="3"] .source-line-code-issue', '14 So') - .checkElementExist('.source-line-code[data-line-number="11"] .source-line-code-issue') - .checkElementInclude('.source-line-code[data-line-number="11"] .source-line-code-issue', 'arQub') + .checkElementExist('.source-line-code[data-line-number="9"] .source-line-code-issue') + .checkElementInclude('.source-line-code[data-line-number="9"] .source-line-code-issue', 'sion') .checkElementExist('.source-line-code[data-line-number="18"] .source-line-code-issue') .checkElementInclude('.source-line-code[data-line-number="18"] .source-line-code-issue', @@ -28,6 +28,46 @@ define(function (require) { .checkElementExist('.source-line-code[data-line-number="19"] .source-line-code-issue') .checkElementInclude('.source-line-code[data-line-number="19"] .source-line-code-issue', ' */'); }); + + bdd.it('should show secondary issue locations on the same line', function () { + return this.remote + .open() + .mockFromFile('/api/components/app', 'source-viewer-spec/app.json', { data: { uuid: 'uuid' } }) + .mockFromFile('/api/sources/lines', 'source-viewer-spec/lines.json', { data: { uuid: 'uuid' } }) + .mockFromFile('/api/issues/search', + 'source-viewer-spec/issues-with-precise-location.json', + { data: { componentUuids: 'uuid' } }) + .startApp('source-viewer', { file: file }) + .checkElementExist('.source-line-code[data-line-number="3"] .source-line-code-issue') + .checkElementInclude('.source-line-code[data-line-number="3"] .source-line-code-issue', '14 So') + .clickElement('.source-line-with-issues[data-line-number="3"]') + .clickElement('.js-issue-locations') + .checkElementExist('.source-line-code[data-line-number="3"] .source-viewer-flow-location') + .checkElementCount('.source-line-code[data-line-number="3"] .source-line-code-secondary-issue', 2) + .checkElementInclude('.source-line-code[data-line-number="3"] .source-line-code-secondary-issue', ') 200') + .checkElementInclude('.source-line-code[data-line-number="3"] .source-line-code-secondary-issue', '14 So'); + }); + + bdd.it('should show secondary issue locations on the different lines', function () { + return this.remote + .open() + .mockFromFile('/api/components/app', 'source-viewer-spec/app.json', { data: { uuid: 'uuid' } }) + .mockFromFile('/api/sources/lines', 'source-viewer-spec/lines.json', { data: { uuid: 'uuid' } }) + .mockFromFile('/api/issues/search', + 'source-viewer-spec/issues-with-precise-location.json', + { data: { componentUuids: 'uuid' } }) + .startApp('source-viewer', { file: file }) + .checkElementExist('.source-line-code[data-line-number="9"] .source-line-code-issue') + .checkElementInclude('.source-line-code[data-line-number="9"] .source-line-code-issue', 'sion') + .clickElement('.source-line-with-issues[data-line-number="9"]') + .clickElement('.js-issue-locations') + .checkElementExist('.source-line-code[data-line-number="8"] .source-viewer-flow-location') + .checkElementExist('.source-line-code[data-line-number="9"] .source-viewer-flow-location') + .checkElementCount('.source-line-code[data-line-number="8"] .source-line-code-secondary-issue', 1) + .checkElementCount('.source-line-code[data-line-number="9"] .source-line-code-secondary-issue', 1) + .checkElementInclude('.source-line-code[data-line-number="8"] .source-line-code-secondary-issue', 'ense ') + .checkElementInclude('.source-line-code[data-line-number="9"] .source-line-code-secondary-issue', 'sion'); + }); }); }); }); diff --git a/server/sonar-web/test/unit/code-with-issue-locations-helper.spec.js b/server/sonar-web/test/unit/code-with-issue-locations-helper.spec.js new file mode 100644 index 00000000000..a1fa2cb0a1e --- /dev/null +++ b/server/sonar-web/test/unit/code-with-issue-locations-helper.spec.js @@ -0,0 +1,56 @@ +define(function (require) { + var bdd = require('intern!bdd'); + var assert = require('intern/chai!assert'); + + var helper = require('build/js/components/source-viewer/helpers/code-with-issue-locations-helper'); + + bdd.describe('Code With Issue Locations Helper', function () { + bdd.it('should exist', function () { + assert.equal(typeof helper, 'function'); + }); + + bdd.it('should mark one location', function () { + var code = 'if (a + 1) {', + locations = [{ from: 1, to: 5 }], + result = helper(code, locations, 'x'); + assert.equal(result, + 'if (a + 1) {'); + }); + + bdd.it('should mark two locations', function () { + var code = 'abcdefghijklmnopqrst', + locations = [ + { from: 1, to: 6 }, + { from: 11, to: 16 } + ], + result = helper(code, locations, 'x'); + assert.equal(result, + ['a', + 'bcdef', + 'ghijk', + 'lmnop', + 'qrst'].join('')); + }); + + bdd.it('should mark one locations', function () { + var code = ' * Copyright (C) 2008-2014 SonarSource', + locations = [{ from: 15, to: 20 }], + result = helper(code, locations, 'x'); + assert.equal(result, + ' * Copyright (C) 2008-2014 SonarSource'); + }); + + bdd.it('should mark two locations', function () { + var code = ' * Copyright (C) 2008-2014 SonarSource', + locations = [ + { from: 24, to: 29 }, + { from: 15, to: 20 } + ], + result = helper(code, locations, 'x'); + assert.equal(result, + ' * Copyright (C) 2008-2014 SonarSource'); + // * Copyright (C) 2008-204 SonarSource + }); + }); +}); + -- 2.39.5