From c252c03401881fd7dbf7fab984285c402eb31d5f Mon Sep 17 00:00:00 2001
From: Luke Bennett <lukeeeebennettplus@gmail.com>
Date: Sun, 9 Oct 2016 23:40:58 +0100
Subject: [PATCH 001/363] Added raven and raven-vue plugin, updated gon_helper
 with data needed for raven and created raven_config, required by
 application.js

Added is_production to define sentry environment

Removed as much jQuery as possible

Added public_sentry_dsn application_settings helper method

Use URI module instead of regex for public dsn

Removed raven-vue and load raven on if sentry is enabled

Add load_script spec

added raven_config spec

added class_spec_helper and tests

added sentry_helper spec

added feature spec
---
 app/assets/javascripts/application.js         |    1 +
 .../javascripts/lib/utils/load_script.js.es6  |   26 +
 app/assets/javascripts/raven_config.js.es6    |   66 +
 app/helpers/sentry_helper.rb                  |    8 +
 config/application.rb                         |    1 +
 lib/gitlab/gon_helper.rb                      |    6 +
 spec/features/raven_js_spec.rb                |   24 +
 spec/helpers/sentry_helper_spec.rb            |   15 +
 spec/javascripts/class_spec_helper.js.es6     |   10 +
 .../javascripts/class_spec_helper_spec.js.es6 |   35 +
 .../lib/utils/load_script_spec.js.es6         |   95 +
 spec/javascripts/raven_config_spec.js.es6     |  142 +
 spec/javascripts/spec_helper.js               |    1 +
 vendor/assets/javascripts/raven.js            | 2547 +++++++++++++++++
 14 files changed, 2977 insertions(+)
 create mode 100644 app/assets/javascripts/lib/utils/load_script.js.es6
 create mode 100644 app/assets/javascripts/raven_config.js.es6
 create mode 100644 spec/features/raven_js_spec.rb
 create mode 100644 spec/helpers/sentry_helper_spec.rb
 create mode 100644 spec/javascripts/class_spec_helper.js.es6
 create mode 100644 spec/javascripts/class_spec_helper_spec.js.es6
 create mode 100644 spec/javascripts/lib/utils/load_script_spec.js.es6
 create mode 100644 spec/javascripts/raven_config_spec.js.es6
 create mode 100644 vendor/assets/javascripts/raven.js

diff --git a/app/assets/javascripts/application.js b/app/assets/javascripts/application.js
index f0615481ed2cd..94902e560a8b4 100644
--- a/app/assets/javascripts/application.js
+++ b/app/assets/javascripts/application.js
@@ -62,6 +62,7 @@
 /*= require_directory . */
 /*= require fuzzaldrin-plus */
 /*= require es6-promise.auto */
+/*= require raven_config */
 
 (function () {
   document.addEventListener('page:fetch', function () {
diff --git a/app/assets/javascripts/lib/utils/load_script.js.es6 b/app/assets/javascripts/lib/utils/load_script.js.es6
new file mode 100644
index 0000000000000..351d96530eddf
--- /dev/null
+++ b/app/assets/javascripts/lib/utils/load_script.js.es6
@@ -0,0 +1,26 @@
+(() => {
+  const global = window.gl || (window.gl = {});
+
+  class LoadScript {
+    static load(source, id = '') {
+      if (!source) return Promise.reject('source url must be defined');
+      if (id && document.querySelector(`#${id}`)) return Promise.reject('script id already exists');
+      return new Promise((resolve, reject) => this.appendScript(source, id, resolve, reject));
+    }
+
+    static appendScript(source, id, resolve, reject) {
+      const scriptElement = document.createElement('script');
+      scriptElement.type = 'text/javascript';
+      if (id) scriptElement.id = id;
+      scriptElement.onload = resolve;
+      scriptElement.onerror = reject;
+      scriptElement.src = source;
+
+      document.body.appendChild(scriptElement);
+    }
+  }
+
+  global.LoadScript = LoadScript;
+
+  return global.LoadScript;
+})();
diff --git a/app/assets/javascripts/raven_config.js.es6 b/app/assets/javascripts/raven_config.js.es6
new file mode 100644
index 0000000000000..e15eeb9f9cd75
--- /dev/null
+++ b/app/assets/javascripts/raven_config.js.es6
@@ -0,0 +1,66 @@
+/* global Raven */
+
+/*= require lib/utils/load_script */
+
+(() => {
+  const global = window.gl || (window.gl = {});
+
+  class RavenConfig {
+    static init(options = {}) {
+      this.options = options;
+      if (!this.options.sentryDsn || !this.options.ravenAssetUrl) return Promise.reject('sentry dsn and raven asset url is required');
+      return global.LoadScript.load(this.options.ravenAssetUrl, 'raven-js')
+        .then(() => {
+          this.configure();
+          this.bindRavenErrors();
+          if (this.options.currentUserId) this.setUser();
+        });
+    }
+
+    static configure() {
+      Raven.config(this.options.sentryDsn, {
+        whitelistUrls: this.options.whitelistUrls,
+        environment: this.options.isProduction ? 'production' : 'development',
+      }).install();
+    }
+
+    static setUser() {
+      Raven.setUserContext({
+        id: this.options.currentUserId,
+      });
+    }
+
+    static bindRavenErrors() {
+      $(document).on('ajaxError.raven', this.handleRavenErrors);
+    }
+
+    static handleRavenErrors(event, req, config, err) {
+      const error = err || req.statusText;
+      Raven.captureMessage(error, {
+        extra: {
+          type: config.type,
+          url: config.url,
+          data: config.data,
+          status: req.status,
+          response: req.responseText.substring(0, 100),
+          error,
+          event,
+        },
+      });
+    }
+  }
+
+  global.RavenConfig = RavenConfig;
+
+  document.addEventListener('DOMContentLoaded', () => {
+    if (!window.gon) return;
+
+    global.RavenConfig.init({
+      sentryDsn: gon.sentry_dsn,
+      ravenAssetUrl: gon.raven_asset_url,
+      currentUserId: gon.current_user_id,
+      whitelistUrls: [gon.gitlab_url],
+      isProduction: gon.is_production,
+    }).catch($.noop);
+  });
+})();
diff --git a/app/helpers/sentry_helper.rb b/app/helpers/sentry_helper.rb
index 3d255df66a0c3..19de38ac52db6 100644
--- a/app/helpers/sentry_helper.rb
+++ b/app/helpers/sentry_helper.rb
@@ -6,4 +6,12 @@ def sentry_enabled?
   def sentry_context
     Gitlab::Sentry.context(current_user)
   end
+
+  def sentry_dsn_public
+    sentry_dsn = ApplicationSetting.current.sentry_dsn
+    return unless sentry_dsn
+    uri = URI.parse(sentry_dsn)
+    uri.password = nil
+    uri.to_s
+  end
 end
diff --git a/config/application.rb b/config/application.rb
index f00e58a36ca89..8af6eccf3fe0c 100644
--- a/config/application.rb
+++ b/config/application.rb
@@ -92,6 +92,7 @@ class Application < Rails::Application
     config.assets.precompile << "katex.css"
     config.assets.precompile << "katex.js"
     config.assets.precompile << "xterm/xterm.css"
+    config.assets.precompile << "raven.js"
     config.assets.precompile << "graphs/graphs_bundle.js"
     config.assets.precompile << "users/users_bundle.js"
     config.assets.precompile << "network/network_bundle.js"
diff --git a/lib/gitlab/gon_helper.rb b/lib/gitlab/gon_helper.rb
index b8a5ac907a44f..70ffb68c9abc7 100644
--- a/lib/gitlab/gon_helper.rb
+++ b/lib/gitlab/gon_helper.rb
@@ -1,3 +1,5 @@
+include SentryHelper
+
 module Gitlab
   module GonHelper
     def add_gon_variables
@@ -10,6 +12,10 @@ def add_gon_variables
       gon.award_menu_url         = emojis_path
       gon.katex_css_url          = ActionController::Base.helpers.asset_path('katex.css')
       gon.katex_js_url           = ActionController::Base.helpers.asset_path('katex.js')
+      gon.sentry_dsn             = sentry_dsn_public if sentry_enabled?
+      gon.raven_asset_url        = ActionController::Base.helpers.asset_path('raven.js') if sentry_enabled?
+      gon.gitlab_url             = Gitlab.config.gitlab.url
+      gon.is_production          = Rails.env.production?
 
       if current_user
         gon.current_user_id = current_user.id
diff --git a/spec/features/raven_js_spec.rb b/spec/features/raven_js_spec.rb
new file mode 100644
index 0000000000000..e64da1e3a974f
--- /dev/null
+++ b/spec/features/raven_js_spec.rb
@@ -0,0 +1,24 @@
+require 'spec_helper'
+
+feature 'RavenJS', feature: true, js: true do
+  let(:raven_path) { '/raven.js' }
+
+  it 'should not load raven if sentry is disabled' do
+    visit new_user_session_path
+
+    expect(has_requested_raven).to eq(false)
+  end
+
+  it 'should load raven if sentry is enabled' do
+    allow_any_instance_of(ApplicationController).to receive_messages(sentry_dsn_public: 'https://mock:sentry@dsn/path',
+                                                                     sentry_enabled?: true)
+
+    visit new_user_session_path
+
+    expect(has_requested_raven).to eq(true)
+  end
+
+  def has_requested_raven
+    page.driver.network_traffic.one? {|request| request.url.end_with?(raven_path)}
+  end
+end
diff --git a/spec/helpers/sentry_helper_spec.rb b/spec/helpers/sentry_helper_spec.rb
new file mode 100644
index 0000000000000..35ecf9355e165
--- /dev/null
+++ b/spec/helpers/sentry_helper_spec.rb
@@ -0,0 +1,15 @@
+require 'spec_helper'
+
+describe SentryHelper do
+  describe '#sentry_dsn_public' do
+    it 'returns nil if no sentry_dsn is set' do
+      allow(ApplicationSetting.current).to receive(:sentry_dsn).and_return(nil)
+      expect(helper.sentry_dsn_public).to eq(nil)
+    end
+
+    it 'returns the uri string with no password if sentry_dsn is set' do
+      allow(ApplicationSetting.current).to receive(:sentry_dsn).and_return('https://test:dsn@host/path')
+      expect(helper.sentry_dsn_public).to eq('https://test@host/path')
+    end
+  end
+end
diff --git a/spec/javascripts/class_spec_helper.js.es6 b/spec/javascripts/class_spec_helper.js.es6
new file mode 100644
index 0000000000000..3a04e170924d5
--- /dev/null
+++ b/spec/javascripts/class_spec_helper.js.es6
@@ -0,0 +1,10 @@
+/* eslint-disable no-unused-vars */
+
+class ClassSpecHelper {
+  static itShouldBeAStaticMethod(base, method) {
+    return it('should be a static method', () => {
+      expect(base[method]).toBeDefined();
+      expect(base.prototype[method]).toBeUndefined();
+    });
+  }
+}
diff --git a/spec/javascripts/class_spec_helper_spec.js.es6 b/spec/javascripts/class_spec_helper_spec.js.es6
new file mode 100644
index 0000000000000..d1155f1bd1edd
--- /dev/null
+++ b/spec/javascripts/class_spec_helper_spec.js.es6
@@ -0,0 +1,35 @@
+/* global ClassSpecHelper */
+//= require ./class_spec_helper
+
+describe('ClassSpecHelper', () => {
+  describe('.itShouldBeAStaticMethod', function () {
+    beforeEach(() => {
+      class TestClass {
+        instanceMethod() { this.prop = 'val'; }
+        static staticMethod() {}
+      }
+
+      this.TestClass = TestClass;
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(ClassSpecHelper, 'itShouldBeAStaticMethod');
+
+    it('should have a defined spec', () => {
+      expect(ClassSpecHelper.itShouldBeAStaticMethod(this.TestClass, 'staticMethod').description).toBe('should be a static method');
+    });
+
+    it('should pass for a static method', () => {
+      const spec = ClassSpecHelper.itShouldBeAStaticMethod(this.TestClass, 'staticMethod');
+      expect(spec.status()).toBe('passed');
+    });
+
+    it('should fail for an instance method', (done) => {
+      const spec = ClassSpecHelper.itShouldBeAStaticMethod(this.TestClass, 'instanceMethod');
+      spec.resultCallback = (result) => {
+        expect(result.status).toBe('failed');
+        done();
+      };
+      spec.execute();
+    });
+  });
+});
diff --git a/spec/javascripts/lib/utils/load_script_spec.js.es6 b/spec/javascripts/lib/utils/load_script_spec.js.es6
new file mode 100644
index 0000000000000..52c53327695c7
--- /dev/null
+++ b/spec/javascripts/lib/utils/load_script_spec.js.es6
@@ -0,0 +1,95 @@
+/* global ClassSpecHelper */
+
+/*= require lib/utils/load_script */
+/*= require class_spec_helper */
+
+describe('LoadScript', () => {
+  const global = window.gl || (window.gl = {});
+  const LoadScript = global.LoadScript;
+
+  it('should be defined in the global scope', () => {
+    expect(LoadScript).toBeDefined();
+  });
+
+  describe('.load', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(LoadScript, 'load');
+
+    it('should reject if no source argument is provided', () => {
+      spyOn(Promise, 'reject');
+      LoadScript.load();
+      expect(Promise.reject).toHaveBeenCalledWith('source url must be defined');
+    });
+
+    it('should reject if the script id already exists', () => {
+      spyOn(Promise, 'reject');
+      spyOn(document, 'querySelector').and.returnValue({});
+      LoadScript.load('src.js', 'src-id');
+
+      expect(Promise.reject).toHaveBeenCalledWith('script id already exists');
+    });
+
+    it('should return a promise on completion', () => {
+      expect(LoadScript.load('src.js')).toEqual(jasmine.any(Promise));
+    });
+
+    it('should call appendScript when the promise is constructed', () => {
+      spyOn(LoadScript, 'appendScript');
+      LoadScript.load('src.js', 'src-id');
+
+      expect(LoadScript.appendScript).toHaveBeenCalledWith('src.js', 'src-id', jasmine.any(Promise.resolve.constructor), jasmine.any(Promise.reject.constructor));
+    });
+  });
+
+  describe('.appendScript', () => {
+    beforeEach(() => {
+      spyOn(document.body, 'appendChild');
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(LoadScript, 'appendScript');
+
+    describe('when called', () => {
+      let mockScriptTag;
+
+      beforeEach(() => {
+        mockScriptTag = {};
+        spyOn(document, 'createElement').and.returnValue(mockScriptTag);
+        LoadScript.appendScript('src.js', 'src-id', () => {}, () => {});
+      });
+
+      it('should create a script tag', () => {
+        expect(document.createElement).toHaveBeenCalledWith('script');
+      });
+
+      it('should set the MIME type', () => {
+        expect(mockScriptTag.type).toBe('text/javascript');
+      });
+
+      it('should set the script id', () => {
+        expect(mockScriptTag.id).toBe('src-id');
+      });
+
+      it('should set an onload handler', () => {
+        expect(mockScriptTag.onload).toEqual(jasmine.any(Function));
+      });
+
+      it('should set an onerror handler', () => {
+        expect(mockScriptTag.onerror).toEqual(jasmine.any(Function));
+      });
+
+      it('should set the src attribute', () => {
+        expect(mockScriptTag.src).toBe('src.js');
+      });
+
+      it('should append the script tag to the body element', () => {
+        expect(document.body.appendChild).toHaveBeenCalledWith(mockScriptTag);
+      });
+    });
+
+    it('should not set the script id if no id is provided', () => {
+      const mockScriptTag = {};
+      spyOn(document, 'createElement').and.returnValue(mockScriptTag);
+      LoadScript.appendScript('src.js', undefined);
+      expect(mockScriptTag.id).toBeUndefined();
+    });
+  });
+});
diff --git a/spec/javascripts/raven_config_spec.js.es6 b/spec/javascripts/raven_config_spec.js.es6
new file mode 100644
index 0000000000000..25df2cec75f03
--- /dev/null
+++ b/spec/javascripts/raven_config_spec.js.es6
@@ -0,0 +1,142 @@
+/* global ClassSpecHelper */
+
+/*= require raven */
+/*= require lib/utils/load_script */
+/*= require raven_config */
+/*= require class_spec_helper */
+
+describe('RavenConfig', () => {
+  const global = window.gl || (window.gl = {});
+  const RavenConfig = global.RavenConfig;
+
+  it('should be defined in the global scope', () => {
+    expect(RavenConfig).toBeDefined();
+  });
+
+  describe('.init', () => {
+    beforeEach(() => {
+      spyOn(global.LoadScript, 'load').and.callThrough();
+      spyOn(document, 'querySelector').and.returnValue(undefined);
+      spyOn(RavenConfig, 'configure');
+      spyOn(RavenConfig, 'bindRavenErrors');
+      spyOn(RavenConfig, 'setUser');
+      spyOn(Promise, 'reject');
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'init');
+
+    describe('when called', () => {
+      let options;
+      let initPromise;
+
+      beforeEach(() => {
+        options = {
+          sentryDsn: '//sentryDsn',
+          ravenAssetUrl: '//ravenAssetUrl',
+          currentUserId: 1,
+          whitelistUrls: ['//gitlabUrl'],
+          isProduction: true,
+        };
+        initPromise = RavenConfig.init(options);
+      });
+
+      it('should set the options property', () => {
+        expect(RavenConfig.options).toEqual(options);
+      });
+
+      it('should load a #raven-js script with the raven asset URL', () => {
+        expect(global.LoadScript.load).toHaveBeenCalledWith(options.ravenAssetUrl, 'raven-js');
+      });
+
+      it('should return a promise', () => {
+        expect(initPromise).toEqual(jasmine.any(Promise));
+      });
+
+      it('should call the configure method', () => {
+        initPromise.then(() => {
+          expect(RavenConfig.configure).toHaveBeenCalled();
+        });
+      });
+
+      it('should call the error bindings method', () => {
+        initPromise.then(() => {
+          expect(RavenConfig.bindRavenErrors).toHaveBeenCalled();
+        });
+      });
+
+      it('should call setUser', () => {
+        initPromise.then(() => {
+          expect(RavenConfig.setUser).toHaveBeenCalled();
+        });
+      });
+    });
+
+    it('should not call setUser if there is no current user ID', () => {
+      RavenConfig.init({
+        sentryDsn: '//sentryDsn',
+        ravenAssetUrl: '//ravenAssetUrl',
+        currentUserId: undefined,
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      });
+
+      expect(RavenConfig.setUser).not.toHaveBeenCalled();
+    });
+
+    it('should reject if there is no Sentry DSN', () => {
+      RavenConfig.init({
+        sentryDsn: undefined,
+        ravenAssetUrl: '//ravenAssetUrl',
+        currentUserId: 1,
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      });
+
+      expect(Promise.reject).toHaveBeenCalledWith('sentry dsn and raven asset url is required');
+    });
+
+    it('should reject if there is no Raven asset URL', () => {
+      RavenConfig.init({
+        sentryDsn: '//sentryDsn',
+        ravenAssetUrl: undefined,
+        currentUserId: 1,
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      });
+
+      expect(Promise.reject).toHaveBeenCalledWith('sentry dsn and raven asset url is required');
+    });
+  });
+
+  describe('.configure', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'configure');
+
+    describe('when called', () => {
+      beforeEach(() => {});
+    });
+  });
+
+  describe('.setUser', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'setUser');
+
+    describe('when called', () => {
+      beforeEach(() => {});
+    });
+  });
+
+  describe('.bindRavenErrors', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'bindRavenErrors');
+
+    describe('when called', () => {
+      beforeEach(() => {});
+    });
+  });
+
+  describe('.handleRavenErrors', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'handleRavenErrors');
+
+    describe('when called', () => {
+      beforeEach(() => {});
+    });
+  });
+});
diff --git a/spec/javascripts/spec_helper.js b/spec/javascripts/spec_helper.js
index f8e3aca29fa52..ee6e6279ac9fc 100644
--- a/spec/javascripts/spec_helper.js
+++ b/spec/javascripts/spec_helper.js
@@ -11,6 +11,7 @@
 /*= require jquery.turbolinks */
 /*= require bootstrap */
 /*= require underscore */
+/*= require es6-promise.auto */
 
 // Teaspoon includes some support files, but you can use anything from your own
 // support path too.
diff --git a/vendor/assets/javascripts/raven.js b/vendor/assets/javascripts/raven.js
new file mode 100644
index 0000000000000..ba416b26be14d
--- /dev/null
+++ b/vendor/assets/javascripts/raven.js
@@ -0,0 +1,2547 @@
+/*! Raven.js 3.9.1 (7bbae7d) | github.com/getsentry/raven-js */
+
+/*
+ * Includes TraceKit
+ * https://github.com/getsentry/TraceKit
+ *
+ * Copyright 2016 Matt Robenolt and other contributors
+ * Released under the BSD license
+ * https://github.com/getsentry/raven-js/blob/master/LICENSE
+ *
+ */
+
+(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.Raven = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(_dereq_,module,exports){
+exports = module.exports = stringify
+exports.getSerialize = serializer
+
+function stringify(obj, replacer, spaces, cycleReplacer) {
+  return JSON.stringify(obj, serializer(replacer, cycleReplacer), spaces)
+}
+
+function serializer(replacer, cycleReplacer) {
+  var stack = [], keys = []
+
+  if (cycleReplacer == null) cycleReplacer = function(key, value) {
+    if (stack[0] === value) return "[Circular ~]"
+    return "[Circular ~." + keys.slice(0, stack.indexOf(value)).join(".") + "]"
+  }
+
+  return function(key, value) {
+    if (stack.length > 0) {
+      var thisPos = stack.indexOf(this)
+      ~thisPos ? stack.splice(thisPos + 1) : stack.push(this)
+      ~thisPos ? keys.splice(thisPos, Infinity, key) : keys.push(key)
+      if (~stack.indexOf(value)) value = cycleReplacer.call(this, key, value)
+    }
+    else stack.push(value)
+
+    return replacer == null ? value : replacer.call(this, key, value)
+  }
+}
+
+},{}],2:[function(_dereq_,module,exports){
+'use strict';
+
+function RavenConfigError(message) {
+    this.name = 'RavenConfigError';
+    this.message = message;
+}
+RavenConfigError.prototype = new Error();
+RavenConfigError.prototype.constructor = RavenConfigError;
+
+module.exports = RavenConfigError;
+
+},{}],3:[function(_dereq_,module,exports){
+'use strict';
+
+var wrapMethod = function(console, level, callback) {
+    var originalConsoleLevel = console[level];
+    var originalConsole = console;
+
+    if (!(level in console)) {
+        return;
+    }
+
+    var sentryLevel = level === 'warn'
+        ? 'warning'
+        : level;
+
+    console[level] = function () {
+        var args = [].slice.call(arguments);
+
+        var msg = '' + args.join(' ');
+        var data = {level: sentryLevel, logger: 'console', extra: {'arguments': args}};
+        callback && callback(msg, data);
+
+        // this fails for some browsers. :(
+        if (originalConsoleLevel) {
+            // IE9 doesn't allow calling apply on console functions directly
+            // See: https://stackoverflow.com/questions/5472938/does-ie9-support-console-log-and-is-it-a-real-function#answer-5473193
+            Function.prototype.apply.call(
+                originalConsoleLevel,
+                originalConsole,
+                args
+            );
+        }
+    };
+};
+
+module.exports = {
+    wrapMethod: wrapMethod
+};
+
+},{}],4:[function(_dereq_,module,exports){
+(function (global){
+/*global XDomainRequest:false, __DEV__:false*/
+'use strict';
+
+var TraceKit = _dereq_(6);
+var RavenConfigError = _dereq_(2);
+var stringify = _dereq_(1);
+
+var wrapConsoleMethod = _dereq_(3).wrapMethod;
+
+var dsnKeys = 'source protocol user pass host port path'.split(' '),
+    dsnPattern = /^(?:(\w+):)?\/\/(?:(\w+)(:\w+)?@)?([\w\.-]+)(?::(\d+))?(\/.*)/;
+
+function now() {
+    return +new Date();
+}
+
+// This is to be defensive in environments where window does not exist (see https://github.com/getsentry/raven-js/pull/785)
+var _window = typeof window !== 'undefined' ? window
+            : typeof global !== 'undefined' ? global
+            : typeof self !== 'undefined' ? self
+            : {};
+var _document = _window.document;
+
+// First, check for JSON support
+// If there is no JSON, we no-op the core features of Raven
+// since JSON is required to encode the payload
+function Raven() {
+    this._hasJSON = !!(typeof JSON === 'object' && JSON.stringify);
+    // Raven can run in contexts where there's no document (react-native)
+    this._hasDocument = !isUndefined(_document);
+    this._lastCapturedException = null;
+    this._lastEventId = null;
+    this._globalServer = null;
+    this._globalKey = null;
+    this._globalProject = null;
+    this._globalContext = {};
+    this._globalOptions = {
+        logger: 'javascript',
+        ignoreErrors: [],
+        ignoreUrls: [],
+        whitelistUrls: [],
+        includePaths: [],
+        crossOrigin: 'anonymous',
+        collectWindowErrors: true,
+        maxMessageLength: 0,
+        stackTraceLimit: 50,
+        autoBreadcrumbs: true
+    };
+    this._ignoreOnError = 0;
+    this._isRavenInstalled = false;
+    this._originalErrorStackTraceLimit = Error.stackTraceLimit;
+    // capture references to window.console *and* all its methods first
+    // before the console plugin has a chance to monkey patch
+    this._originalConsole = _window.console || {};
+    this._originalConsoleMethods = {};
+    this._plugins = [];
+    this._startTime = now();
+    this._wrappedBuiltIns = [];
+    this._breadcrumbs = [];
+    this._lastCapturedEvent = null;
+    this._keypressTimeout;
+    this._location = _window.location;
+    this._lastHref = this._location && this._location.href;
+
+    for (var method in this._originalConsole) {  // eslint-disable-line guard-for-in
+      this._originalConsoleMethods[method] = this._originalConsole[method];
+    }
+}
+
+/*
+ * The core Raven singleton
+ *
+ * @this {Raven}
+ */
+
+Raven.prototype = {
+    // Hardcode version string so that raven source can be loaded directly via
+    // webpack (using a build step causes webpack #1617). Grunt verifies that
+    // this value matches package.json during build.
+    //   See: https://github.com/getsentry/raven-js/issues/465
+    VERSION: '3.9.1',
+
+    debug: false,
+
+    TraceKit: TraceKit, // alias to TraceKit
+
+    /*
+     * Configure Raven with a DSN and extra options
+     *
+     * @param {string} dsn The public Sentry DSN
+     * @param {object} options Optional set of of global options [optional]
+     * @return {Raven}
+     */
+    config: function(dsn, options) {
+        var self = this;
+
+        if (self._globalServer) {
+                this._logDebug('error', 'Error: Raven has already been configured');
+            return self;
+        }
+        if (!dsn) return self;
+
+        var globalOptions = self._globalOptions;
+
+        // merge in options
+        if (options) {
+            each(options, function(key, value){
+                // tags and extra are special and need to be put into context
+                if (key === 'tags' || key === 'extra' || key === 'user') {
+                    self._globalContext[key] = value;
+                } else {
+                    globalOptions[key] = value;
+                }
+            });
+        }
+
+        self.setDSN(dsn);
+
+        // "Script error." is hard coded into browsers for errors that it can't read.
+        // this is the result of a script being pulled in from an external domain and CORS.
+        globalOptions.ignoreErrors.push(/^Script error\.?$/);
+        globalOptions.ignoreErrors.push(/^Javascript error: Script error\.? on line 0$/);
+
+        // join regexp rules into one big rule
+        globalOptions.ignoreErrors = joinRegExp(globalOptions.ignoreErrors);
+        globalOptions.ignoreUrls = globalOptions.ignoreUrls.length ? joinRegExp(globalOptions.ignoreUrls) : false;
+        globalOptions.whitelistUrls = globalOptions.whitelistUrls.length ? joinRegExp(globalOptions.whitelistUrls) : false;
+        globalOptions.includePaths = joinRegExp(globalOptions.includePaths);
+        globalOptions.maxBreadcrumbs = Math.max(0, Math.min(globalOptions.maxBreadcrumbs || 100, 100)); // default and hard limit is 100
+
+        var autoBreadcrumbDefaults = {
+            xhr: true,
+            console: true,
+            dom: true,
+            location: true
+        };
+
+        var autoBreadcrumbs = globalOptions.autoBreadcrumbs;
+        if ({}.toString.call(autoBreadcrumbs) === '[object Object]') {
+            autoBreadcrumbs = objectMerge(autoBreadcrumbDefaults, autoBreadcrumbs);
+        } else if (autoBreadcrumbs !== false) {
+            autoBreadcrumbs = autoBreadcrumbDefaults;
+        }
+        globalOptions.autoBreadcrumbs = autoBreadcrumbs;
+
+        TraceKit.collectWindowErrors = !!globalOptions.collectWindowErrors;
+
+        // return for chaining
+        return self;
+    },
+
+    /*
+     * Installs a global window.onerror error handler
+     * to capture and report uncaught exceptions.
+     * At this point, install() is required to be called due
+     * to the way TraceKit is set up.
+     *
+     * @return {Raven}
+     */
+    install: function() {
+        var self = this;
+        if (self.isSetup() && !self._isRavenInstalled) {
+            TraceKit.report.subscribe(function () {
+                self._handleOnErrorStackInfo.apply(self, arguments);
+            });
+            self._instrumentTryCatch();
+            if (self._globalOptions.autoBreadcrumbs)
+                self._instrumentBreadcrumbs();
+
+            // Install all of the plugins
+            self._drainPlugins();
+
+            self._isRavenInstalled = true;
+        }
+
+        Error.stackTraceLimit = self._globalOptions.stackTraceLimit;
+        return this;
+    },
+
+    /*
+     * Set the DSN (can be called multiple time unlike config)
+     *
+     * @param {string} dsn The public Sentry DSN
+     */
+    setDSN: function(dsn) {
+        var self = this,
+            uri = self._parseDSN(dsn),
+          lastSlash = uri.path.lastIndexOf('/'),
+          path = uri.path.substr(1, lastSlash);
+
+        self._dsn = dsn;
+        self._globalKey = uri.user;
+        self._globalSecret = uri.pass && uri.pass.substr(1);
+        self._globalProject = uri.path.substr(lastSlash + 1);
+
+        self._globalServer = self._getGlobalServer(uri);
+
+        self._globalEndpoint = self._globalServer +
+            '/' + path + 'api/' + self._globalProject + '/store/';
+    },
+
+    /*
+     * Wrap code within a context so Raven can capture errors
+     * reliably across domains that is executed immediately.
+     *
+     * @param {object} options A specific set of options for this context [optional]
+     * @param {function} func The callback to be immediately executed within the context
+     * @param {array} args An array of arguments to be called with the callback [optional]
+     */
+    context: function(options, func, args) {
+        if (isFunction(options)) {
+            args = func || [];
+            func = options;
+            options = undefined;
+        }
+
+        return this.wrap(options, func).apply(this, args);
+    },
+
+    /*
+     * Wrap code within a context and returns back a new function to be executed
+     *
+     * @param {object} options A specific set of options for this context [optional]
+     * @param {function} func The function to be wrapped in a new context
+     * @param {function} func A function to call before the try/catch wrapper [optional, private]
+     * @return {function} The newly wrapped functions with a context
+     */
+    wrap: function(options, func, _before) {
+        var self = this;
+        // 1 argument has been passed, and it's not a function
+        // so just return it
+        if (isUndefined(func) && !isFunction(options)) {
+            return options;
+        }
+
+        // options is optional
+        if (isFunction(options)) {
+            func = options;
+            options = undefined;
+        }
+
+        // At this point, we've passed along 2 arguments, and the second one
+        // is not a function either, so we'll just return the second argument.
+        if (!isFunction(func)) {
+            return func;
+        }
+
+        // We don't wanna wrap it twice!
+        try {
+            if (func.__raven__) {
+                return func;
+            }
+
+            // If this has already been wrapped in the past, return that
+            if (func.__raven_wrapper__ ){
+                return func.__raven_wrapper__ ;
+            }
+        } catch (e) {
+            // Just accessing custom props in some Selenium environments
+            // can cause a "Permission denied" exception (see raven-js#495).
+            // Bail on wrapping and return the function as-is (defers to window.onerror).
+            return func;
+        }
+
+        function wrapped() {
+            var args = [], i = arguments.length,
+                deep = !options || options && options.deep !== false;
+
+            if (_before && isFunction(_before)) {
+                _before.apply(this, arguments);
+            }
+
+            // Recursively wrap all of a function's arguments that are
+            // functions themselves.
+            while(i--) args[i] = deep ? self.wrap(options, arguments[i]) : arguments[i];
+
+            try {
+                return func.apply(this, args);
+            } catch(e) {
+                self._ignoreNextOnError();
+                self.captureException(e, options);
+                throw e;
+            }
+        }
+
+        // copy over properties of the old function
+        for (var property in func) {
+            if (hasKey(func, property)) {
+                wrapped[property] = func[property];
+            }
+        }
+        wrapped.prototype = func.prototype;
+
+        func.__raven_wrapper__ = wrapped;
+        // Signal that this function has been wrapped already
+        // for both debugging and to prevent it to being wrapped twice
+        wrapped.__raven__ = true;
+        wrapped.__inner__ = func;
+
+        return wrapped;
+    },
+
+    /*
+     * Uninstalls the global error handler.
+     *
+     * @return {Raven}
+     */
+    uninstall: function() {
+        TraceKit.report.uninstall();
+
+        this._restoreBuiltIns();
+
+        Error.stackTraceLimit = this._originalErrorStackTraceLimit;
+        this._isRavenInstalled = false;
+
+        return this;
+    },
+
+    /*
+     * Manually capture an exception and send it over to Sentry
+     *
+     * @param {error} ex An exception to be logged
+     * @param {object} options A specific set of options for this error [optional]
+     * @return {Raven}
+     */
+    captureException: function(ex, options) {
+        // If not an Error is passed through, recall as a message instead
+        if (!isError(ex)) {
+            return this.captureMessage(ex, objectMerge({
+                trimHeadFrames: 1,
+                stacktrace: true // if we fall back to captureMessage, default to attempting a new trace
+            }, options));
+        }
+
+        // Store the raw exception object for potential debugging and introspection
+        this._lastCapturedException = ex;
+
+        // TraceKit.report will re-raise any exception passed to it,
+        // which means you have to wrap it in try/catch. Instead, we
+        // can wrap it here and only re-raise if TraceKit.report
+        // raises an exception different from the one we asked to
+        // report on.
+        try {
+            var stack = TraceKit.computeStackTrace(ex);
+            this._handleStackInfo(stack, options);
+        } catch(ex1) {
+            if(ex !== ex1) {
+                throw ex1;
+            }
+        }
+
+        return this;
+    },
+
+    /*
+     * Manually send a message to Sentry
+     *
+     * @param {string} msg A plain message to be captured in Sentry
+     * @param {object} options A specific set of options for this message [optional]
+     * @return {Raven}
+     */
+    captureMessage: function(msg, options) {
+        // config() automagically converts ignoreErrors from a list to a RegExp so we need to test for an
+        // early call; we'll error on the side of logging anything called before configuration since it's
+        // probably something you should see:
+        if (!!this._globalOptions.ignoreErrors.test && this._globalOptions.ignoreErrors.test(msg)) {
+            return;
+        }
+
+        options = options || {};
+
+        var data = objectMerge({
+            message: msg + ''  // Make sure it's actually a string
+        }, options);
+
+        if (this._globalOptions.stacktrace || (options && options.stacktrace)) {
+            var ex;
+            // create a stack trace from this point; just trim
+            // off extra frames so they don't include this function call (or
+            // earlier Raven.js library fn calls)
+            try {
+                throw new Error(msg);
+            } catch (ex1) {
+                ex = ex1;
+            }
+
+            // null exception name so `Error` isn't prefixed to msg
+            ex.name = null;
+
+            options = objectMerge({
+                // fingerprint on msg, not stack trace (legacy behavior, could be
+                // revisited)
+                fingerprint: msg,
+                trimHeadFrames: (options.trimHeadFrames || 0) + 1
+            }, options);
+
+            var stack = TraceKit.computeStackTrace(ex);
+            var frames = this._prepareFrames(stack, options);
+            data.stacktrace = {
+                // Sentry expects frames oldest to newest
+                frames: frames.reverse()
+            }
+        }
+
+        // Fire away!
+        this._send(data);
+
+        return this;
+    },
+
+    captureBreadcrumb: function (obj) {
+        var crumb = objectMerge({
+            timestamp: now() / 1000
+        }, obj);
+
+        if (isFunction(this._globalOptions.breadcrumbCallback)) {
+            var result = this._globalOptions.breadcrumbCallback(crumb);
+
+            if (isObject(result) && !isEmptyObject(result)) {
+                crumb = result;
+            } else if (result === false) {
+                return this;
+            }
+        }
+
+        this._breadcrumbs.push(crumb);
+        if (this._breadcrumbs.length > this._globalOptions.maxBreadcrumbs) {
+            this._breadcrumbs.shift();
+        }
+        return this;
+    },
+
+    addPlugin: function(plugin /*arg1, arg2, ... argN*/) {
+        var pluginArgs = [].slice.call(arguments, 1);
+
+        this._plugins.push([plugin, pluginArgs]);
+        if (this._isRavenInstalled) {
+            this._drainPlugins();
+        }
+
+        return this;
+    },
+
+    /*
+     * Set/clear a user to be sent along with the payload.
+     *
+     * @param {object} user An object representing user data [optional]
+     * @return {Raven}
+     */
+    setUserContext: function(user) {
+        // Intentionally do not merge here since that's an unexpected behavior.
+        this._globalContext.user = user;
+
+        return this;
+    },
+
+    /*
+     * Merge extra attributes to be sent along with the payload.
+     *
+     * @param {object} extra An object representing extra data [optional]
+     * @return {Raven}
+     */
+    setExtraContext: function(extra) {
+        this._mergeContext('extra', extra);
+
+        return this;
+    },
+
+    /*
+     * Merge tags to be sent along with the payload.
+     *
+     * @param {object} tags An object representing tags [optional]
+     * @return {Raven}
+     */
+    setTagsContext: function(tags) {
+        this._mergeContext('tags', tags);
+
+        return this;
+    },
+
+    /*
+     * Clear all of the context.
+     *
+     * @return {Raven}
+     */
+    clearContext: function() {
+        this._globalContext = {};
+
+        return this;
+    },
+
+    /*
+     * Get a copy of the current context. This cannot be mutated.
+     *
+     * @return {object} copy of context
+     */
+    getContext: function() {
+        // lol javascript
+        return JSON.parse(stringify(this._globalContext));
+    },
+
+
+    /*
+     * Set environment of application
+     *
+     * @param {string} environment Typically something like 'production'.
+     * @return {Raven}
+     */
+    setEnvironment: function(environment) {
+        this._globalOptions.environment = environment;
+
+        return this;
+    },
+
+    /*
+     * Set release version of application
+     *
+     * @param {string} release Typically something like a git SHA to identify version
+     * @return {Raven}
+     */
+    setRelease: function(release) {
+        this._globalOptions.release = release;
+
+        return this;
+    },
+
+    /*
+     * Set the dataCallback option
+     *
+     * @param {function} callback The callback to run which allows the
+     *                            data blob to be mutated before sending
+     * @return {Raven}
+     */
+    setDataCallback: function(callback) {
+        var original = this._globalOptions.dataCallback;
+        this._globalOptions.dataCallback = isFunction(callback)
+          ? function (data) { return callback(data, original); }
+          : callback;
+
+        return this;
+    },
+
+    /*
+     * Set the breadcrumbCallback option
+     *
+     * @param {function} callback The callback to run which allows filtering
+     *                            or mutating breadcrumbs
+     * @return {Raven}
+     */
+    setBreadcrumbCallback: function(callback) {
+        var original = this._globalOptions.breadcrumbCallback;
+        this._globalOptions.breadcrumbCallback = isFunction(callback)
+          ? function (data) { return callback(data, original); }
+          : callback;
+
+        return this;
+    },
+
+    /*
+     * Set the shouldSendCallback option
+     *
+     * @param {function} callback The callback to run which allows
+     *                            introspecting the blob before sending
+     * @return {Raven}
+     */
+    setShouldSendCallback: function(callback) {
+        var original = this._globalOptions.shouldSendCallback;
+        this._globalOptions.shouldSendCallback = isFunction(callback)
+            ? function (data) { return callback(data, original); }
+            : callback;
+
+        return this;
+    },
+
+    /**
+     * Override the default HTTP transport mechanism that transmits data
+     * to the Sentry server.
+     *
+     * @param {function} transport Function invoked instead of the default
+     *                             `makeRequest` handler.
+     *
+     * @return {Raven}
+     */
+    setTransport: function(transport) {
+        this._globalOptions.transport = transport;
+
+        return this;
+    },
+
+    /*
+     * Get the latest raw exception that was captured by Raven.
+     *
+     * @return {error}
+     */
+    lastException: function() {
+        return this._lastCapturedException;
+    },
+
+    /*
+     * Get the last event id
+     *
+     * @return {string}
+     */
+    lastEventId: function() {
+        return this._lastEventId;
+    },
+
+    /*
+     * Determine if Raven is setup and ready to go.
+     *
+     * @return {boolean}
+     */
+    isSetup: function() {
+        if (!this._hasJSON) return false;  // needs JSON support
+        if (!this._globalServer) {
+            if (!this.ravenNotConfiguredError) {
+              this.ravenNotConfiguredError = true;
+              this._logDebug('error', 'Error: Raven has not been configured.');
+            }
+            return false;
+        }
+        return true;
+    },
+
+    afterLoad: function () {
+        // TODO: remove window dependence?
+
+        // Attempt to initialize Raven on load
+        var RavenConfig = _window.RavenConfig;
+        if (RavenConfig) {
+            this.config(RavenConfig.dsn, RavenConfig.config).install();
+        }
+    },
+
+    showReportDialog: function (options) {
+        if (!_document) // doesn't work without a document (React native)
+            return;
+
+        options = options || {};
+
+        var lastEventId = options.eventId || this.lastEventId();
+        if (!lastEventId) {
+            throw new RavenConfigError('Missing eventId');
+        }
+
+        var dsn = options.dsn || this._dsn;
+        if (!dsn) {
+            throw new RavenConfigError('Missing DSN');
+        }
+
+        var encode = encodeURIComponent;
+        var qs = '';
+        qs += '?eventId=' + encode(lastEventId);
+        qs += '&dsn=' + encode(dsn);
+
+        var user = options.user || this._globalContext.user;
+        if (user) {
+            if (user.name)  qs += '&name=' + encode(user.name);
+            if (user.email) qs += '&email=' + encode(user.email);
+        }
+
+        var globalServer = this._getGlobalServer(this._parseDSN(dsn));
+
+        var script = _document.createElement('script');
+        script.async = true;
+        script.src = globalServer + '/api/embed/error-page/' + qs;
+        (_document.head || _document.body).appendChild(script);
+    },
+
+    /**** Private functions ****/
+    _ignoreNextOnError: function () {
+        var self = this;
+        this._ignoreOnError += 1;
+        setTimeout(function () {
+            // onerror should trigger before setTimeout
+            self._ignoreOnError -= 1;
+        });
+    },
+
+    _triggerEvent: function(eventType, options) {
+        // NOTE: `event` is a native browser thing, so let's avoid conflicting wiht it
+        var evt, key;
+
+        if (!this._hasDocument)
+            return;
+
+        options = options || {};
+
+        eventType = 'raven' + eventType.substr(0,1).toUpperCase() + eventType.substr(1);
+
+        if (_document.createEvent) {
+            evt = _document.createEvent('HTMLEvents');
+            evt.initEvent(eventType, true, true);
+        } else {
+            evt = _document.createEventObject();
+            evt.eventType = eventType;
+        }
+
+        for (key in options) if (hasKey(options, key)) {
+            evt[key] = options[key];
+        }
+
+        if (_document.createEvent) {
+            // IE9 if standards
+            _document.dispatchEvent(evt);
+        } else {
+            // IE8 regardless of Quirks or Standards
+            // IE9 if quirks
+            try {
+                _document.fireEvent('on' + evt.eventType.toLowerCase(), evt);
+            } catch(e) {
+                // Do nothing
+            }
+        }
+    },
+
+    /**
+     * Wraps addEventListener to capture UI breadcrumbs
+     * @param evtName the event name (e.g. "click")
+     * @returns {Function}
+     * @private
+     */
+    _breadcrumbEventHandler: function(evtName) {
+        var self = this;
+        return function (evt) {
+            // reset keypress timeout; e.g. triggering a 'click' after
+            // a 'keypress' will reset the keypress debounce so that a new
+            // set of keypresses can be recorded
+            self._keypressTimeout = null;
+
+            // It's possible this handler might trigger multiple times for the same
+            // event (e.g. event propagation through node ancestors). Ignore if we've
+            // already captured the event.
+            if (self._lastCapturedEvent === evt)
+                return;
+
+            self._lastCapturedEvent = evt;
+            var elem = evt.target;
+
+            var target;
+
+            // try/catch htmlTreeAsString because it's particularly complicated, and
+            // just accessing the DOM incorrectly can throw an exception in some circumstances.
+            try {
+                target = htmlTreeAsString(elem);
+            } catch (e) {
+                target = '<unknown>';
+            }
+
+            self.captureBreadcrumb({
+                category: 'ui.' + evtName, // e.g. ui.click, ui.input
+                message: target
+            });
+        };
+    },
+
+    /**
+     * Wraps addEventListener to capture keypress UI events
+     * @returns {Function}
+     * @private
+     */
+    _keypressEventHandler: function() {
+        var self = this,
+            debounceDuration = 1000; // milliseconds
+
+        // TODO: if somehow user switches keypress target before
+        //       debounce timeout is triggered, we will only capture
+        //       a single breadcrumb from the FIRST target (acceptable?)
+        return function (evt) {
+            var target = evt.target,
+                tagName = target && target.tagName;
+
+            // only consider keypress events on actual input elements
+            // this will disregard keypresses targeting body (e.g. tabbing
+            // through elements, hotkeys, etc)
+            if (!tagName || tagName !== 'INPUT' && tagName !== 'TEXTAREA' && !target.isContentEditable)
+                return;
+
+            // record first keypress in a series, but ignore subsequent
+            // keypresses until debounce clears
+            var timeout = self._keypressTimeout;
+            if (!timeout) {
+                self._breadcrumbEventHandler('input')(evt);
+            }
+            clearTimeout(timeout);
+            self._keypressTimeout = setTimeout(function () {
+                self._keypressTimeout = null;
+            }, debounceDuration);
+        };
+    },
+
+    /**
+     * Captures a breadcrumb of type "navigation", normalizing input URLs
+     * @param to the originating URL
+     * @param from the target URL
+     * @private
+     */
+    _captureUrlChange: function(from, to) {
+        var parsedLoc = parseUrl(this._location.href);
+        var parsedTo = parseUrl(to);
+        var parsedFrom = parseUrl(from);
+
+        // because onpopstate only tells you the "new" (to) value of location.href, and
+        // not the previous (from) value, we need to track the value of the current URL
+        // state ourselves
+        this._lastHref = to;
+
+        // Use only the path component of the URL if the URL matches the current
+        // document (almost all the time when using pushState)
+        if (parsedLoc.protocol === parsedTo.protocol && parsedLoc.host === parsedTo.host)
+            to = parsedTo.relative;
+        if (parsedLoc.protocol === parsedFrom.protocol && parsedLoc.host === parsedFrom.host)
+            from = parsedFrom.relative;
+
+        this.captureBreadcrumb({
+            category: 'navigation',
+            data: {
+                to: to,
+                from: from
+            }
+        });
+    },
+
+    /**
+     * Install any queued plugins
+     */
+    _instrumentTryCatch: function() {
+        var self = this;
+
+        var wrappedBuiltIns = self._wrappedBuiltIns;
+
+        function wrapTimeFn(orig) {
+            return function (fn, t) { // preserve arity
+                // Make a copy of the arguments to prevent deoptimization
+                // https://github.com/petkaantonov/bluebird/wiki/Optimization-killers#32-leaking-arguments
+                var args = new Array(arguments.length);
+                for(var i = 0; i < args.length; ++i) {
+                    args[i] = arguments[i];
+                }
+                var originalCallback = args[0];
+                if (isFunction(originalCallback)) {
+                    args[0] = self.wrap(originalCallback);
+                }
+
+                // IE < 9 doesn't support .call/.apply on setInterval/setTimeout, but it
+                // also supports only two arguments and doesn't care what this is, so we
+                // can just call the original function directly.
+                if (orig.apply) {
+                    return orig.apply(this, args);
+                } else {
+                    return orig(args[0], args[1]);
+                }
+            };
+        }
+
+        var autoBreadcrumbs = this._globalOptions.autoBreadcrumbs;
+
+        function wrapEventTarget(global) {
+            var proto = _window[global] && _window[global].prototype;
+            if (proto && proto.hasOwnProperty && proto.hasOwnProperty('addEventListener')) {
+                fill(proto, 'addEventListener', function(orig) {
+                    return function (evtName, fn, capture, secure) { // preserve arity
+                        try {
+                            if (fn && fn.handleEvent) {
+                                fn.handleEvent = self.wrap(fn.handleEvent);
+                            }
+                        } catch (err) {
+                            // can sometimes get 'Permission denied to access property "handle Event'
+                        }
+
+                        // More breadcrumb DOM capture ... done here and not in `_instrumentBreadcrumbs`
+                        // so that we don't have more than one wrapper function
+                        var before,
+                            clickHandler,
+                            keypressHandler;
+
+                        if (autoBreadcrumbs && autoBreadcrumbs.dom && (global === 'EventTarget' || global === 'Node')) {
+                            // NOTE: generating multiple handlers per addEventListener invocation, should
+                            //       revisit and verify we can just use one (almost certainly)
+                            clickHandler = self._breadcrumbEventHandler('click');
+                            keypressHandler = self._keypressEventHandler();
+                            before = function (evt) {
+                                // need to intercept every DOM event in `before` argument, in case that
+                                // same wrapped method is re-used for different events (e.g. mousemove THEN click)
+                                // see #724
+                                if (!evt) return;
+
+                                if (evt.type === 'click')
+                                    return clickHandler(evt);
+                                else if (evt.type === 'keypress')
+                                    return keypressHandler(evt);
+                            };
+                        }
+                        return orig.call(this, evtName, self.wrap(fn, undefined, before), capture, secure);
+                    };
+                }, wrappedBuiltIns);
+                fill(proto, 'removeEventListener', function (orig) {
+                    return function (evt, fn, capture, secure) {
+                        try {
+                            fn = fn && (fn.__raven_wrapper__ ? fn.__raven_wrapper__  : fn);
+                        } catch (e) {
+                            // ignore, accessing __raven_wrapper__ will throw in some Selenium environments
+                        }
+                        return orig.call(this, evt, fn, capture, secure);
+                    };
+                }, wrappedBuiltIns);
+            }
+        }
+
+        fill(_window, 'setTimeout', wrapTimeFn, wrappedBuiltIns);
+        fill(_window, 'setInterval', wrapTimeFn, wrappedBuiltIns);
+        if (_window.requestAnimationFrame) {
+            fill(_window, 'requestAnimationFrame', function (orig) {
+                return function (cb) {
+                    return orig(self.wrap(cb));
+                };
+            }, wrappedBuiltIns);
+        }
+
+        // event targets borrowed from bugsnag-js:
+        // https://github.com/bugsnag/bugsnag-js/blob/master/src/bugsnag.js#L666
+        var eventTargets = ['EventTarget', 'Window', 'Node', 'ApplicationCache', 'AudioTrackList', 'ChannelMergerNode', 'CryptoOperation', 'EventSource', 'FileReader', 'HTMLUnknownElement', 'IDBDatabase', 'IDBRequest', 'IDBTransaction', 'KeyOperation', 'MediaController', 'MessagePort', 'ModalWindow', 'Notification', 'SVGElementInstance', 'Screen', 'TextTrack', 'TextTrackCue', 'TextTrackList', 'WebSocket', 'WebSocketWorker', 'Worker', 'XMLHttpRequest', 'XMLHttpRequestEventTarget', 'XMLHttpRequestUpload'];
+        for (var i = 0; i < eventTargets.length; i++) {
+            wrapEventTarget(eventTargets[i]);
+        }
+
+        var $ = _window.jQuery || _window.$;
+        if ($ && $.fn && $.fn.ready) {
+            fill($.fn, 'ready', function (orig) {
+                return function (fn) {
+                    return orig.call(this, self.wrap(fn));
+                };
+            }, wrappedBuiltIns);
+        }
+    },
+
+
+    /**
+     * Instrument browser built-ins w/ breadcrumb capturing
+     *  - XMLHttpRequests
+     *  - DOM interactions (click/typing)
+     *  - window.location changes
+     *  - console
+     *
+     * Can be disabled or individually configured via the `autoBreadcrumbs` config option
+     */
+    _instrumentBreadcrumbs: function () {
+        var self = this;
+        var autoBreadcrumbs = this._globalOptions.autoBreadcrumbs;
+
+        var wrappedBuiltIns = self._wrappedBuiltIns;
+
+        function wrapProp(prop, xhr) {
+            if (prop in xhr && isFunction(xhr[prop])) {
+                fill(xhr, prop, function (orig) {
+                    return self.wrap(orig);
+                }); // intentionally don't track filled methods on XHR instances
+            }
+        }
+
+        if (autoBreadcrumbs.xhr && 'XMLHttpRequest' in _window) {
+            var xhrproto = XMLHttpRequest.prototype;
+            fill(xhrproto, 'open', function(origOpen) {
+                return function (method, url) { // preserve arity
+
+                    // if Sentry key appears in URL, don't capture
+                    if (isString(url) && url.indexOf(self._globalKey) === -1) {
+                        this.__raven_xhr = {
+                            method: method,
+                            url: url,
+                            status_code: null
+                        };
+                    }
+
+                    return origOpen.apply(this, arguments);
+                };
+            }, wrappedBuiltIns);
+
+            fill(xhrproto, 'send', function(origSend) {
+                return function (data) { // preserve arity
+                    var xhr = this;
+
+                    function onreadystatechangeHandler() {
+                        if (xhr.__raven_xhr && (xhr.readyState === 1 || xhr.readyState === 4)) {
+                            try {
+                                // touching statusCode in some platforms throws
+                                // an exception
+                                xhr.__raven_xhr.status_code = xhr.status;
+                            } catch (e) { /* do nothing */ }
+                            self.captureBreadcrumb({
+                                type: 'http',
+                                category: 'xhr',
+                                data: xhr.__raven_xhr
+                            });
+                        }
+                    }
+
+                    var props = ['onload', 'onerror', 'onprogress'];
+                    for (var j = 0; j < props.length; j++) {
+                        wrapProp(props[j], xhr);
+                    }
+
+                    if ('onreadystatechange' in xhr && isFunction(xhr.onreadystatechange)) {
+                        fill(xhr, 'onreadystatechange', function (orig) {
+                            return self.wrap(orig, undefined, onreadystatechangeHandler);
+                        } /* intentionally don't track this instrumentation */);
+                    } else {
+                        // if onreadystatechange wasn't actually set by the page on this xhr, we
+                        // are free to set our own and capture the breadcrumb
+                        xhr.onreadystatechange = onreadystatechangeHandler;
+                    }
+
+                    return origSend.apply(this, arguments);
+                };
+            }, wrappedBuiltIns);
+        }
+
+        if (autoBreadcrumbs.xhr && 'fetch' in _window) {
+            fill(_window, 'fetch', function(origFetch) {
+                return function (fn, t) { // preserve arity
+                    // Make a copy of the arguments to prevent deoptimization
+                    // https://github.com/petkaantonov/bluebird/wiki/Optimization-killers#32-leaking-arguments
+                    var args = new Array(arguments.length);
+                    for(var i = 0; i < args.length; ++i) {
+                        args[i] = arguments[i];
+                    }
+
+                    var method = 'GET';
+
+                    if (args[1] && args[1].method) {
+                        method = args[1].method;
+                    }
+
+                    var fetchData = {
+                        method: method,
+                        url: args[0],
+                        status_code: null
+                    };
+
+                    self.captureBreadcrumb({
+                        type: 'http',
+                        category: 'fetch',
+                        data: fetchData
+                    });
+
+                    return origFetch.apply(this, args).then(function (response) {
+                        fetchData.status_code = response.status;
+
+                        return response;
+                    });
+                };
+            }, wrappedBuiltIns);
+        }
+
+        // Capture breadcrumbs from any click that is unhandled / bubbled up all the way
+        // to the document. Do this before we instrument addEventListener.
+        if (autoBreadcrumbs.dom && this._hasDocument) {
+            if (_document.addEventListener) {
+                _document.addEventListener('click', self._breadcrumbEventHandler('click'), false);
+                _document.addEventListener('keypress', self._keypressEventHandler(), false);
+            }
+            else {
+                // IE8 Compatibility
+                _document.attachEvent('onclick', self._breadcrumbEventHandler('click'));
+                _document.attachEvent('onkeypress', self._keypressEventHandler());
+            }
+        }
+
+        // record navigation (URL) changes
+        // NOTE: in Chrome App environment, touching history.pushState, *even inside
+        //       a try/catch block*, will cause Chrome to output an error to console.error
+        // borrowed from: https://github.com/angular/angular.js/pull/13945/files
+        var chrome = _window.chrome;
+        var isChromePackagedApp = chrome && chrome.app && chrome.app.runtime;
+        var hasPushState = !isChromePackagedApp && _window.history && history.pushState;
+        if (autoBreadcrumbs.location && hasPushState) {
+            // TODO: remove onpopstate handler on uninstall()
+            var oldOnPopState = _window.onpopstate;
+            _window.onpopstate = function () {
+                var currentHref = self._location.href;
+                self._captureUrlChange(self._lastHref, currentHref);
+
+                if (oldOnPopState) {
+                    return oldOnPopState.apply(this, arguments);
+                }
+            };
+
+            fill(history, 'pushState', function (origPushState) {
+                // note history.pushState.length is 0; intentionally not declaring
+                // params to preserve 0 arity
+                return function (/* state, title, url */) {
+                    var url = arguments.length > 2 ? arguments[2] : undefined;
+
+                    // url argument is optional
+                    if (url) {
+                        // coerce to string (this is what pushState does)
+                        self._captureUrlChange(self._lastHref, url + '');
+                    }
+
+                    return origPushState.apply(this, arguments);
+                };
+            }, wrappedBuiltIns);
+        }
+
+        if (autoBreadcrumbs.console && 'console' in _window && console.log) {
+            // console
+            var consoleMethodCallback = function (msg, data) {
+                self.captureBreadcrumb({
+                    message: msg,
+                    level: data.level,
+                    category: 'console'
+                });
+            };
+
+            each(['debug', 'info', 'warn', 'error', 'log'], function (_, level) {
+                wrapConsoleMethod(console, level, consoleMethodCallback);
+            });
+        }
+
+    },
+
+    _restoreBuiltIns: function () {
+        // restore any wrapped builtins
+        var builtin;
+        while (this._wrappedBuiltIns.length) {
+            builtin = this._wrappedBuiltIns.shift();
+
+            var obj = builtin[0],
+              name = builtin[1],
+              orig = builtin[2];
+
+            obj[name] = orig;
+        }
+    },
+
+    _drainPlugins: function() {
+        var self = this;
+
+        // FIX ME TODO
+        each(this._plugins, function(_, plugin) {
+            var installer = plugin[0];
+            var args = plugin[1];
+            installer.apply(self, [self].concat(args));
+        });
+    },
+
+    _parseDSN: function(str) {
+        var m = dsnPattern.exec(str),
+            dsn = {},
+            i = 7;
+
+        try {
+            while (i--) dsn[dsnKeys[i]] = m[i] || '';
+        } catch(e) {
+            throw new RavenConfigError('Invalid DSN: ' + str);
+        }
+
+        if (dsn.pass && !this._globalOptions.allowSecretKey) {
+            throw new RavenConfigError('Do not specify your secret key in the DSN. See: http://bit.ly/raven-secret-key');
+        }
+
+        return dsn;
+    },
+
+    _getGlobalServer: function(uri) {
+        // assemble the endpoint from the uri pieces
+        var globalServer = '//' + uri.host +
+            (uri.port ? ':' + uri.port : '');
+
+        if (uri.protocol) {
+            globalServer = uri.protocol + ':' + globalServer;
+        }
+        return globalServer;
+    },
+
+    _handleOnErrorStackInfo: function() {
+        // if we are intentionally ignoring errors via onerror, bail out
+        if (!this._ignoreOnError) {
+            this._handleStackInfo.apply(this, arguments);
+        }
+    },
+
+    _handleStackInfo: function(stackInfo, options) {
+        var frames = this._prepareFrames(stackInfo, options);
+
+        this._triggerEvent('handle', {
+            stackInfo: stackInfo,
+            options: options
+        });
+
+        this._processException(
+            stackInfo.name,
+            stackInfo.message,
+            stackInfo.url,
+            stackInfo.lineno,
+            frames,
+            options
+        );
+    },
+
+    _prepareFrames: function(stackInfo, options) {
+        var self = this;
+        var frames = [];
+        if (stackInfo.stack && stackInfo.stack.length) {
+            each(stackInfo.stack, function(i, stack) {
+                var frame = self._normalizeFrame(stack);
+                if (frame) {
+                    frames.push(frame);
+                }
+            });
+
+            // e.g. frames captured via captureMessage throw
+            if (options && options.trimHeadFrames) {
+                for (var j = 0; j < options.trimHeadFrames && j < frames.length; j++) {
+                    frames[j].in_app = false;
+                }
+            }
+        }
+        frames = frames.slice(0, this._globalOptions.stackTraceLimit);
+        return frames;
+    },
+
+
+    _normalizeFrame: function(frame) {
+        if (!frame.url) return;
+
+        // normalize the frames data
+        var normalized = {
+            filename:   frame.url,
+            lineno:     frame.line,
+            colno:      frame.column,
+            'function': frame.func || '?'
+        };
+
+        normalized.in_app = !( // determine if an exception came from outside of our app
+            // first we check the global includePaths list.
+            !!this._globalOptions.includePaths.test && !this._globalOptions.includePaths.test(normalized.filename) ||
+            // Now we check for fun, if the function name is Raven or TraceKit
+            /(Raven|TraceKit)\./.test(normalized['function']) ||
+            // finally, we do a last ditch effort and check for raven.min.js
+            /raven\.(min\.)?js$/.test(normalized.filename)
+        );
+
+        return normalized;
+    },
+
+    _processException: function(type, message, fileurl, lineno, frames, options) {
+        var stacktrace;
+        if (!!this._globalOptions.ignoreErrors.test && this._globalOptions.ignoreErrors.test(message)) return;
+
+        message += '';
+
+        if (frames && frames.length) {
+            fileurl = frames[0].filename || fileurl;
+            // Sentry expects frames oldest to newest
+            // and JS sends them as newest to oldest
+            frames.reverse();
+            stacktrace = {frames: frames};
+        } else if (fileurl) {
+            stacktrace = {
+                frames: [{
+                    filename: fileurl,
+                    lineno: lineno,
+                    in_app: true
+                }]
+            };
+        }
+
+        if (!!this._globalOptions.ignoreUrls.test && this._globalOptions.ignoreUrls.test(fileurl)) return;
+        if (!!this._globalOptions.whitelistUrls.test && !this._globalOptions.whitelistUrls.test(fileurl)) return;
+
+        var data = objectMerge({
+            // sentry.interfaces.Exception
+            exception: {
+                values: [{
+                    type: type,
+                    value: message,
+                    stacktrace: stacktrace
+                }]
+            },
+            culprit: fileurl
+        }, options);
+
+        // Fire away!
+        this._send(data);
+    },
+
+    _trimPacket: function(data) {
+        // For now, we only want to truncate the two different messages
+        // but this could/should be expanded to just trim everything
+        var max = this._globalOptions.maxMessageLength;
+        if (data.message) {
+            data.message = truncate(data.message, max);
+        }
+        if (data.exception) {
+            var exception = data.exception.values[0];
+            exception.value = truncate(exception.value, max);
+        }
+
+        return data;
+    },
+
+    _getHttpData: function() {
+        if (!this._hasDocument || !_document.location || !_document.location.href) {
+            return;
+        }
+
+        var httpData = {
+            headers: {
+                'User-Agent': navigator.userAgent
+            }
+        };
+
+        httpData.url = _document.location.href;
+
+        if (_document.referrer) {
+            httpData.headers.Referer = _document.referrer;
+        }
+
+        return httpData;
+    },
+
+
+    _send: function(data) {
+        var globalOptions = this._globalOptions;
+
+        var baseData = {
+            project: this._globalProject,
+            logger: globalOptions.logger,
+            platform: 'javascript'
+        }, httpData = this._getHttpData();
+
+        if (httpData) {
+            baseData.request = httpData;
+        }
+
+        // HACK: delete `trimHeadFrames` to prevent from appearing in outbound payload
+        if (data.trimHeadFrames) delete data.trimHeadFrames;
+
+        data = objectMerge(baseData, data);
+
+        // Merge in the tags and extra separately since objectMerge doesn't handle a deep merge
+        data.tags = objectMerge(objectMerge({}, this._globalContext.tags), data.tags);
+        data.extra = objectMerge(objectMerge({}, this._globalContext.extra), data.extra);
+
+        // Send along our own collected metadata with extra
+        data.extra['session:duration'] = now() - this._startTime;
+
+        if (this._breadcrumbs && this._breadcrumbs.length > 0) {
+            // intentionally make shallow copy so that additions
+            // to breadcrumbs aren't accidentally sent in this request
+            data.breadcrumbs = {
+                values: [].slice.call(this._breadcrumbs, 0)
+            };
+        }
+
+        // If there are no tags/extra, strip the key from the payload alltogther.
+        if (isEmptyObject(data.tags)) delete data.tags;
+
+        if (this._globalContext.user) {
+            // sentry.interfaces.User
+            data.user = this._globalContext.user;
+        }
+
+        // Include the environment if it's defined in globalOptions
+        if (globalOptions.environment) data.environment = globalOptions.environment;
+
+        // Include the release if it's defined in globalOptions
+        if (globalOptions.release) data.release = globalOptions.release;
+
+        // Include server_name if it's defined in globalOptions
+        if (globalOptions.serverName) data.server_name = globalOptions.serverName;
+
+        if (isFunction(globalOptions.dataCallback)) {
+            data = globalOptions.dataCallback(data) || data;
+        }
+
+        // Why??????????
+        if (!data || isEmptyObject(data)) {
+            return;
+        }
+
+        // Check if the request should be filtered or not
+        if (isFunction(globalOptions.shouldSendCallback) && !globalOptions.shouldSendCallback(data)) {
+            return;
+        }
+
+        this._sendProcessedPayload(data);
+    },
+
+    _getUuid: function () {
+      return uuid4();
+    },
+
+    _sendProcessedPayload: function(data, callback) {
+        var self = this;
+        var globalOptions = this._globalOptions;
+
+        // Send along an event_id if not explicitly passed.
+        // This event_id can be used to reference the error within Sentry itself.
+        // Set lastEventId after we know the error should actually be sent
+        this._lastEventId = data.event_id || (data.event_id = this._getUuid());
+
+        // Try and clean up the packet before sending by truncating long values
+        data = this._trimPacket(data);
+
+        this._logDebug('debug', 'Raven about to send:', data);
+
+        if (!this.isSetup()) return;
+
+        var auth = {
+            sentry_version: '7',
+            sentry_client: 'raven-js/' + this.VERSION,
+            sentry_key: this._globalKey
+        };
+        if (this._globalSecret) {
+            auth.sentry_secret = this._globalSecret;
+        }
+
+        var exception = data.exception && data.exception.values[0];
+        this.captureBreadcrumb({
+            category: 'sentry',
+            message: exception
+                ? (exception.type ? exception.type + ': ' : '') + exception.value
+                : data.message,
+            event_id: data.event_id,
+            level: data.level || 'error' // presume error unless specified
+        });
+
+        var url = this._globalEndpoint;
+        (globalOptions.transport || this._makeRequest).call(this, {
+            url: url,
+            auth: auth,
+            data: data,
+            options: globalOptions,
+            onSuccess: function success() {
+                self._triggerEvent('success', {
+                    data: data,
+                    src: url
+                });
+                callback && callback();
+            },
+            onError: function failure(error) {
+                self._triggerEvent('failure', {
+                    data: data,
+                    src: url
+                });
+                error = error || new Error('Raven send failed (no additional details provided)');
+                callback && callback(error);
+            }
+        });
+    },
+
+    _makeRequest: function(opts) {
+        var request = new XMLHttpRequest();
+
+        // if browser doesn't support CORS (e.g. IE7), we are out of luck
+        var hasCORS =
+            'withCredentials' in request ||
+            typeof XDomainRequest !== 'undefined';
+
+        if (!hasCORS) return;
+
+        var url = opts.url;
+        function handler() {
+            if (request.status === 200) {
+                if (opts.onSuccess) {
+                    opts.onSuccess();
+                }
+            } else if (opts.onError) {
+                opts.onError(new Error('Sentry error code: ' + request.status));
+            }
+        }
+
+        if ('withCredentials' in request) {
+            request.onreadystatechange = function () {
+                if (request.readyState !== 4) {
+                    return;
+                }
+                handler();
+            };
+        } else {
+            request = new XDomainRequest();
+            // xdomainrequest cannot go http -> https (or vice versa),
+            // so always use protocol relative
+            url = url.replace(/^https?:/, '');
+
+            // onreadystatechange not supported by XDomainRequest
+            request.onload = handler;
+        }
+
+        // NOTE: auth is intentionally sent as part of query string (NOT as custom
+        //       HTTP header) so as to avoid preflight CORS requests
+        request.open('POST', url + '?' + urlencode(opts.auth));
+        request.send(stringify(opts.data));
+    },
+
+    _logDebug: function(level) {
+        if (this._originalConsoleMethods[level] && this.debug) {
+            // In IE<10 console methods do not have their own 'apply' method
+            Function.prototype.apply.call(
+                this._originalConsoleMethods[level],
+                this._originalConsole,
+                [].slice.call(arguments, 1)
+            );
+        }
+    },
+
+    _mergeContext: function(key, context) {
+        if (isUndefined(context)) {
+            delete this._globalContext[key];
+        } else {
+            this._globalContext[key] = objectMerge(this._globalContext[key] || {}, context);
+        }
+    }
+};
+
+/*------------------------------------------------
+ * utils
+ *
+ * conditionally exported for test via Raven.utils
+ =================================================
+ */
+var objectPrototype = Object.prototype;
+
+function isUndefined(what) {
+    return what === void 0;
+}
+
+function isFunction(what) {
+    return typeof what === 'function';
+}
+
+function isString(what) {
+    return objectPrototype.toString.call(what) === '[object String]';
+}
+
+function isObject(what) {
+    return typeof what === 'object' && what !== null;
+}
+
+function isEmptyObject(what) {
+    for (var _ in what) return false;  // eslint-disable-line guard-for-in, no-unused-vars
+    return true;
+}
+
+// Sorta yanked from https://github.com/joyent/node/blob/aa3b4b4/lib/util.js#L560
+// with some tiny modifications
+function isError(what) {
+    var toString = objectPrototype.toString.call(what);
+    return isObject(what) &&
+        toString === '[object Error]' ||
+        toString === '[object Exception]' || // Firefox NS_ERROR_FAILURE Exceptions
+        what instanceof Error;
+}
+
+function each(obj, callback) {
+    var i, j;
+
+    if (isUndefined(obj.length)) {
+        for (i in obj) {
+            if (hasKey(obj, i)) {
+                callback.call(null, i, obj[i]);
+            }
+        }
+    } else {
+        j = obj.length;
+        if (j) {
+            for (i = 0; i < j; i++) {
+                callback.call(null, i, obj[i]);
+            }
+        }
+    }
+}
+
+function objectMerge(obj1, obj2) {
+    if (!obj2) {
+        return obj1;
+    }
+    each(obj2, function(key, value){
+        obj1[key] = value;
+    });
+    return obj1;
+}
+
+function truncate(str, max) {
+    return !max || str.length <= max ? str : str.substr(0, max) + '\u2026';
+}
+
+/**
+ * hasKey, a better form of hasOwnProperty
+ * Example: hasKey(MainHostObject, property) === true/false
+ *
+ * @param {Object} host object to check property
+ * @param {string} key to check
+ */
+function hasKey(object, key) {
+    return objectPrototype.hasOwnProperty.call(object, key);
+}
+
+function joinRegExp(patterns) {
+    // Combine an array of regular expressions and strings into one large regexp
+    // Be mad.
+    var sources = [],
+        i = 0, len = patterns.length,
+        pattern;
+
+    for (; i < len; i++) {
+        pattern = patterns[i];
+        if (isString(pattern)) {
+            // If it's a string, we need to escape it
+            // Taken from: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions
+            sources.push(pattern.replace(/([.*+?^=!:${}()|\[\]\/\\])/g, '\\$1'));
+        } else if (pattern && pattern.source) {
+            // If it's a regexp already, we want to extract the source
+            sources.push(pattern.source);
+        }
+        // Intentionally skip other cases
+    }
+    return new RegExp(sources.join('|'), 'i');
+}
+
+function urlencode(o) {
+    var pairs = [];
+    each(o, function(key, value) {
+        pairs.push(encodeURIComponent(key) + '=' + encodeURIComponent(value));
+    });
+    return pairs.join('&');
+}
+
+// borrowed from https://tools.ietf.org/html/rfc3986#appendix-B
+// intentionally using regex and not <a/> href parsing trick because React Native and other
+// environments where DOM might not be available
+function parseUrl(url) {
+    var match = url.match(/^(([^:\/?#]+):)?(\/\/([^\/?#]*))?([^?#]*)(\?([^#]*))?(#(.*))?$/);
+    if (!match) return {};
+
+    // coerce to undefined values to empty string so we don't get 'undefined'
+    var query = match[6] || '';
+    var fragment = match[8] || '';
+    return {
+        protocol: match[2],
+        host: match[4],
+        path: match[5],
+        relative: match[5] + query + fragment // everything minus origin
+    };
+}
+function uuid4() {
+    var crypto = _window.crypto || _window.msCrypto;
+
+    if (!isUndefined(crypto) && crypto.getRandomValues) {
+        // Use window.crypto API if available
+        var arr = new Uint16Array(8);
+        crypto.getRandomValues(arr);
+
+        // set 4 in byte 7
+        arr[3] = arr[3] & 0xFFF | 0x4000;
+        // set 2 most significant bits of byte 9 to '10'
+        arr[4] = arr[4] & 0x3FFF | 0x8000;
+
+        var pad = function(num) {
+            var v = num.toString(16);
+            while (v.length < 4) {
+                v = '0' + v;
+            }
+            return v;
+        };
+
+        return pad(arr[0]) + pad(arr[1]) + pad(arr[2]) + pad(arr[3]) + pad(arr[4]) +
+        pad(arr[5]) + pad(arr[6]) + pad(arr[7]);
+    } else {
+        // http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#2117523
+        return 'xxxxxxxxxxxx4xxxyxxxxxxxxxxxxxxx'.replace(/[xy]/g, function(c) {
+            var r = Math.random()*16|0,
+                v = c === 'x' ? r : r&0x3|0x8;
+            return v.toString(16);
+        });
+    }
+}
+
+/**
+ * Given a child DOM element, returns a query-selector statement describing that
+ * and its ancestors
+ * e.g. [HTMLElement] => body > div > input#foo.btn[name=baz]
+ * @param elem
+ * @returns {string}
+ */
+function htmlTreeAsString(elem) {
+    /* eslint no-extra-parens:0*/
+    var MAX_TRAVERSE_HEIGHT = 5,
+        MAX_OUTPUT_LEN = 80,
+        out = [],
+        height = 0,
+        len = 0,
+        separator = ' > ',
+        sepLength = separator.length,
+        nextStr;
+
+    while (elem && height++ < MAX_TRAVERSE_HEIGHT) {
+
+        nextStr = htmlElementAsString(elem);
+        // bail out if
+        // - nextStr is the 'html' element
+        // - the length of the string that would be created exceeds MAX_OUTPUT_LEN
+        //   (ignore this limit if we are on the first iteration)
+        if (nextStr === 'html' || height > 1 && len + (out.length * sepLength) + nextStr.length >= MAX_OUTPUT_LEN) {
+            break;
+        }
+
+        out.push(nextStr);
+
+        len += nextStr.length;
+        elem = elem.parentNode;
+    }
+
+    return out.reverse().join(separator);
+}
+
+/**
+ * Returns a simple, query-selector representation of a DOM element
+ * e.g. [HTMLElement] => input#foo.btn[name=baz]
+ * @param HTMLElement
+ * @returns {string}
+ */
+function htmlElementAsString(elem) {
+    var out = [],
+        className,
+        classes,
+        key,
+        attr,
+        i;
+
+    if (!elem || !elem.tagName) {
+        return '';
+    }
+
+    out.push(elem.tagName.toLowerCase());
+    if (elem.id) {
+        out.push('#' + elem.id);
+    }
+
+    className = elem.className;
+    if (className && isString(className)) {
+        classes = className.split(' ');
+        for (i = 0; i < classes.length; i++) {
+            out.push('.' + classes[i]);
+        }
+    }
+    var attrWhitelist = ['type', 'name', 'title', 'alt'];
+    for (i = 0; i < attrWhitelist.length; i++) {
+        key = attrWhitelist[i];
+        attr = elem.getAttribute(key);
+        if (attr) {
+            out.push('[' + key + '="' + attr + '"]');
+        }
+    }
+    return out.join('');
+}
+
+/**
+ * Polyfill a method
+ * @param obj object e.g. `document`
+ * @param name method name present on object e.g. `addEventListener`
+ * @param replacement replacement function
+ * @param track {optional} record instrumentation to an array
+ */
+function fill(obj, name, replacement, track) {
+    var orig = obj[name];
+    obj[name] = replacement(orig);
+    if (track) {
+        track.push([obj, name, orig]);
+    }
+}
+
+if (typeof __DEV__ !== 'undefined' && __DEV__) {
+    Raven.utils = {
+        isUndefined: isUndefined,
+        isFunction: isFunction,
+        isString: isString,
+        isObject: isObject,
+        isEmptyObject: isEmptyObject,
+        isError: isError,
+        each: each,
+        objectMerge: objectMerge,
+        truncate: truncate,
+        hasKey: hasKey,
+        joinRegExp: joinRegExp,
+        urlencode: urlencode,
+        uuid4: uuid4,
+        htmlTreeAsString: htmlTreeAsString,
+        htmlElementAsString: htmlElementAsString,
+        parseUrl: parseUrl,
+        fill: fill
+    };
+};
+
+// Deprecations
+Raven.prototype.setUser = Raven.prototype.setUserContext;
+Raven.prototype.setReleaseContext = Raven.prototype.setRelease;
+
+module.exports = Raven;
+
+}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
+},{"1":1,"2":2,"3":3,"6":6}],5:[function(_dereq_,module,exports){
+(function (global){
+/**
+ * Enforces a single instance of the Raven client, and the
+ * main entry point for Raven. If you are a consumer of the
+ * Raven library, you SHOULD load this file (vs raven.js).
+ **/
+
+'use strict';
+
+var RavenConstructor = _dereq_(4);
+
+// This is to be defensive in environments where window does not exist (see https://github.com/getsentry/raven-js/pull/785)
+var _window = typeof window !== 'undefined' ? window
+            : typeof global !== 'undefined' ? global
+            : typeof self !== 'undefined' ? self
+            : {};
+var _Raven = _window.Raven;
+
+var Raven = new RavenConstructor();
+
+/*
+ * Allow multiple versions of Raven to be installed.
+ * Strip Raven from the global context and returns the instance.
+ *
+ * @return {Raven}
+ */
+Raven.noConflict = function () {
+	_window.Raven = _Raven;
+	return Raven;
+};
+
+Raven.afterLoad();
+
+module.exports = Raven;
+
+}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
+},{"4":4}],6:[function(_dereq_,module,exports){
+(function (global){
+'use strict';
+
+/*
+ TraceKit - Cross brower stack traces - github.com/occ/TraceKit
+ MIT license
+*/
+
+var TraceKit = {
+    collectWindowErrors: true,
+    debug: false
+};
+
+// This is to be defensive in environments where window does not exist (see https://github.com/getsentry/raven-js/pull/785)
+var _window = typeof window !== 'undefined' ? window
+            : typeof global !== 'undefined' ? global
+            : typeof self !== 'undefined' ? self
+            : {};
+
+// global reference to slice
+var _slice = [].slice;
+var UNKNOWN_FUNCTION = '?';
+
+// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Error#Error_types
+var ERROR_TYPES_RE = /^(?:Uncaught (?:exception: )?)?((?:Eval|Internal|Range|Reference|Syntax|Type|URI)Error): ?(.*)$/;
+
+function getLocationHref() {
+    if (typeof document === 'undefined')
+        return '';
+
+    return document.location.href;
+}
+
+/**
+ * TraceKit.report: cross-browser processing of unhandled exceptions
+ *
+ * Syntax:
+ *   TraceKit.report.subscribe(function(stackInfo) { ... })
+ *   TraceKit.report.unsubscribe(function(stackInfo) { ... })
+ *   TraceKit.report(exception)
+ *   try { ...code... } catch(ex) { TraceKit.report(ex); }
+ *
+ * Supports:
+ *   - Firefox: full stack trace with line numbers, plus column number
+ *              on top frame; column number is not guaranteed
+ *   - Opera:   full stack trace with line and column numbers
+ *   - Chrome:  full stack trace with line and column numbers
+ *   - Safari:  line and column number for the top frame only; some frames
+ *              may be missing, and column number is not guaranteed
+ *   - IE:      line and column number for the top frame only; some frames
+ *              may be missing, and column number is not guaranteed
+ *
+ * In theory, TraceKit should work on all of the following versions:
+ *   - IE5.5+ (only 8.0 tested)
+ *   - Firefox 0.9+ (only 3.5+ tested)
+ *   - Opera 7+ (only 10.50 tested; versions 9 and earlier may require
+ *     Exceptions Have Stacktrace to be enabled in opera:config)
+ *   - Safari 3+ (only 4+ tested)
+ *   - Chrome 1+ (only 5+ tested)
+ *   - Konqueror 3.5+ (untested)
+ *
+ * Requires TraceKit.computeStackTrace.
+ *
+ * Tries to catch all unhandled exceptions and report them to the
+ * subscribed handlers. Please note that TraceKit.report will rethrow the
+ * exception. This is REQUIRED in order to get a useful stack trace in IE.
+ * If the exception does not reach the top of the browser, you will only
+ * get a stack trace from the point where TraceKit.report was called.
+ *
+ * Handlers receive a stackInfo object as described in the
+ * TraceKit.computeStackTrace docs.
+ */
+TraceKit.report = (function reportModuleWrapper() {
+    var handlers = [],
+        lastArgs = null,
+        lastException = null,
+        lastExceptionStack = null;
+
+    /**
+     * Add a crash handler.
+     * @param {Function} handler
+     */
+    function subscribe(handler) {
+        installGlobalHandler();
+        handlers.push(handler);
+    }
+
+    /**
+     * Remove a crash handler.
+     * @param {Function} handler
+     */
+    function unsubscribe(handler) {
+        for (var i = handlers.length - 1; i >= 0; --i) {
+            if (handlers[i] === handler) {
+                handlers.splice(i, 1);
+            }
+        }
+    }
+
+    /**
+     * Remove all crash handlers.
+     */
+    function unsubscribeAll() {
+        uninstallGlobalHandler();
+        handlers = [];
+    }
+
+    /**
+     * Dispatch stack information to all handlers.
+     * @param {Object.<string, *>} stack
+     */
+    function notifyHandlers(stack, isWindowError) {
+        var exception = null;
+        if (isWindowError && !TraceKit.collectWindowErrors) {
+          return;
+        }
+        for (var i in handlers) {
+            if (handlers.hasOwnProperty(i)) {
+                try {
+                    handlers[i].apply(null, [stack].concat(_slice.call(arguments, 2)));
+                } catch (inner) {
+                    exception = inner;
+                }
+            }
+        }
+
+        if (exception) {
+            throw exception;
+        }
+    }
+
+    var _oldOnerrorHandler, _onErrorHandlerInstalled;
+
+    /**
+     * Ensures all global unhandled exceptions are recorded.
+     * Supported by Gecko and IE.
+     * @param {string} message Error message.
+     * @param {string} url URL of script that generated the exception.
+     * @param {(number|string)} lineNo The line number at which the error
+     * occurred.
+     * @param {?(number|string)} colNo The column number at which the error
+     * occurred.
+     * @param {?Error} ex The actual Error object.
+     */
+    function traceKitWindowOnError(message, url, lineNo, colNo, ex) {
+        var stack = null;
+
+        if (lastExceptionStack) {
+            TraceKit.computeStackTrace.augmentStackTraceWithInitialElement(lastExceptionStack, url, lineNo, message);
+            processLastException();
+        } else if (ex) {
+            // New chrome and blink send along a real error object
+            // Let's just report that like a normal error.
+            // See: https://mikewest.org/2013/08/debugging-runtime-errors-with-window-onerror
+            stack = TraceKit.computeStackTrace(ex);
+            notifyHandlers(stack, true);
+        } else {
+            var location = {
+                'url': url,
+                'line': lineNo,
+                'column': colNo
+            };
+
+            var name = undefined;
+            var msg = message; // must be new var or will modify original `arguments`
+            var groups;
+            if ({}.toString.call(message) === '[object String]') {
+                var groups = message.match(ERROR_TYPES_RE);
+                if (groups) {
+                    name = groups[1];
+                    msg = groups[2];
+                }
+            }
+
+            location.func = UNKNOWN_FUNCTION;
+
+            stack = {
+                'name': name,
+                'message': msg,
+                'url': getLocationHref(),
+                'stack': [location]
+            };
+            notifyHandlers(stack, true);
+        }
+
+        if (_oldOnerrorHandler) {
+            return _oldOnerrorHandler.apply(this, arguments);
+        }
+
+        return false;
+    }
+
+    function installGlobalHandler ()
+    {
+        if (_onErrorHandlerInstalled) {
+            return;
+        }
+        _oldOnerrorHandler = _window.onerror;
+        _window.onerror = traceKitWindowOnError;
+        _onErrorHandlerInstalled = true;
+    }
+
+    function uninstallGlobalHandler ()
+    {
+        if (!_onErrorHandlerInstalled) {
+            return;
+        }
+        _window.onerror = _oldOnerrorHandler;
+        _onErrorHandlerInstalled = false;
+        _oldOnerrorHandler = undefined;
+    }
+
+    function processLastException() {
+        var _lastExceptionStack = lastExceptionStack,
+            _lastArgs = lastArgs;
+        lastArgs = null;
+        lastExceptionStack = null;
+        lastException = null;
+        notifyHandlers.apply(null, [_lastExceptionStack, false].concat(_lastArgs));
+    }
+
+    /**
+     * Reports an unhandled Error to TraceKit.
+     * @param {Error} ex
+     * @param {?boolean} rethrow If false, do not re-throw the exception.
+     * Only used for window.onerror to not cause an infinite loop of
+     * rethrowing.
+     */
+    function report(ex, rethrow) {
+        var args = _slice.call(arguments, 1);
+        if (lastExceptionStack) {
+            if (lastException === ex) {
+                return; // already caught by an inner catch block, ignore
+            } else {
+              processLastException();
+            }
+        }
+
+        var stack = TraceKit.computeStackTrace(ex);
+        lastExceptionStack = stack;
+        lastException = ex;
+        lastArgs = args;
+
+        // If the stack trace is incomplete, wait for 2 seconds for
+        // slow slow IE to see if onerror occurs or not before reporting
+        // this exception; otherwise, we will end up with an incomplete
+        // stack trace
+        setTimeout(function () {
+            if (lastException === ex) {
+                processLastException();
+            }
+        }, (stack.incomplete ? 2000 : 0));
+
+        if (rethrow !== false) {
+            throw ex; // re-throw to propagate to the top level (and cause window.onerror)
+        }
+    }
+
+    report.subscribe = subscribe;
+    report.unsubscribe = unsubscribe;
+    report.uninstall = unsubscribeAll;
+    return report;
+}());
+
+/**
+ * TraceKit.computeStackTrace: cross-browser stack traces in JavaScript
+ *
+ * Syntax:
+ *   s = TraceKit.computeStackTrace(exception) // consider using TraceKit.report instead (see below)
+ * Returns:
+ *   s.name              - exception name
+ *   s.message           - exception message
+ *   s.stack[i].url      - JavaScript or HTML file URL
+ *   s.stack[i].func     - function name, or empty for anonymous functions (if guessing did not work)
+ *   s.stack[i].args     - arguments passed to the function, if known
+ *   s.stack[i].line     - line number, if known
+ *   s.stack[i].column   - column number, if known
+ *
+ * Supports:
+ *   - Firefox:  full stack trace with line numbers and unreliable column
+ *               number on top frame
+ *   - Opera 10: full stack trace with line and column numbers
+ *   - Opera 9-: full stack trace with line numbers
+ *   - Chrome:   full stack trace with line and column numbers
+ *   - Safari:   line and column number for the topmost stacktrace element
+ *               only
+ *   - IE:       no line numbers whatsoever
+ *
+ * Tries to guess names of anonymous functions by looking for assignments
+ * in the source code. In IE and Safari, we have to guess source file names
+ * by searching for function bodies inside all page scripts. This will not
+ * work for scripts that are loaded cross-domain.
+ * Here be dragons: some function names may be guessed incorrectly, and
+ * duplicate functions may be mismatched.
+ *
+ * TraceKit.computeStackTrace should only be used for tracing purposes.
+ * Logging of unhandled exceptions should be done with TraceKit.report,
+ * which builds on top of TraceKit.computeStackTrace and provides better
+ * IE support by utilizing the window.onerror event to retrieve information
+ * about the top of the stack.
+ *
+ * Note: In IE and Safari, no stack trace is recorded on the Error object,
+ * so computeStackTrace instead walks its *own* chain of callers.
+ * This means that:
+ *  * in Safari, some methods may be missing from the stack trace;
+ *  * in IE, the topmost function in the stack trace will always be the
+ *    caller of computeStackTrace.
+ *
+ * This is okay for tracing (because you are likely to be calling
+ * computeStackTrace from the function you want to be the topmost element
+ * of the stack trace anyway), but not okay for logging unhandled
+ * exceptions (because your catch block will likely be far away from the
+ * inner function that actually caused the exception).
+ *
+ */
+TraceKit.computeStackTrace = (function computeStackTraceWrapper() {
+    /**
+     * Escapes special characters, except for whitespace, in a string to be
+     * used inside a regular expression as a string literal.
+     * @param {string} text The string.
+     * @return {string} The escaped string literal.
+     */
+    function escapeRegExp(text) {
+        return text.replace(/[\-\[\]{}()*+?.,\\\^$|#]/g, '\\$&');
+    }
+
+    /**
+     * Escapes special characters in a string to be used inside a regular
+     * expression as a string literal. Also ensures that HTML entities will
+     * be matched the same as their literal friends.
+     * @param {string} body The string.
+     * @return {string} The escaped string.
+     */
+    function escapeCodeAsRegExpForMatchingInsideHTML(body) {
+        return escapeRegExp(body).replace('<', '(?:<|&lt;)').replace('>', '(?:>|&gt;)').replace('&', '(?:&|&amp;)').replace('"', '(?:"|&quot;)').replace(/\s+/g, '\\s+');
+    }
+
+    // Contents of Exception in various browsers.
+    //
+    // SAFARI:
+    // ex.message = Can't find variable: qq
+    // ex.line = 59
+    // ex.sourceId = 580238192
+    // ex.sourceURL = http://...
+    // ex.expressionBeginOffset = 96
+    // ex.expressionCaretOffset = 98
+    // ex.expressionEndOffset = 98
+    // ex.name = ReferenceError
+    //
+    // FIREFOX:
+    // ex.message = qq is not defined
+    // ex.fileName = http://...
+    // ex.lineNumber = 59
+    // ex.columnNumber = 69
+    // ex.stack = ...stack trace... (see the example below)
+    // ex.name = ReferenceError
+    //
+    // CHROME:
+    // ex.message = qq is not defined
+    // ex.name = ReferenceError
+    // ex.type = not_defined
+    // ex.arguments = ['aa']
+    // ex.stack = ...stack trace...
+    //
+    // INTERNET EXPLORER:
+    // ex.message = ...
+    // ex.name = ReferenceError
+    //
+    // OPERA:
+    // ex.message = ...message... (see the example below)
+    // ex.name = ReferenceError
+    // ex.opera#sourceloc = 11  (pretty much useless, duplicates the info in ex.message)
+    // ex.stacktrace = n/a; see 'opera:config#UserPrefs|Exceptions Have Stacktrace'
+
+    /**
+     * Computes stack trace information from the stack property.
+     * Chrome and Gecko use this property.
+     * @param {Error} ex
+     * @return {?Object.<string, *>} Stack trace information.
+     */
+    function computeStackTraceFromStackProp(ex) {
+        if (typeof ex.stack === 'undefined' || !ex.stack) return;
+
+        var chrome = /^\s*at (.*?) ?\(((?:file|https?|blob|chrome-extension|native|eval|<anonymous>).*?)(?::(\d+))?(?::(\d+))?\)?\s*$/i,
+            gecko = /^\s*(.*?)(?:\((.*?)\))?(?:^|@)((?:file|https?|blob|chrome|\[native).*?)(?::(\d+))?(?::(\d+))?\s*$/i,
+            winjs = /^\s*at (?:((?:\[object object\])?.+) )?\(?((?:file|ms-appx|https?|blob):.*?):(\d+)(?::(\d+))?\)?\s*$/i,
+            lines = ex.stack.split('\n'),
+            stack = [],
+            parts,
+            element,
+            reference = /^(.*) is undefined$/.exec(ex.message);
+
+        for (var i = 0, j = lines.length; i < j; ++i) {
+            if ((parts = chrome.exec(lines[i]))) {
+                var isNative = parts[2] && parts[2].indexOf('native') !== -1;
+                element = {
+                    'url': !isNative ? parts[2] : null,
+                    'func': parts[1] || UNKNOWN_FUNCTION,
+                    'args': isNative ? [parts[2]] : [],
+                    'line': parts[3] ? +parts[3] : null,
+                    'column': parts[4] ? +parts[4] : null
+                };
+            } else if ( parts = winjs.exec(lines[i]) ) {
+                element = {
+                    'url': parts[2],
+                    'func': parts[1] || UNKNOWN_FUNCTION,
+                    'args': [],
+                    'line': +parts[3],
+                    'column': parts[4] ? +parts[4] : null
+                };
+            } else if ((parts = gecko.exec(lines[i]))) {
+                element = {
+                    'url': parts[3],
+                    'func': parts[1] || UNKNOWN_FUNCTION,
+                    'args': parts[2] ? parts[2].split(',') : [],
+                    'line': parts[4] ? +parts[4] : null,
+                    'column': parts[5] ? +parts[5] : null
+                };
+            } else {
+                continue;
+            }
+
+            if (!element.func && element.line) {
+                element.func = UNKNOWN_FUNCTION;
+            }
+
+            stack.push(element);
+        }
+
+        if (!stack.length) {
+            return null;
+        }
+
+        if (!stack[0].column && typeof ex.columnNumber !== 'undefined') {
+            // FireFox uses this awesome columnNumber property for its top frame
+            // Also note, Firefox's column number is 0-based and everything else expects 1-based,
+            // so adding 1
+            stack[0].column = ex.columnNumber + 1;
+        }
+
+        return {
+            'name': ex.name,
+            'message': ex.message,
+            'url': getLocationHref(),
+            'stack': stack
+        };
+    }
+
+    /**
+     * Adds information about the first frame to incomplete stack traces.
+     * Safari and IE require this to get complete data on the first frame.
+     * @param {Object.<string, *>} stackInfo Stack trace information from
+     * one of the compute* methods.
+     * @param {string} url The URL of the script that caused an error.
+     * @param {(number|string)} lineNo The line number of the script that
+     * caused an error.
+     * @param {string=} message The error generated by the browser, which
+     * hopefully contains the name of the object that caused the error.
+     * @return {boolean} Whether or not the stack information was
+     * augmented.
+     */
+    function augmentStackTraceWithInitialElement(stackInfo, url, lineNo, message) {
+        var initial = {
+            'url': url,
+            'line': lineNo
+        };
+
+        if (initial.url && initial.line) {
+            stackInfo.incomplete = false;
+
+            if (!initial.func) {
+                initial.func = UNKNOWN_FUNCTION;
+            }
+
+            if (stackInfo.stack.length > 0) {
+                if (stackInfo.stack[0].url === initial.url) {
+                    if (stackInfo.stack[0].line === initial.line) {
+                        return false; // already in stack trace
+                    } else if (!stackInfo.stack[0].line && stackInfo.stack[0].func === initial.func) {
+                        stackInfo.stack[0].line = initial.line;
+                        return false;
+                    }
+                }
+            }
+
+            stackInfo.stack.unshift(initial);
+            stackInfo.partial = true;
+            return true;
+        } else {
+            stackInfo.incomplete = true;
+        }
+
+        return false;
+    }
+
+    /**
+     * Computes stack trace information by walking the arguments.caller
+     * chain at the time the exception occurred. This will cause earlier
+     * frames to be missed but is the only way to get any stack trace in
+     * Safari and IE. The top frame is restored by
+     * {@link augmentStackTraceWithInitialElement}.
+     * @param {Error} ex
+     * @return {?Object.<string, *>} Stack trace information.
+     */
+    function computeStackTraceByWalkingCallerChain(ex, depth) {
+        var functionName = /function\s+([_$a-zA-Z\xA0-\uFFFF][_$a-zA-Z0-9\xA0-\uFFFF]*)?\s*\(/i,
+            stack = [],
+            funcs = {},
+            recursion = false,
+            parts,
+            item,
+            source;
+
+        for (var curr = computeStackTraceByWalkingCallerChain.caller; curr && !recursion; curr = curr.caller) {
+            if (curr === computeStackTrace || curr === TraceKit.report) {
+                // console.log('skipping internal function');
+                continue;
+            }
+
+            item = {
+                'url': null,
+                'func': UNKNOWN_FUNCTION,
+                'line': null,
+                'column': null
+            };
+
+            if (curr.name) {
+                item.func = curr.name;
+            } else if ((parts = functionName.exec(curr.toString()))) {
+                item.func = parts[1];
+            }
+
+            if (typeof item.func === 'undefined') {
+              try {
+                item.func = parts.input.substring(0, parts.input.indexOf('{'));
+              } catch (e) { }
+            }
+
+            if (funcs['' + curr]) {
+                recursion = true;
+            }else{
+                funcs['' + curr] = true;
+            }
+
+            stack.push(item);
+        }
+
+        if (depth) {
+            // console.log('depth is ' + depth);
+            // console.log('stack is ' + stack.length);
+            stack.splice(0, depth);
+        }
+
+        var result = {
+            'name': ex.name,
+            'message': ex.message,
+            'url': getLocationHref(),
+            'stack': stack
+        };
+        augmentStackTraceWithInitialElement(result, ex.sourceURL || ex.fileName, ex.line || ex.lineNumber, ex.message || ex.description);
+        return result;
+    }
+
+    /**
+     * Computes a stack trace for an exception.
+     * @param {Error} ex
+     * @param {(string|number)=} depth
+     */
+    function computeStackTrace(ex, depth) {
+        var stack = null;
+        depth = (depth == null ? 0 : +depth);
+
+        try {
+            stack = computeStackTraceFromStackProp(ex);
+            if (stack) {
+                return stack;
+            }
+        } catch (e) {
+            if (TraceKit.debug) {
+                throw e;
+            }
+        }
+
+        try {
+            stack = computeStackTraceByWalkingCallerChain(ex, depth + 1);
+            if (stack) {
+                return stack;
+            }
+        } catch (e) {
+            if (TraceKit.debug) {
+                throw e;
+            }
+        }
+
+        return {
+            'name': ex.name,
+            'message': ex.message,
+            'url': getLocationHref()
+        };
+    }
+
+    computeStackTrace.augmentStackTraceWithInitialElement = augmentStackTraceWithInitialElement;
+    computeStackTrace.computeStackTraceFromStackProp = computeStackTraceFromStackProp;
+
+    return computeStackTrace;
+}());
+
+module.exports = TraceKit;
+
+}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
+},{}]},{},[5])(5)
+});
-- 
GitLab


From e0dc73527a478188cfa28b456b64798639aa73c9 Mon Sep 17 00:00:00 2001
From: Jarka Kadlecova <jarka@gitlab.com>
Date: Fri, 24 Mar 2017 11:11:36 +0100
Subject: [PATCH 002/363] Project deploy keys json end point

---
 .../projects/deploy_keys_controller.rb        |  9 ++-
 .../settings/deploy_keys_presenter.rb         | 11 ++++
 app/serializers/deploy_key_entity.rb          | 12 ++++
 app/serializers/deploy_key_serializer.rb      |  3 +
 app/serializers/project_entity.rb             | 12 ++++
 changelogs/unreleased/29667-deploy-keys.yml   |  4 ++
 .../projects/deploy_keys_controller_spec.rb   | 66 +++++++++++++++++++
 spec/serializers/deploy_key_entity_spec.rb    | 29 ++++++++
 8 files changed, 144 insertions(+), 2 deletions(-)
 create mode 100644 app/serializers/deploy_key_entity.rb
 create mode 100644 app/serializers/deploy_key_serializer.rb
 create mode 100644 app/serializers/project_entity.rb
 create mode 100644 changelogs/unreleased/29667-deploy-keys.yml
 create mode 100644 spec/controllers/projects/deploy_keys_controller_spec.rb
 create mode 100644 spec/serializers/deploy_key_entity_spec.rb

diff --git a/app/controllers/projects/deploy_keys_controller.rb b/app/controllers/projects/deploy_keys_controller.rb
index d0c44e297e3b4..a47e15a192bff 100644
--- a/app/controllers/projects/deploy_keys_controller.rb
+++ b/app/controllers/projects/deploy_keys_controller.rb
@@ -8,7 +8,12 @@ class Projects::DeployKeysController < Projects::ApplicationController
   layout "project_settings"
 
   def index
-    redirect_to_repository_settings(@project)
+    respond_to do |format|
+      format.html { redirect_to_repository_settings(@project) }
+      format.json do
+        render json: Projects::Settings::DeployKeysPresenter.new(@project, current_user: current_user).as_json
+      end
+    end
   end
 
   def new
@@ -19,7 +24,7 @@ def create
     @key = DeployKey.new(deploy_key_params.merge(user: current_user))
 
     unless @key.valid? && @project.deploy_keys << @key
-      flash[:alert] = @key.errors.full_messages.join(', ').html_safe      
+      flash[:alert] = @key.errors.full_messages.join(', ').html_safe
     end
     redirect_to_repository_settings(@project)
   end
diff --git a/app/presenters/projects/settings/deploy_keys_presenter.rb b/app/presenters/projects/settings/deploy_keys_presenter.rb
index 86ac513b3c07e..070b0c35e3673 100644
--- a/app/presenters/projects/settings/deploy_keys_presenter.rb
+++ b/app/presenters/projects/settings/deploy_keys_presenter.rb
@@ -48,6 +48,17 @@ def any_available_public_keys_enabled?
         available_public_keys.any?
       end
 
+      def as_json
+        serializer = DeployKeySerializer.new
+        opts = { user: current_user }
+
+        {
+          enabled_keys: serializer.represent(enabled_keys, opts),
+          available_project_keys: serializer.represent(available_project_keys, opts),
+          public_keys: serializer.represent(available_public_keys, opts)
+        }
+      end
+
       def to_partial_path
         'projects/deploy_keys/index'
       end
diff --git a/app/serializers/deploy_key_entity.rb b/app/serializers/deploy_key_entity.rb
new file mode 100644
index 0000000000000..cdedc2c7bd018
--- /dev/null
+++ b/app/serializers/deploy_key_entity.rb
@@ -0,0 +1,12 @@
+class DeployKeyEntity < Grape::Entity
+  expose :id
+  expose :user_id
+  expose :title
+  expose :fingerprint
+  expose :can_push
+  expose :created_at
+  expose :updated_at
+  expose :projects, using: ProjectEntity do |deploy_key|
+    deploy_key.projects.select { |project| options[:user].can?(:read_project, project) }
+  end
+end
diff --git a/app/serializers/deploy_key_serializer.rb b/app/serializers/deploy_key_serializer.rb
new file mode 100644
index 0000000000000..8f849eb88b717
--- /dev/null
+++ b/app/serializers/deploy_key_serializer.rb
@@ -0,0 +1,3 @@
+class DeployKeySerializer < BaseSerializer
+  entity DeployKeyEntity
+end
diff --git a/app/serializers/project_entity.rb b/app/serializers/project_entity.rb
new file mode 100644
index 0000000000000..6f8061f753095
--- /dev/null
+++ b/app/serializers/project_entity.rb
@@ -0,0 +1,12 @@
+class ProjectEntity < Grape::Entity
+  expose :id
+  expose :name
+
+  expose :full_path do |project|
+    project.full_path
+  end
+
+  expose :full_name do |project|
+    project.full_name
+  end
+end
diff --git a/changelogs/unreleased/29667-deploy-keys.yml b/changelogs/unreleased/29667-deploy-keys.yml
new file mode 100644
index 0000000000000..0f202ebf1ee56
--- /dev/null
+++ b/changelogs/unreleased/29667-deploy-keys.yml
@@ -0,0 +1,4 @@
+---
+title: Project deploy keys json end point
+merge_request:
+author:
diff --git a/spec/controllers/projects/deploy_keys_controller_spec.rb b/spec/controllers/projects/deploy_keys_controller_spec.rb
new file mode 100644
index 0000000000000..efe1a78415b7d
--- /dev/null
+++ b/spec/controllers/projects/deploy_keys_controller_spec.rb
@@ -0,0 +1,66 @@
+require 'spec_helper'
+
+describe Projects::DeployKeysController do
+  let(:project) { create(:project, :repository) }
+  let(:user) { create(:user) }
+
+  before do
+    project.team << [user, :master]
+
+    sign_in(user)
+  end
+
+  describe 'GET index' do
+    let(:params) do
+      { namespace_id: project.namespace, project_id: project }
+    end
+
+    context 'when html requested' do
+      it 'redirects to blob' do
+        get :index, params
+
+        expect(response).to redirect_to(namespace_project_settings_repository_path(params))
+      end
+    end
+
+    context 'when json requested' do
+      let(:project2) { create(:empty_project, :internal)}
+      let(:project_private) { create(:empty_project, :private)}
+
+      let(:deploy_key_internal) do
+        create(:deploy_key, key: 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQCdMHEHyhRjbhEZVddFn6lTWdgEy5Q6Bz4nwGB76xWZI5YT/1WJOMEW+sL5zYd31kk7sd3FJ5L9ft8zWMWrr/iWXQikC2cqZK24H1xy+ZUmrRuJD4qGAaIVoyyzBL+avL+lF8J5lg6YSw8gwJY/lX64/vnJHUlWw2n5BF8IFOWhiw== dummy@gitlab.com')
+      end
+      let(:deploy_key_actual) do
+        create(:deploy_key, key: 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQDNd/UJWhPrpb+b/G5oL109y57yKuCxE+WUGJGYaj7WQKsYRJmLYh1mgjrl+KVyfsWpq4ylOxIfFSnN9xBBFN8mlb0Fma5DC7YsSsibJr3MZ19ZNBprwNcdogET7aW9I0In7Wu5f2KqI6e5W/spJHCy4JVxzVMUvk6Myab0LnJ2iQ== dummy@gitlab.com')
+      end
+      let!(:deploy_key_public) { create(:deploy_key, public: true) }
+
+      let!(:deploy_keys_project_internal) do
+        create(:deploy_keys_project, project: project2, deploy_key: deploy_key_internal)
+      end
+
+      let!(:deploy_keys_actual_project) do
+        create(:deploy_keys_project, project: project, deploy_key: deploy_key_actual)
+      end
+
+      let!(:deploy_keys_project_private) do
+        create(:deploy_keys_project, project: project_private, deploy_key: create(:another_deploy_key))
+      end
+
+      before do
+        project2.team << [user, :developer]
+      end
+
+      it 'returns json in a correct format' do
+        get :index, params.merge(format: :json)
+
+        json = JSON.parse(response.body)
+
+        expect(json.keys).to match_array(%w(enabled_keys available_project_keys public_keys))
+        expect(json['enabled_keys'].count).to eq(1)
+        expect(json['available_project_keys'].count).to eq(1)
+        expect(json['public_keys'].count).to eq(1)
+      end
+    end
+  end
+end
diff --git a/spec/serializers/deploy_key_entity_spec.rb b/spec/serializers/deploy_key_entity_spec.rb
new file mode 100644
index 0000000000000..da1d33b42ac80
--- /dev/null
+++ b/spec/serializers/deploy_key_entity_spec.rb
@@ -0,0 +1,29 @@
+require 'spec_helper'
+
+describe DeployKeyEntity do
+  let(:user) { create(:user) }
+  let(:project) { create(:empty_project, :internal)}
+  let(:project_private) { create(:empty_project, :private)}
+  let(:deploy_key) { create(:deploy_key) }
+  let!(:deploy_key_internal) { create(:deploy_keys_project, project: project, deploy_key: deploy_key) }
+  let!(:deploy_key_private)  { create(:deploy_keys_project, project: project_private, deploy_key: deploy_key) }
+
+  let(:entity) { described_class.new(deploy_key, user: user) }
+
+  it 'returns deploy keys with projects a user can read' do
+    expected_result = {
+      id: deploy_key.id,
+      user_id: deploy_key.user_id,
+      title: deploy_key.title,
+      fingerprint: deploy_key.fingerprint,
+      can_push: deploy_key.can_push,
+      created_at: deploy_key.created_at,
+      updated_at: deploy_key.updated_at,
+      projects: [
+        { id: project.id, name: project.name, full_path: project.full_path, full_name: project.full_name }
+      ]
+    }
+
+    expect(entity.as_json).to eq(expected_result)
+  end
+end
-- 
GitLab


From ee592f0df36aba995d04a969697fb7a064b6ef3c Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 11:37:23 +0200
Subject: [PATCH 003/363] Improve specs related to CI/CD job environment

---
 spec/models/ci/build_spec.rb | 20 ++++++++++++--------
 1 file changed, 12 insertions(+), 8 deletions(-)

diff --git a/spec/models/ci/build_spec.rb b/spec/models/ci/build_spec.rb
index 8dbcf50ee0c5c..ae390afbd80e9 100644
--- a/spec/models/ci/build_spec.rb
+++ b/spec/models/ci/build_spec.rb
@@ -902,22 +902,26 @@ def create_mr(build, pipeline, factory: :merge_request, created_at: Time.now)
   end
 
   describe '#persisted_environment' do
-    before do
-      @environment = create(:environment, project: project, name: "foo-#{project.default_branch}")
+    let!(:environment) do
+      create(:environment, project: project, name: "foo-#{project.default_branch}")
     end
 
     subject { build.persisted_environment }
 
-    context 'referenced literally' do
-      let(:build) { create(:ci_build, pipeline: pipeline, environment: "foo-#{project.default_branch}") }
+    context 'when referenced literally' do
+      let(:build) do
+        create(:ci_build, pipeline: pipeline, environment: "foo-#{project.default_branch}")
+      end
 
-      it { is_expected.to eq(@environment) }
+      it { is_expected.to eq(environment) }
     end
 
-    context 'referenced with a variable' do
-      let(:build) { create(:ci_build, pipeline: pipeline, environment: "foo-$CI_COMMIT_REF_NAME") }
+    context 'when  referenced with a variable' do
+      let(:build) do
+        create(:ci_build, pipeline: pipeline, environment: "foo-$CI_COMMIT_REF_NAME")
+      end
 
-      it { is_expected.to eq(@environment) }
+      it { is_expected.to eq(environment) }
     end
   end
 
-- 
GitLab


From 5525db8b33bf1f76c4a8023ea4fb571d73e41b0f Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 11:52:37 +0200
Subject: [PATCH 004/363] Check branch access when user triggers manual action

---
 app/models/ci/build.rb       | 10 ++++++
 spec/models/ci/build_spec.rb | 65 +++++++++++++++++++++++++++++-------
 2 files changed, 63 insertions(+), 12 deletions(-)

diff --git a/app/models/ci/build.rb b/app/models/ci/build.rb
index 8431c5f228cb1..159b3b2e10153 100644
--- a/app/models/ci/build.rb
+++ b/app/models/ci/build.rb
@@ -115,7 +115,17 @@ def has_commands?
       commands.present?
     end
 
+    def can_play?(current_user)
+      ::Gitlab::UserAccess
+        .new(current_user, project: project)
+        .can_push_to_branch?(ref)
+    end
+
     def play(current_user)
+      unless can_play?(current_user)
+        raise Gitlab::Access::AccessDeniedError
+      end
+
       # Try to queue a current build
       if self.enqueue
         self.update(user: current_user)
diff --git a/spec/models/ci/build_spec.rb b/spec/models/ci/build_spec.rb
index ae390afbd80e9..8124d263fd417 100644
--- a/spec/models/ci/build_spec.rb
+++ b/spec/models/ci/build_spec.rb
@@ -925,6 +925,33 @@ def create_mr(build, pipeline, factory: :merge_request, created_at: Time.now)
     end
   end
 
+  describe '#can_play?' do
+    before do
+      project.add_developer(user)
+    end
+
+    let(:build) do
+      create(:ci_build, ref: 'some-ref', pipeline: pipeline)
+    end
+
+    context 'when branch build is running for is protected' do
+      before do
+        create(:protected_branch, :no_one_can_push,
+               name: 'some-ref', project: project)
+      end
+
+      it 'indicates that user can not trigger an action' do
+        expect(build.can_play?(user)).to be_falsey
+      end
+    end
+
+    context 'when branch build is running for is not protected' do
+      it 'indicates that user can trigger an action' do
+        expect(build.can_play?(user)).to be_truthy
+      end
+    end
+  end
+
   describe '#play' do
     let(:build) { create(:ci_build, :manual, pipeline: pipeline) }
 
@@ -932,25 +959,39 @@ def create_mr(build, pipeline, factory: :merge_request, created_at: Time.now)
       project.add_developer(user)
     end
 
-    context 'when build is manual' do
-      it 'enqueues a build' do
-        new_build = build.play(user)
+    context 'when user does not have ability to trigger action' do
+      before do
+        create(:protected_branch, :no_one_can_push,
+               name: build.ref, project: project)
+      end
 
-        expect(new_build).to be_pending
-        expect(new_build).to eq(build)
+      it 'raises an error' do
+        expect { build.play(user) }
+          .to raise_error Gitlab::Access::AccessDeniedError
       end
     end
 
-    context 'when build is passed' do
-      before do
-        build.update(status: 'success')
+    context 'when user has ability to trigger manual action' do
+      context 'when build is manual' do
+        it 'enqueues a build' do
+          new_build = build.play(user)
+
+          expect(new_build).to be_pending
+          expect(new_build).to eq(build)
+        end
       end
 
-      it 'creates a new build' do
-        new_build = build.play(user)
+      context 'when build is not manual' do
+        before do
+          build.update(status: 'success')
+        end
+
+        it 'creates a new build' do
+          new_build = build.play(user)
 
-        expect(new_build).to be_pending
-        expect(new_build).not_to eq(build)
+          expect(new_build).to be_pending
+          expect(new_build).not_to eq(build)
+        end
       end
     end
   end
-- 
GitLab


From 3dbef510c187606ef12eb9f2aaf51f30ddc3e30d Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 12:58:13 +0200
Subject: [PATCH 005/363] Expose manual action abilities from a build entity

---
 app/serializers/build_entity.rb       | 10 +++++++---
 spec/serializers/build_entity_spec.rb | 28 +++++++++++++++++++++++++--
 2 files changed, 33 insertions(+), 5 deletions(-)

diff --git a/app/serializers/build_entity.rb b/app/serializers/build_entity.rb
index b804d6d0e8a7b..401a277dadc53 100644
--- a/app/serializers/build_entity.rb
+++ b/app/serializers/build_entity.rb
@@ -12,7 +12,7 @@ class BuildEntity < Grape::Entity
     path_to(:retry_namespace_project_build, build)
   end
 
-  expose :play_path, if: ->(build, _) { build.playable? } do |build|
+  expose :play_path, if: proc { playable? } do |build|
     path_to(:play_namespace_project_build, build)
   end
 
@@ -25,11 +25,15 @@ class BuildEntity < Grape::Entity
 
   alias_method :build, :object
 
-  def path_to(route, build)
-    send("#{route}_path", build.project.namespace, build.project, build)
+  def playable?
+    build.playable? && build.can_play?(request.user)
   end
 
   def detailed_status
     build.detailed_status(request.user)
   end
+
+  def path_to(route, build)
+    send("#{route}_path", build.project.namespace, build.project, build)
+  end
 end
diff --git a/spec/serializers/build_entity_spec.rb b/spec/serializers/build_entity_spec.rb
index f76a5cf72d1a9..897a28b7305a8 100644
--- a/spec/serializers/build_entity_spec.rb
+++ b/spec/serializers/build_entity_spec.rb
@@ -41,13 +41,37 @@
     it 'does not contain path to play action' do
       expect(subject).not_to include(:play_path)
     end
+
+    it 'is not a playable job' do
+      expect(subject[:playable]).to be false
+    end
   end
 
   context 'when build is a manual action' do
     let(:build) { create(:ci_build, :manual) }
 
-    it 'contains path to play action' do
-      expect(subject).to include(:play_path)
+    context 'when user is allowed to trigger action' do
+      before do
+        build.project.add_master(user)
+      end
+
+      it 'contains path to play action' do
+        expect(subject).to include(:play_path)
+      end
+
+      it 'is a playable action' do
+        expect(subject[:playable]).to be true
+      end
+    end
+
+    context 'when user is not allowed to trigger action' do
+      it 'does not contain path to play action' do
+        expect(subject).not_to include(:play_path)
+      end
+
+      it 'is not a playable action' do
+        expect(subject[:playable]).to be false
+      end
     end
   end
 end
-- 
GitLab


From 5b6202cce1cf7ad5da1fcbfc4e089194311f205b Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 13:08:45 +0200
Subject: [PATCH 006/363] Do not show play action if user not allowed to run it

---
 lib/gitlab/ci/status/build/play.rb           |  2 +-
 spec/lib/gitlab/ci/status/build/play_spec.rb | 12 ++++++++++--
 2 files changed, 11 insertions(+), 3 deletions(-)

diff --git a/lib/gitlab/ci/status/build/play.rb b/lib/gitlab/ci/status/build/play.rb
index 3495b8d0448b9..29d0558a265c5 100644
--- a/lib/gitlab/ci/status/build/play.rb
+++ b/lib/gitlab/ci/status/build/play.rb
@@ -10,7 +10,7 @@ def label
           end
 
           def has_action?
-            can?(user, :update_build, subject)
+            can?(user, :update_build, subject) && subject.can_play?(user)
           end
 
           def action_icon
diff --git a/spec/lib/gitlab/ci/status/build/play_spec.rb b/spec/lib/gitlab/ci/status/build/play_spec.rb
index 6c97a4fe5cad4..48aeb89e11f44 100644
--- a/spec/lib/gitlab/ci/status/build/play_spec.rb
+++ b/spec/lib/gitlab/ci/status/build/play_spec.rb
@@ -17,9 +17,17 @@
 
     describe '#has_action?' do
       context 'when user is allowed to update build' do
-        before { build.project.team << [user, :developer] }
+        context 'when user can push to branch' do
+          before { build.project.add_master(user) }
 
-        it { is_expected.to have_action }
+          it { is_expected.to have_action }
+        end
+
+        context 'when user can not push to the branch' do
+          before { build.project.add_developer(user) }
+
+          it { is_expected.not_to have_action }
+        end
       end
 
       context 'when user is not allowed to update build' do
-- 
GitLab


From f647ad8f6fd076c6adcc3876b5e4a521a49c0ca8 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 13:40:04 +0200
Subject: [PATCH 007/363] Fix chatops specs with incorrectly defined project

---
 spec/lib/gitlab/chat_commands/command_spec.rb | 14 ++++++++++----
 spec/lib/gitlab/chat_commands/deploy_spec.rb  | 16 ++++++++++------
 2 files changed, 20 insertions(+), 10 deletions(-)

diff --git a/spec/lib/gitlab/chat_commands/command_spec.rb b/spec/lib/gitlab/chat_commands/command_spec.rb
index b6e924d67bebc..eb4f06b371c43 100644
--- a/spec/lib/gitlab/chat_commands/command_spec.rb
+++ b/spec/lib/gitlab/chat_commands/command_spec.rb
@@ -40,11 +40,15 @@
 
     context 'when trying to do deployment' do
       let(:params) { { text: 'deploy staging to production' } }
-      let!(:build) { create(:ci_build, project: project) }
+      let!(:build) { create(:ci_build, pipeline: pipeline) }
+      let!(:pipeline) { create(:ci_pipeline, project: project) }
       let!(:staging) { create(:environment, name: 'staging', project: project) }
       let!(:deployment) { create(:deployment, environment: staging, deployable: build) }
+
       let!(:manual) do
-        create(:ci_build, :manual, project: project, pipeline: build.pipeline, name: 'first', environment: 'production')
+        create(:ci_build, :manual, pipeline: pipeline,
+                                   name: 'first',
+                                   environment: 'production')
       end
 
       context 'and user can not create deployment' do
@@ -56,7 +60,7 @@
 
       context 'and user does have deployment permission' do
         before do
-          project.team << [user, :developer]
+          build.project.add_master(user)
         end
 
         it 'returns action' do
@@ -66,7 +70,9 @@
 
         context 'when duplicate action exists' do
           let!(:manual2) do
-            create(:ci_build, :manual, project: project, pipeline: build.pipeline, name: 'second', environment: 'production')
+            create(:ci_build, :manual, pipeline: pipeline,
+                                       name: 'second',
+                                       environment: 'production')
           end
 
           it 'returns error' do
diff --git a/spec/lib/gitlab/chat_commands/deploy_spec.rb b/spec/lib/gitlab/chat_commands/deploy_spec.rb
index b3358a321618d..b33389d959e30 100644
--- a/spec/lib/gitlab/chat_commands/deploy_spec.rb
+++ b/spec/lib/gitlab/chat_commands/deploy_spec.rb
@@ -7,7 +7,7 @@
     let(:regex_match) { described_class.match('deploy staging to production') }
 
     before do
-      project.team << [user, :master]
+      project.add_master(user)
     end
 
     subject do
@@ -23,7 +23,8 @@
 
     context 'with environment' do
       let!(:staging) { create(:environment, name: 'staging', project: project) }
-      let!(:build) { create(:ci_build, project: project) }
+      let!(:pipeline) { create(:ci_pipeline, project: project) }
+      let!(:build) { create(:ci_build, pipeline: pipeline) }
       let!(:deployment) { create(:deployment, environment: staging, deployable: build) }
 
       context 'without actions' do
@@ -35,7 +36,9 @@
 
       context 'with action' do
         let!(:manual1) do
-          create(:ci_build, :manual, project: project, pipeline: build.pipeline, name: 'first', environment: 'production')
+          create(:ci_build, :manual, pipeline: pipeline,
+                                     name: 'first',
+                                     environment: 'production')
         end
 
         it 'returns success result' do
@@ -45,7 +48,9 @@
 
         context 'when duplicate action exists' do
           let!(:manual2) do
-            create(:ci_build, :manual, project: project, pipeline: build.pipeline, name: 'second', environment: 'production')
+            create(:ci_build, :manual, pipeline: pipeline,
+                                       name: 'second',
+                                       environment: 'production')
           end
 
           it 'returns error' do
@@ -57,8 +62,7 @@
         context 'when teardown action exists' do
           let!(:teardown) do
             create(:ci_build, :manual, :teardown_environment,
-                   project: project, pipeline: build.pipeline,
-                   name: 'teardown', environment: 'production')
+                   pipeline: pipeline, name: 'teardown', environment: 'production')
           end
 
           it 'returns the success message' do
-- 
GitLab


From e533d43a8c03a9b47a7016f3fea01a00ca797778 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 14:28:40 +0200
Subject: [PATCH 008/363] Fix specs related to new manual actions permissions

---
 spec/factories/environments.rb                |  4 +-
 .../projects/environments/environment_spec.rb |  4 ++
 spec/models/environment_spec.rb               | 53 ++++++++++++++-----
 .../ci/process_pipeline_service_spec.rb       |  7 +++
 4 files changed, 54 insertions(+), 14 deletions(-)

diff --git a/spec/factories/environments.rb b/spec/factories/environments.rb
index 3fbf24b5c7d29..f6595751d1e27 100644
--- a/spec/factories/environments.rb
+++ b/spec/factories/environments.rb
@@ -18,6 +18,8 @@
       # interconnected objects to simulate a review app.
       #
       after(:create) do |environment, evaluator|
+        pipeline = create(:ci_pipeline, project: environment.project)
+
         deployment = create(:deployment,
                             environment: environment,
                             project: environment.project,
@@ -26,7 +28,7 @@
 
         teardown_build = create(:ci_build, :manual,
                                 name: "#{deployment.environment.name}:teardown",
-                                pipeline: deployment.deployable.pipeline)
+                                pipeline: pipeline)
 
         deployment.update_column(:on_stop, teardown_build.name)
         environment.update_attribute(:deployments, [deployment])
diff --git a/spec/features/projects/environments/environment_spec.rb b/spec/features/projects/environments/environment_spec.rb
index acc3efe04e672..2b7f67eee3265 100644
--- a/spec/features/projects/environments/environment_spec.rb
+++ b/spec/features/projects/environments/environment_spec.rb
@@ -62,6 +62,8 @@
                                        name: 'deploy to production')
           end
 
+          given(:role) { :master }
+
           scenario 'does show a play button' do
             expect(page).to have_link(action.name.humanize)
           end
@@ -132,6 +134,8 @@
                                     on_stop: 'close_app')
               end
 
+              given(:role) { :master }
+
               scenario 'does allow to stop environment' do
                 click_link('Stop')
 
diff --git a/spec/models/environment_spec.rb b/spec/models/environment_spec.rb
index 9f0e7fbbe268c..9e00f2247e8b1 100644
--- a/spec/models/environment_spec.rb
+++ b/spec/models/environment_spec.rb
@@ -191,25 +191,52 @@
     end
 
     context 'when matching action is defined' do
-      let(:build) { create(:ci_build) }
-      let!(:deployment) { create(:deployment, environment: environment, deployable: build, on_stop: 'close_app') }
+      let(:pipeline) { create(:ci_pipeline, project: project) }
+      let(:build) { create(:ci_build, pipeline: pipeline) }
+
+      let!(:deployment) do
+        create(:deployment, environment: environment,
+                            deployable: build,
+                            on_stop: 'close_app')
+      end
 
-      context 'when action did not yet finish' do
-        let!(:close_action) { create(:ci_build, :manual, pipeline: build.pipeline, name: 'close_app') }
+      context 'when user is not allowed to stop environment' do
+        let!(:close_action) do
+          create(:ci_build, :manual, pipeline: pipeline, name: 'close_app')
+        end
 
-        it 'returns the same action' do
-          expect(subject).to eq(close_action)
-          expect(subject.user).to eq(user)
+        it 'raises an exception' do
+          expect { subject }.to raise_error(Gitlab::Access::AccessDeniedError)
         end
       end
 
-      context 'if action did finish' do
-        let!(:close_action) { create(:ci_build, :manual, :success, pipeline: build.pipeline, name: 'close_app') }
+      context 'when user is allowed to stop environment' do
+        before do
+          project.add_master(user)
+        end
+
+        context 'when action did not yet finish' do
+          let!(:close_action) do
+            create(:ci_build, :manual, pipeline: pipeline, name: 'close_app')
+          end
+
+          it 'returns the same action' do
+            expect(subject).to eq(close_action)
+            expect(subject.user).to eq(user)
+          end
+        end
 
-        it 'returns a new action of the same type' do
-          is_expected.to be_persisted
-          expect(subject.name).to eq(close_action.name)
-          expect(subject.user).to eq(user)
+        context 'if action did finish' do
+          let!(:close_action) do
+            create(:ci_build, :manual, :success,
+                   pipeline: pipeline, name: 'close_app')
+          end
+
+          it 'returns a new action of the same type' do
+            expect(subject).to be_persisted
+            expect(subject.name).to eq(close_action.name)
+            expect(subject.user).to eq(user)
+          end
         end
       end
     end
diff --git a/spec/services/ci/process_pipeline_service_spec.rb b/spec/services/ci/process_pipeline_service_spec.rb
index bb98fb37a9018..7df6a81b0aba0 100644
--- a/spec/services/ci/process_pipeline_service_spec.rb
+++ b/spec/services/ci/process_pipeline_service_spec.rb
@@ -314,6 +314,13 @@
     end
 
     context 'when pipeline is promoted sequentially up to the end' do
+      before do
+        # We are using create(:empty_project), and users has to be master in
+        # order to execute manual action when repository does not exist.
+        #
+        project.add_master(user)
+      end
+
       it 'properly processes entire pipeline' do
         process_pipeline
 
-- 
GitLab


From 6c0fc62ef5c4fa4535174a9f187b9853f0fb90ac Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 15:19:52 +0200
Subject: [PATCH 009/363] Take branch access into account when stopping
 environment

---
 app/models/deployment.rb                      |  4 +--
 app/models/environment.rb                     |  6 +++++
 app/services/ci/stop_environments_service.rb  |  8 +++---
 spec/factories/environments.rb                |  6 ++++-
 spec/models/environment_spec.rb               | 25 +++++++++++++++++++
 .../ci/stop_environments_service_spec.rb      | 16 +++++++++++-
 6 files changed, 58 insertions(+), 7 deletions(-)

diff --git a/app/models/deployment.rb b/app/models/deployment.rb
index afad001d50f91..37adfb4de7342 100644
--- a/app/models/deployment.rb
+++ b/app/models/deployment.rb
@@ -85,8 +85,8 @@ def previous_deployment
   end
 
   def stop_action
-    return nil unless on_stop.present?
-    return nil unless manual_actions
+    return unless on_stop.present?
+    return unless manual_actions
 
     @stop_action ||= manual_actions.find_by(name: on_stop)
   end
diff --git a/app/models/environment.rb b/app/models/environment.rb
index bf33010fd21f2..f8b9a21c08ea3 100644
--- a/app/models/environment.rb
+++ b/app/models/environment.rb
@@ -122,6 +122,12 @@ def stop_action?
     available? && stop_action.present?
   end
 
+  def can_trigger_stop_action?(current_user)
+    return false unless stop_action?
+
+    stop_action.can_play?(current_user)
+  end
+
   def stop_with_action!(current_user)
     return unless available?
 
diff --git a/app/services/ci/stop_environments_service.rb b/app/services/ci/stop_environments_service.rb
index 42c72aba7ddfc..d1e341bc9b509 100644
--- a/app/services/ci/stop_environments_service.rb
+++ b/app/services/ci/stop_environments_service.rb
@@ -6,9 +6,10 @@ def execute(branch_name)
       @ref = branch_name
 
       return unless has_ref?
+      return unless can?(current_user, :create_deployment, project)
 
       environments.each do |environment|
-        next unless can?(current_user, :create_deployment, project)
+        next unless environment.can_trigger_stop_action?(current_user)
 
         environment.stop_with_action!(current_user)
       end
@@ -21,8 +22,9 @@ def has_ref?
     end
 
     def environments
-      @environments ||=
-        EnvironmentsFinder.new(project, current_user, ref: @ref, recently_updated: true).execute
+      @environments ||= EnvironmentsFinder
+        .new(project, current_user, ref: @ref, recently_updated: true)
+        .execute
     end
   end
 end
diff --git a/spec/factories/environments.rb b/spec/factories/environments.rb
index f6595751d1e27..d8d699fb3aace 100644
--- a/spec/factories/environments.rb
+++ b/spec/factories/environments.rb
@@ -20,14 +20,18 @@
       after(:create) do |environment, evaluator|
         pipeline = create(:ci_pipeline, project: environment.project)
 
+        deployable = create(:ci_build, name: "#{environment.name}:deploy",
+                                       pipeline: pipeline)
+
         deployment = create(:deployment,
                             environment: environment,
                             project: environment.project,
+                            deployable: deployable,
                             ref: evaluator.ref,
                             sha: environment.project.commit(evaluator.ref).id)
 
         teardown_build = create(:ci_build, :manual,
-                                name: "#{deployment.environment.name}:teardown",
+                                name: "#{environment.name}:teardown",
                                 pipeline: pipeline)
 
         deployment.update_column(:on_stop, teardown_build.name)
diff --git a/spec/models/environment_spec.rb b/spec/models/environment_spec.rb
index 9e00f2247e8b1..fb7cee47ae8e2 100644
--- a/spec/models/environment_spec.rb
+++ b/spec/models/environment_spec.rb
@@ -155,6 +155,31 @@
     end
   end
 
+  describe '#can_trigger_stop_action?' do
+    let(:user) { create(:user) }
+    let(:project) { create(:project) }
+
+    let(:environment) do
+      create(:environment, :with_review_app, project: project)
+    end
+
+    context 'when user can trigger stop action' do
+      before do
+        project.add_developer(user)
+      end
+
+      it 'returns value that evaluates to true' do
+        expect(environment.can_trigger_stop_action?(user)).to be_truthy
+      end
+    end
+
+    context 'when user is not allowed to trigger stop action' do
+      it 'returns value that evaluates to false' do
+        expect(environment.can_trigger_stop_action?(user)).to be_falsey
+      end
+    end
+  end
+
   describe '#stop_with_action!' do
     let(:user) { create(:admin) }
 
diff --git a/spec/services/ci/stop_environments_service_spec.rb b/spec/services/ci/stop_environments_service_spec.rb
index 32c72a9cf5ef0..98044ad232ed9 100644
--- a/spec/services/ci/stop_environments_service_spec.rb
+++ b/spec/services/ci/stop_environments_service_spec.rb
@@ -55,8 +55,22 @@
       end
 
       context 'when user does not have permission to stop environment' do
+        context 'when user has no access to manage deployments' do
+          before do
+            project.team << [user, :guest]
+          end
+
+          it 'does not stop environment' do
+            expect_environment_not_stopped_on('master')
+          end
+        end
+      end
+
+      context 'when branch for stop action is protected' do
         before do
-          project.team << [user, :guest]
+          project.add_developer(user)
+          create(:protected_branch, :no_one_can_push,
+                 name: 'master', project: project)
         end
 
         it 'does not stop environment' do
-- 
GitLab


From d96e90734783a83ca4d87b961699072712077d9f Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 15:24:48 +0200
Subject: [PATCH 010/363] Fix specs for build status factory and manual status

---
 spec/lib/gitlab/ci/status/build/factory_spec.rb | 17 ++++++++++++++++-
 1 file changed, 16 insertions(+), 1 deletion(-)

diff --git a/spec/lib/gitlab/ci/status/build/factory_spec.rb b/spec/lib/gitlab/ci/status/build/factory_spec.rb
index e648a3ac3a27c..2ab67127b1edd 100644
--- a/spec/lib/gitlab/ci/status/build/factory_spec.rb
+++ b/spec/lib/gitlab/ci/status/build/factory_spec.rb
@@ -218,9 +218,24 @@
         expect(status.favicon).to eq 'favicon_status_manual'
         expect(status.label).to eq 'manual play action'
         expect(status).to have_details
-        expect(status).to have_action
         expect(status.action_path).to include 'play'
       end
+
+      context 'when user has ability to play action' do
+        before do
+          build.project.add_master(user)
+        end
+
+        it 'fabricates status that has action' do
+          expect(status).to have_action
+        end
+      end
+
+      context 'when user does not have ability to play action' do
+        it 'fabricates status that has no action' do
+          expect(status).not_to have_action
+        end
+      end
     end
 
     context 'when build is an environment stop action' do
-- 
GitLab


From 3e55e0742218fe20b269dfbc4f44d3798e4b0eb6 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 15:32:23 +0200
Subject: [PATCH 011/363] Check if user can trigger manual action in the UI

---
 app/views/projects/ci/builds/_build.html.haml | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/views/projects/ci/builds/_build.html.haml b/app/views/projects/ci/builds/_build.html.haml
index aeed293a72427..ceec36e24407c 100644
--- a/app/views/projects/ci/builds/_build.html.haml
+++ b/app/views/projects/ci/builds/_build.html.haml
@@ -101,7 +101,7 @@
           = link_to cancel_namespace_project_build_path(build.project.namespace, build.project, build, return_to: request.original_url), method: :post, title: 'Cancel', class: 'btn btn-build' do
             = icon('remove', class: 'cred')
         - elsif allow_retry
-          - if build.playable? && !admin
+          - if build.playable? && !admin && build.can_play?(current_user)
             = link_to play_namespace_project_build_path(build.project.namespace, build.project, build, return_to: request.original_url), method: :post, title: 'Play', class: 'btn btn-build' do
               = custom_icon('icon_play')
           - elsif build.retryable?
-- 
GitLab


From 7bcca2284b09e18438e6163c6ead72e10fdd2f57 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 6 Apr 2017 17:16:19 +0200
Subject: [PATCH 012/363] Check branch permission in manual action entity

---
 app/serializers/build_action_entity.rb | 8 ++++++++
 1 file changed, 8 insertions(+)

diff --git a/app/serializers/build_action_entity.rb b/app/serializers/build_action_entity.rb
index 184b4b7a6813e..3d12c64b88aa4 100644
--- a/app/serializers/build_action_entity.rb
+++ b/app/serializers/build_action_entity.rb
@@ -13,4 +13,12 @@ class BuildActionEntity < Grape::Entity
   end
 
   expose :playable?, as: :playable
+
+  private
+
+  alias_method :build, :object
+
+  def playable?
+    build.playable? && build.can_play?(request.user)
+  end
 end
-- 
GitLab


From 703df2881bb137a79284baafe2cc12ff32ab9ff5 Mon Sep 17 00:00:00 2001
From: Jarka Kadlecova <jarka@gitlab.com>
Date: Fri, 7 Apr 2017 13:34:39 +0200
Subject: [PATCH 013/363] expose additional values in deploy key entity

---
 app/serializers/deploy_key_entity.rb       | 2 ++
 spec/serializers/deploy_key_entity_spec.rb | 2 ++
 2 files changed, 4 insertions(+)

diff --git a/app/serializers/deploy_key_entity.rb b/app/serializers/deploy_key_entity.rb
index cdedc2c7bd018..d75a83d0fa529 100644
--- a/app/serializers/deploy_key_entity.rb
+++ b/app/serializers/deploy_key_entity.rb
@@ -4,6 +4,8 @@ class DeployKeyEntity < Grape::Entity
   expose :title
   expose :fingerprint
   expose :can_push
+  expose :destroyed_when_orphaned?, as: :destroyed_when_orphaned
+  expose :almost_orphaned?, as: :almost_orphaned
   expose :created_at
   expose :updated_at
   expose :projects, using: ProjectEntity do |deploy_key|
diff --git a/spec/serializers/deploy_key_entity_spec.rb b/spec/serializers/deploy_key_entity_spec.rb
index da1d33b42ac80..cc3fb193f1be2 100644
--- a/spec/serializers/deploy_key_entity_spec.rb
+++ b/spec/serializers/deploy_key_entity_spec.rb
@@ -17,6 +17,8 @@
       title: deploy_key.title,
       fingerprint: deploy_key.fingerprint,
       can_push: deploy_key.can_push,
+      destroyed_when_orphaned: true,
+      almost_orphaned: false,
       created_at: deploy_key.created_at,
       updated_at: deploy_key.updated_at,
       projects: [
-- 
GitLab


From acea881bb012cce0b59f3d5874a630b16d0caaef Mon Sep 17 00:00:00 2001
From: Jacob Schatz <jschatz1@gmail.com>
Date: Fri, 7 Apr 2017 23:38:43 -0400
Subject: [PATCH 014/363] Initial balsamiq support

---
 app/models/blob.rb                         | 6 ++++++
 app/views/projects/blob/_bmpr.html.haml    | 5 +++++
 app/views/projects/blob/balsamiq_viewer.js | 3 +++
 config/webpack.config.js                   | 1 +
 db/schema.rb                               | 2 +-
 5 files changed, 16 insertions(+), 1 deletion(-)
 create mode 100644 app/views/projects/blob/_bmpr.html.haml
 create mode 100644 app/views/projects/blob/balsamiq_viewer.js

diff --git a/app/models/blob.rb b/app/models/blob.rb
index 801d344280367..0c1730f6801bc 100644
--- a/app/models/blob.rb
+++ b/app/models/blob.rb
@@ -58,6 +58,10 @@ def sketch?
     binary? && extname.downcase.delete('.') == 'sketch'
   end
 
+  def balsamiq?
+    binary? && extname.downcase.delete('.') == 'bmpr'
+  end
+
   def stl?
     extname.downcase.delete('.') == 'stl'
   end
@@ -87,6 +91,8 @@ def to_partial_path(project)
       'sketch'
     elsif stl?
       'stl'
+    elsif balsamiq?
+      'bmpr'
     elsif text?
       'text'
     else
diff --git a/app/views/projects/blob/_bmpr.html.haml b/app/views/projects/blob/_bmpr.html.haml
new file mode 100644
index 0000000000000..048bdf9dcb5a6
--- /dev/null
+++ b/app/views/projects/blob/_bmpr.html.haml
@@ -0,0 +1,5 @@
+- content_for :page_specific_javascripts do
+  = page_specific_javascript_bundle_tag('common_vue')
+  = page_specific_javascript_bundle_tag('balsamiq_viewer')
+
+.file-content#js-balsamiq-viewer{ data: { endpoint: namespace_project_raw_path(@project.namespace, @project, @id) } }
\ No newline at end of file
diff --git a/app/views/projects/blob/balsamiq_viewer.js b/app/views/projects/blob/balsamiq_viewer.js
new file mode 100644
index 0000000000000..b60cfe165a4e9
--- /dev/null
+++ b/app/views/projects/blob/balsamiq_viewer.js
@@ -0,0 +1,3 @@
+import renderBalsamiq from './balsamiq';
+
+document.addEventListener('DOMContentLoaded', renderBalsamiq);
\ No newline at end of file
diff --git a/config/webpack.config.js b/config/webpack.config.js
index e3bc939d57847..42638d47cfadd 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -39,6 +39,7 @@ var config = {
     notebook_viewer:      './blob/notebook_viewer.js',
     sketch_viewer:        './blob/sketch_viewer.js',
     pdf_viewer:           './blob/pdf_viewer.js',
+    balsamiq_viewer:      './blob/balsamiq_viewer.js',
     profile:              './profile/profile_bundle.js',
     protected_branches:   './protected_branches/protected_branches_bundle.js',
     protected_tags:       './protected_tags',
diff --git a/db/schema.rb b/db/schema.rb
index 3422847d72985..ceae051db0c60 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -950,9 +950,9 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
   end
 
   add_index "projects", ["ci_id"], name: "index_projects_on_ci_id", using: :btree
-- 
GitLab


From ef07200cd0f059a2e0493779263aa526a2ade2e3 Mon Sep 17 00:00:00 2001
From: Jacob Schatz <jschatz1@gmail.com>
Date: Sat, 8 Apr 2017 00:18:37 -0400
Subject: [PATCH 015/363] Get initial sql values back from file which is
 database

---
 app/assets/javascripts/blob/balsamiq/index.js | 27 +++++++++++++++++++
 .../javascripts/blob/balsamiq_viewer.js       |  5 ++++
 app/views/projects/blob/balsamiq_viewer.js    |  3 ---
 config/webpack.config.js                      |  5 ++++
 package.json                                  |  1 +
 yarn.lock                                     |  4 +++
 6 files changed, 42 insertions(+), 3 deletions(-)
 create mode 100644 app/assets/javascripts/blob/balsamiq/index.js
 create mode 100644 app/assets/javascripts/blob/balsamiq_viewer.js
 delete mode 100644 app/views/projects/blob/balsamiq_viewer.js

diff --git a/app/assets/javascripts/blob/balsamiq/index.js b/app/assets/javascripts/blob/balsamiq/index.js
new file mode 100644
index 0000000000000..042013668974b
--- /dev/null
+++ b/app/assets/javascripts/blob/balsamiq/index.js
@@ -0,0 +1,27 @@
+import Vue from 'vue';
+import sqljs from 'sql.js';
+
+export default class BalsamiqViewer {
+  constructor(el) {
+    this.el = el;
+    this.loadSqlFile();
+  }
+  
+  
+
+  loadSqlFile() {
+    var xhr = new XMLHttpRequest();
+    console.log(this.el)
+    xhr.open('GET', this.el.dataset.endpoint, true);
+    xhr.responseType = 'arraybuffer';
+
+    xhr.onload = function(e) {
+      var uInt8Array = new Uint8Array(this.response);
+      var db = new SQL.Database(uInt8Array);
+      var contents = db.exec("SELECT * FROM thumbnails");
+      console.log(contents)
+      // contents is now [{columns:['col1','col2',...], values:[[first row], [second row], ...]}]
+    };
+    xhr.send();
+  }
+}
\ No newline at end of file
diff --git a/app/assets/javascripts/blob/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq_viewer.js
new file mode 100644
index 0000000000000..b14933980998d
--- /dev/null
+++ b/app/assets/javascripts/blob/balsamiq_viewer.js
@@ -0,0 +1,5 @@
+import BalsamiqViewer from './balsamiq';
+
+document.addEventListener('DOMContentLoaded', () => {
+  new BalsamiqViewer(document.getElementById('js-balsamiq-viewer'));
+});
\ No newline at end of file
diff --git a/app/views/projects/blob/balsamiq_viewer.js b/app/views/projects/blob/balsamiq_viewer.js
deleted file mode 100644
index b60cfe165a4e9..0000000000000
--- a/app/views/projects/blob/balsamiq_viewer.js
+++ /dev/null
@@ -1,3 +0,0 @@
-import renderBalsamiq from './balsamiq';
-
-document.addEventListener('DOMContentLoaded', renderBalsamiq);
\ No newline at end of file
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 42638d47cfadd..bc4126bce02d4 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -15,6 +15,10 @@ var DEV_SERVER_LIVERELOAD = process.env.DEV_SERVER_LIVERELOAD !== 'false';
 var WEBPACK_REPORT = process.env.WEBPACK_REPORT;
 
 var config = {
+  // because sqljs requires fs.
+  node: {
+    fs: "empty"
+  },
   context: path.join(ROOT_PATH, 'app/assets/javascripts'),
   entry: {
     common:               './commons/index.js',
@@ -118,6 +122,7 @@ var config = {
         'merge_conflicts',
         'notebook_viewer',
         'pdf_viewer',
+        'balsamiq_viewer',
         'vue_pipelines',
       ],
       minChunks: function(module, count) {
diff --git a/package.json b/package.json
index 6d4f99e33b30c..eb680b0d76c94 100644
--- a/package.json
+++ b/package.json
@@ -35,6 +35,7 @@
     "raphael": "^2.2.7",
     "raw-loader": "^0.5.1",
     "select2": "3.5.2-browserify",
+    "sql.js": "^0.4.0",
     "stats-webpack-plugin": "^0.4.3",
     "three": "^0.84.0",
     "three-orbit-controls": "^82.1.0",
diff --git a/yarn.lock b/yarn.lock
index 2434b3a8a48a6..d0175ca9f986e 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -4154,6 +4154,10 @@ sprintf-js@~1.0.2:
   version "1.0.3"
   resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c"
 
+sql.js@^0.4.0:
+  version "0.4.0"
+  resolved "https://registry.yarnpkg.com/sql.js/-/sql.js-0.4.0.tgz#23be9635520eb0ff43a741e7e830397266e88445"
+
 sshpk@^1.7.0:
   version "1.10.2"
   resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.10.2.tgz#d5a804ce22695515638e798dbe23273de070a5fa"
-- 
GitLab


From e1b0ed391fcbd1c622b6e3c866674b85ccd0edea Mon Sep 17 00:00:00 2001
From: Jacob Schatz <jschatz1@gmail.com>
Date: Sat, 8 Apr 2017 10:36:42 -0400
Subject: [PATCH 016/363] Show thumbnails and their titles.

---
 app/assets/javascripts/blob/balsamiq/index.js | 22 +++++++++++++++----
 1 file changed, 18 insertions(+), 4 deletions(-)

diff --git a/app/assets/javascripts/blob/balsamiq/index.js b/app/assets/javascripts/blob/balsamiq/index.js
index 042013668974b..61f4631b423fe 100644
--- a/app/assets/javascripts/blob/balsamiq/index.js
+++ b/app/assets/javascripts/blob/balsamiq/index.js
@@ -8,19 +8,33 @@ export default class BalsamiqViewer {
   }
   
   
-
   loadSqlFile() {
     var xhr = new XMLHttpRequest();
-    console.log(this.el)
+    var self = this;
     xhr.open('GET', this.el.dataset.endpoint, true);
     xhr.responseType = 'arraybuffer';
 
     xhr.onload = function(e) {
+      var list = document.createElement('ul');
       var uInt8Array = new Uint8Array(this.response);
       var db = new SQL.Database(uInt8Array);
       var contents = db.exec("SELECT * FROM thumbnails");
-      console.log(contents)
-      // contents is now [{columns:['col1','col2',...], values:[[first row], [second row], ...]}]
+      var previews = contents[0].values.map((i)=>{return JSON.parse(i[1])});
+      previews.forEach((prev) => {
+        var li = document.createElement('li');
+        var title = db.exec(`select * from resources where id = '${prev.resourceID}'`)
+        var template = 
+        `<div class="panel panel-default">
+            <div class="panel-heading">${JSON.parse(title[0].values[0][2]).name}</div>
+            <div class="panel-body">
+              <img class="img-thumbnail" src="data:image/png;base64,${prev.image}"/>
+            </div>
+          </div>`;
+        li.innerHTML = template;
+        list.appendChild(li);
+      });
+      list.classList += 'list-inline';
+      self.el.appendChild(list);
     };
     xhr.send();
   }
-- 
GitLab


From 440ff838f7646ac7ea7a660280b51740ebeed70f Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Mon, 10 Apr 2017 15:48:23 +0100
Subject: [PATCH 017/363] Tidy balsamiq viewer and remove unused Vue

---
 .../blob/balsamiq/balsamiq_viewer.js          | 89 +++++++++++++++++++
 app/assets/javascripts/blob/balsamiq/index.js | 41 ---------
 .../javascripts/blob/balsamiq_viewer.js       |  7 +-
 app/views/projects/blob/_bmpr.html.haml       |  3 +-
 4 files changed, 94 insertions(+), 46 deletions(-)
 create mode 100644 app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
 delete mode 100644 app/assets/javascripts/blob/balsamiq/index.js

diff --git a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
new file mode 100644
index 0000000000000..e1c837cb09efc
--- /dev/null
+++ b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
@@ -0,0 +1,89 @@
+/* global Flash */
+
+import sqljs from 'sql.js';
+
+class BalsamiqViewer {
+  constructor(viewer) {
+    this.viewer = viewer;
+    this.endpoint = this.viewer.dataset.endpoint;
+  }
+
+  loadFile() {
+    const xhr = new XMLHttpRequest();
+
+    xhr.open('GET', this.endpoint, true);
+    xhr.responseType = 'arraybuffer';
+
+    xhr.onload = this.renderFile.bind(this);
+    xhr.onerror = BalsamiqViewer.onError;
+
+    xhr.send();
+  }
+
+  renderFile(loadEvent) {
+    const container = document.createElement('ul');
+
+    this.initDatabase(loadEvent.target.response);
+
+    const previews = this.getPreviews();
+    const renderedPreviews = previews.map(preview => this.renderPreview(preview, container));
+
+    container.innerHTML = renderedPreviews.join('');
+    container.classList.add('list-inline');
+
+    this.viewer.appendChild(container);
+  }
+
+  initDatabase(data) {
+    const previewBinary = new Uint8Array(data);
+
+    this.database = new sqljs.Database(previewBinary);
+  }
+
+  getPreviews() {
+    const thumnails = this.database.exec('SELECT * FROM thumbnails');
+
+    return thumnails[0].values.map(BalsamiqViewer.parsePreview);
+  }
+
+  renderPreview(preview) {
+    const previewElement = document.createElement('li');
+
+    previewElement.innerHTML = this.renderTemplate(preview);
+
+    return previewElement.outerHTML;
+  }
+
+  renderTemplate(preview) {
+    let template = BalsamiqViewer.PREVIEW_TEMPLATE;
+
+    const title = this.database.exec(`SELECT * FROM resources WHERE id = '${preview.resourceID}'`);
+    const name = JSON.parse(title[0].values[0][2]).name;
+    const image = preview.image;
+
+    template = template.replace(/{{name}}/, name).replace(/{{image}}/, image);
+
+    return template;
+  }
+
+  static parsePreview(preview) {
+    return JSON.parse(preview[1]);
+  }
+
+  static onError() {
+    const flash = new Flash('Balsamiq file could not be loaded.');
+
+    return flash;
+  }
+}
+
+BalsamiqViewer.PREVIEW_TEMPLATE = `
+  <div class="panel panel-default">
+    <div class="panel-heading">{{name}}</div>
+    <div class="panel-body">
+      <img class="img-thumbnail" src="data:image/png;base64,{{image}}"/>
+    </div>
+  </div>
+`;
+
+export default BalsamiqViewer;
diff --git a/app/assets/javascripts/blob/balsamiq/index.js b/app/assets/javascripts/blob/balsamiq/index.js
deleted file mode 100644
index 61f4631b423fe..0000000000000
--- a/app/assets/javascripts/blob/balsamiq/index.js
+++ /dev/null
@@ -1,41 +0,0 @@
-import Vue from 'vue';
-import sqljs from 'sql.js';
-
-export default class BalsamiqViewer {
-  constructor(el) {
-    this.el = el;
-    this.loadSqlFile();
-  }
-  
-  
-  loadSqlFile() {
-    var xhr = new XMLHttpRequest();
-    var self = this;
-    xhr.open('GET', this.el.dataset.endpoint, true);
-    xhr.responseType = 'arraybuffer';
-
-    xhr.onload = function(e) {
-      var list = document.createElement('ul');
-      var uInt8Array = new Uint8Array(this.response);
-      var db = new SQL.Database(uInt8Array);
-      var contents = db.exec("SELECT * FROM thumbnails");
-      var previews = contents[0].values.map((i)=>{return JSON.parse(i[1])});
-      previews.forEach((prev) => {
-        var li = document.createElement('li');
-        var title = db.exec(`select * from resources where id = '${prev.resourceID}'`)
-        var template = 
-        `<div class="panel panel-default">
-            <div class="panel-heading">${JSON.parse(title[0].values[0][2]).name}</div>
-            <div class="panel-body">
-              <img class="img-thumbnail" src="data:image/png;base64,${prev.image}"/>
-            </div>
-          </div>`;
-        li.innerHTML = template;
-        list.appendChild(li);
-      });
-      list.classList += 'list-inline';
-      self.el.appendChild(list);
-    };
-    xhr.send();
-  }
-}
\ No newline at end of file
diff --git a/app/assets/javascripts/blob/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq_viewer.js
index b14933980998d..1dacf84470f3f 100644
--- a/app/assets/javascripts/blob/balsamiq_viewer.js
+++ b/app/assets/javascripts/blob/balsamiq_viewer.js
@@ -1,5 +1,6 @@
-import BalsamiqViewer from './balsamiq';
+import BalsamiqViewer from './balsamiq/balsamiq_viewer';
 
 document.addEventListener('DOMContentLoaded', () => {
-  new BalsamiqViewer(document.getElementById('js-balsamiq-viewer'));
-});
\ No newline at end of file
+  const balsamiqViewer = new BalsamiqViewer(document.getElementById('js-balsamiq-viewer'));
+  balsamiqViewer.loadFile();
+});
diff --git a/app/views/projects/blob/_bmpr.html.haml b/app/views/projects/blob/_bmpr.html.haml
index 048bdf9dcb5a6..36ec0cbcce8bb 100644
--- a/app/views/projects/blob/_bmpr.html.haml
+++ b/app/views/projects/blob/_bmpr.html.haml
@@ -1,5 +1,4 @@
 - content_for :page_specific_javascripts do
-  = page_specific_javascript_bundle_tag('common_vue')
   = page_specific_javascript_bundle_tag('balsamiq_viewer')
 
-.file-content#js-balsamiq-viewer{ data: { endpoint: namespace_project_raw_path(@project.namespace, @project, @id) } }
\ No newline at end of file
+.file-content#js-balsamiq-viewer{ data: { endpoint: namespace_project_raw_path(@project.namespace, @project, @id) } }
-- 
GitLab


From 80745bc81da089af088d99574088dc6bf8959e4d Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Mon, 10 Apr 2017 16:16:09 +0100
Subject: [PATCH 018/363] Revert schema.db

---
 db/schema.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/db/schema.rb b/db/schema.rb
index ceae051db0c60..3422847d72985 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -950,9 +950,9 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
   end
 
   add_index "projects", ["ci_id"], name: "index_projects_on_ci_id", using: :btree
-- 
GitLab


From 75d3ed9a43eceda6a375e1538945a20c97925841 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Mon, 10 Apr 2017 16:44:49 +0100
Subject: [PATCH 019/363] Added Spinner class

---
 .../blob/balsamiq/balsamiq_viewer.js          | 13 +++++++--
 app/assets/javascripts/spinner.js             | 28 +++++++++++++++++++
 app/assets/stylesheets/framework/files.scss   | 11 ++++++++
 3 files changed, 49 insertions(+), 3 deletions(-)
 create mode 100644 app/assets/javascripts/spinner.js

diff --git a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
index e1c837cb09efc..0065584cec5e5 100644
--- a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
+++ b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
@@ -1,11 +1,13 @@
 /* global Flash */
 
+import Spinner from '../../spinner';
 import sqljs from 'sql.js';
 
 class BalsamiqViewer {
   constructor(viewer) {
     this.viewer = viewer;
     this.endpoint = this.viewer.dataset.endpoint;
+    this.spinner = new Spinner(this.viewer);
   }
 
   loadFile() {
@@ -17,10 +19,14 @@ class BalsamiqViewer {
     xhr.onload = this.renderFile.bind(this);
     xhr.onerror = BalsamiqViewer.onError;
 
+    this.spinner.start();
+
     xhr.send();
   }
 
   renderFile(loadEvent) {
+    this.spinner.stop();
+
     const container = document.createElement('ul');
 
     this.initDatabase(loadEvent.target.response);
@@ -29,7 +35,7 @@ class BalsamiqViewer {
     const renderedPreviews = previews.map(preview => this.renderPreview(preview, container));
 
     container.innerHTML = renderedPreviews.join('');
-    container.classList.add('list-inline');
+    container.classList.add('list-inline', 'previews');
 
     this.viewer.appendChild(container);
   }
@@ -41,14 +47,15 @@ class BalsamiqViewer {
   }
 
   getPreviews() {
-    const thumnails = this.database.exec('SELECT * FROM thumbnails');
+    const thumbnails = this.database.exec('SELECT * FROM thumbnails');
 
-    return thumnails[0].values.map(BalsamiqViewer.parsePreview);
+    return thumbnails[0].values.map(BalsamiqViewer.parsePreview);
   }
 
   renderPreview(preview) {
     const previewElement = document.createElement('li');
 
+    previewElement.classList.add('preview');
     previewElement.innerHTML = this.renderTemplate(preview);
 
     return previewElement.outerHTML;
diff --git a/app/assets/javascripts/spinner.js b/app/assets/javascripts/spinner.js
new file mode 100644
index 0000000000000..b7bfe1a2572de
--- /dev/null
+++ b/app/assets/javascripts/spinner.js
@@ -0,0 +1,28 @@
+class Spinner {
+  constructor(renderable) {
+    this.renderable = renderable;
+
+    this.container = Spinner.createContainer();
+  }
+
+  start() {
+    this.renderable.prepend(this.container);
+  }
+
+  stop() {
+    this.container.remove();
+  }
+
+  static createContainer() {
+    const container = document.createElement('div');
+    container.classList.add('loading');
+
+    container.innerHTML = Spinner.TEMPLATE;
+
+    return container;
+  }
+}
+
+Spinner.TEMPLATE = '<i class="fa fa-spinner fa-spin"></i>';
+
+export default Spinner;
diff --git a/app/assets/stylesheets/framework/files.scss b/app/assets/stylesheets/framework/files.scss
index a5a8522739e3d..b8dab538feef8 100644
--- a/app/assets/stylesheets/framework/files.scss
+++ b/app/assets/stylesheets/framework/files.scss
@@ -168,6 +168,17 @@
     &.code {
       padding: 0;
     }
+
+    .list-inline.previews {
+      display: flex;
+      flex-wrap: wrap;
+      justify-content: space-between;
+      padding: $gl-padding;
+
+      .preview {
+        flex-shrink: 0;
+      }
+    }
   }
 }
 
-- 
GitLab


From 40e2be25c74aef14a94cf9dff4e80554958fd347 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Mon, 10 Apr 2017 18:20:54 +0100
Subject: [PATCH 020/363] Added unit tests

---
 .../blob/balsamiq/balsamiq_viewer.js          |   6 +-
 .../blob/balsamiq/balsamiq_viewer_spec.js     | 357 ++++++++++++++++++
 spec/javascripts/spinner_spec.js              |  97 +++++
 3 files changed, 457 insertions(+), 3 deletions(-)
 create mode 100644 spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
 create mode 100644 spec/javascripts/spinner_spec.js

diff --git a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
index 0065584cec5e5..d0c161f2aad55 100644
--- a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
+++ b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
@@ -1,7 +1,7 @@
 /* global Flash */
 
-import Spinner from '../../spinner';
 import sqljs from 'sql.js';
+import Spinner from '../../spinner';
 
 class BalsamiqViewer {
   constructor(viewer) {
@@ -32,7 +32,7 @@ class BalsamiqViewer {
     this.initDatabase(loadEvent.target.response);
 
     const previews = this.getPreviews();
-    const renderedPreviews = previews.map(preview => this.renderPreview(preview, container));
+    const renderedPreviews = previews.map(preview => this.renderPreview(preview));
 
     container.innerHTML = renderedPreviews.join('');
     container.classList.add('list-inline', 'previews');
@@ -68,7 +68,7 @@ class BalsamiqViewer {
     const name = JSON.parse(title[0].values[0][2]).name;
     const image = preview.image;
 
-    template = template.replace(/{{name}}/, name).replace(/{{image}}/, image);
+    template = template.replace(/{{name}}/g, name).replace(/{{image}}/g, image);
 
     return template;
   }
diff --git a/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js b/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
new file mode 100644
index 0000000000000..10db4175ca4e3
--- /dev/null
+++ b/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
@@ -0,0 +1,357 @@
+import sqljs from 'sql.js';
+import BalsamiqViewer from '~/blob/balsamiq/balsamiq_viewer';
+import * as spinnerSrc from '~/spinner';
+import ClassSpecHelper from '../../helpers/class_spec_helper';
+
+describe('BalsamiqViewer', () => {
+  let balsamiqViewer;
+  let endpoint;
+  let viewer;
+
+  describe('class constructor', () => {
+    beforeEach(() => {
+      endpoint = 'endpoint';
+      viewer = {
+        dataset: {
+          endpoint,
+        },
+      };
+
+      spyOn(spinnerSrc, 'default');
+
+      balsamiqViewer = new BalsamiqViewer(viewer);
+    });
+
+    it('should set .viewer', () => {
+      expect(balsamiqViewer.viewer).toBe(viewer);
+    });
+
+    it('should set .endpoint', () => {
+      expect(balsamiqViewer.endpoint).toBe(endpoint);
+    });
+
+    it('should instantiate Spinner', () => {
+      expect(spinnerSrc.default).toHaveBeenCalledWith(viewer);
+    });
+
+    it('should set .spinner', () => {
+      expect(balsamiqViewer.spinner).toEqual(jasmine.any(spinnerSrc.default));
+    });
+  });
+
+  describe('loadFile', () => {
+    let xhr;
+    let spinner;
+
+    beforeEach(() => {
+      endpoint = 'endpoint';
+      xhr = jasmine.createSpyObj('xhr', ['open', 'send']);
+      spinner = jasmine.createSpyObj('spinner', ['start']);
+
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['renderFile']);
+      balsamiqViewer.endpoint = endpoint;
+      balsamiqViewer.spinner = spinner;
+
+      spyOn(window, 'XMLHttpRequest').and.returnValue(xhr);
+
+      BalsamiqViewer.prototype.loadFile.call(balsamiqViewer);
+    });
+
+    it('should instantiate XMLHttpRequest', () => {
+      expect(window.XMLHttpRequest).toHaveBeenCalled();
+    });
+
+    it('should call .open', () => {
+      expect(xhr.open).toHaveBeenCalledWith('GET', endpoint, true);
+    });
+
+    it('should set .responseType', () => {
+      expect(xhr.responseType).toBe('arraybuffer');
+    });
+
+    it('should set .onload', () => {
+      expect(xhr.onload).toEqual(jasmine.any(Function));
+    });
+
+    it('should set .onerror', () => {
+      expect(xhr.onerror).toBe(BalsamiqViewer.onError);
+    });
+
+    it('should call spinner.start', () => {
+      expect(spinner.start).toHaveBeenCalled();
+    });
+
+    it('should call .send', () => {
+      expect(xhr.send).toHaveBeenCalled();
+    });
+  });
+
+  describe('renderFile', () => {
+    let spinner;
+    let container;
+    let loadEvent;
+    let previews;
+
+    beforeEach(() => {
+      loadEvent = { target: { response: {} } };
+      viewer = jasmine.createSpyObj('viewer', ['appendChild']);
+      spinner = jasmine.createSpyObj('spinner', ['stop']);
+      previews = [0, 1, 2];
+
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['initDatabase', 'getPreviews', 'renderPreview']);
+      balsamiqViewer.viewer = viewer;
+      balsamiqViewer.spinner = spinner;
+
+      balsamiqViewer.getPreviews.and.returnValue(previews);
+      balsamiqViewer.renderPreview.and.callFake(preview => preview);
+      viewer.appendChild.and.callFake((containerElement) => {
+        container = containerElement;
+      });
+
+      BalsamiqViewer.prototype.renderFile.call(balsamiqViewer, loadEvent);
+    });
+
+    it('should call spinner.stop', () => {
+      expect(spinner.stop).toHaveBeenCalled();
+    });
+
+    it('should call .initDatabase', () => {
+      expect(balsamiqViewer.initDatabase).toHaveBeenCalledWith(loadEvent.target.response);
+    });
+
+    it('should call .getPreviews', () => {
+      expect(balsamiqViewer.getPreviews).toHaveBeenCalled();
+    });
+
+    it('should call .renderPreview for each preview', () => {
+      const allArgs = balsamiqViewer.renderPreview.calls.allArgs();
+
+      expect(allArgs.length).toBe(3);
+
+      previews.forEach((preview, i) => {
+        expect(allArgs[i][0]).toBe(preview);
+      });
+    });
+
+    it('should set .innerHTML', () => {
+      expect(container.innerHTML).toBe('012');
+    });
+
+    it('should add inline preview classes', () => {
+      expect(container.classList[0]).toBe('list-inline');
+      expect(container.classList[1]).toBe('previews');
+    });
+
+    it('should call viewer.appendChild', () => {
+      expect(viewer.appendChild).toHaveBeenCalledWith(container);
+    });
+  });
+
+  describe('initDatabase', () => {
+    let database;
+    let uint8Array;
+    let data;
+
+    beforeEach(() => {
+      uint8Array = {};
+      database = {};
+      data = 'data';
+
+      balsamiqViewer = {};
+
+      spyOn(window, 'Uint8Array').and.returnValue(uint8Array);
+      spyOn(sqljs, 'Database').and.returnValue(database);
+
+      BalsamiqViewer.prototype.initDatabase.call(balsamiqViewer, data);
+    });
+
+    it('should instantiate Uint8Array', () => {
+      expect(window.Uint8Array).toHaveBeenCalledWith(data);
+    });
+
+    it('should call sqljs.Database', () => {
+      expect(sqljs.Database).toHaveBeenCalledWith(uint8Array);
+    });
+
+    it('should set .database', () => {
+      expect(balsamiqViewer.database).toBe(database);
+    });
+  });
+
+  describe('getPreviews', () => {
+    let database;
+    let thumbnails;
+    let getPreviews;
+
+    beforeEach(() => {
+      database = jasmine.createSpyObj('database', ['exec']);
+      thumbnails = [{ values: [0, 1, 2] }];
+
+      balsamiqViewer = {
+        database,
+      };
+
+      spyOn(BalsamiqViewer, 'parsePreview').and.callFake(preview => preview.toString());
+      database.exec.and.returnValue(thumbnails);
+
+      getPreviews = BalsamiqViewer.prototype.getPreviews.call(balsamiqViewer);
+    });
+
+    it('should call database.exec', () => {
+      expect(database.exec).toHaveBeenCalledWith('SELECT * FROM thumbnails');
+    });
+
+    it('should call .parsePreview for each value', () => {
+      const allArgs = BalsamiqViewer.parsePreview.calls.allArgs();
+
+      expect(allArgs.length).toBe(3);
+
+      thumbnails[0].values.forEach((value, i) => {
+        expect(allArgs[i][0]).toBe(value);
+      });
+    });
+
+    it('should return an array of parsed values', () => {
+      expect(getPreviews).toEqual(['0', '1', '2']);
+    });
+  });
+
+  describe('renderPreview', () => {
+    let previewElement;
+    let innerHTML;
+    let preview;
+    let renderPreview;
+
+    beforeEach(() => {
+      innerHTML = '<a>innerHTML</a>';
+      previewElement = {
+        outerHTML: '<p>outerHTML</p>',
+        classList: jasmine.createSpyObj('classList', ['add']),
+      };
+      preview = {};
+
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['renderTemplate']);
+
+      spyOn(document, 'createElement').and.returnValue(previewElement);
+      balsamiqViewer.renderTemplate.and.returnValue(innerHTML);
+
+      renderPreview = BalsamiqViewer.prototype.renderPreview.call(balsamiqViewer, preview);
+    });
+
+    it('should call document.createElement', () => {
+      expect(document.createElement).toHaveBeenCalledWith('li');
+    });
+
+    it('should call classList.add', () => {
+      expect(previewElement.classList.add).toHaveBeenCalledWith('preview');
+    });
+
+    it('should call .renderTemplate', () => {
+      expect(balsamiqViewer.renderTemplate).toHaveBeenCalledWith(preview);
+    });
+
+    it('should set .innerHTML', () => {
+      expect(previewElement.innerHTML).toBe(innerHTML);
+    });
+
+    it('should return .outerHTML', () => {
+      expect(renderPreview).toBe(previewElement.outerHTML);
+    });
+  });
+
+  describe('renderTemplate', () => {
+    let preview;
+    let database;
+    let title;
+    let renderTemplate;
+
+    beforeEach(() => {
+      preview = { reosourceID: 1, image: 'image' };
+      title = [{ values: [['{}', '{}', '{ "name": "name" }']] }];
+      database = jasmine.createSpyObj('database', ['exec']);
+
+      database.exec.and.returnValue(title);
+
+      balsamiqViewer = {
+        database,
+      };
+
+      spyOn(JSON, 'parse').and.callThrough();
+      spyOn(String.prototype, 'replace').and.callThrough();
+
+      renderTemplate = BalsamiqViewer.prototype.renderTemplate.call(balsamiqViewer, preview);
+    });
+
+    it('should call database.exec', () => {
+      expect(database.exec).toHaveBeenCalledWith(`SELECT * FROM resources WHERE id = '${preview.resourceID}'`);
+    });
+
+    it('should call JSON.parse', () => {
+      expect(JSON.parse).toHaveBeenCalledWith(title[0].values[0][2]);
+    });
+
+    it('should call String.prototype.replace', () => {
+      const allArgs = String.prototype.replace.calls.allArgs();
+
+      expect(allArgs.length).toBe(2);
+      expect(allArgs[0]).toEqual([/{{name}}/g, 'name']);
+      expect(allArgs[1]).toEqual([/{{image}}/g, 'image']);
+    });
+
+    it('should return the template string', function () {
+      const template = `
+        <div class="panel panel-default">
+          <div class="panel-heading">name</div>
+          <div class="panel-body">
+            <img class="img-thumbnail" src=""/>
+          </div>
+        </div>
+      `;
+
+      expect(renderTemplate.replace(/\s/g, '')).toEqual(template.replace(/\s/g, ''));
+    });
+  });
+
+  describe('parsePreview', () => {
+    let preview;
+    let parsePreview;
+
+    beforeEach(() => {
+      preview = ['{}', '{ "id": 1 }'];
+
+      spyOn(JSON, 'parse').and.callThrough();
+
+      parsePreview = BalsamiqViewer.parsePreview(preview);
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'parsePreview');
+
+    it('should call JSON.parse', () => {
+      expect(JSON.parse).toHaveBeenCalledWith(preview[1]);
+    });
+
+    it('should return the parsed JSON', () => {
+      expect(parsePreview).toEqual(JSON.parse('{ "id": 1 }'));
+    });
+  });
+
+  describe('onError', () => {
+    let onError;
+
+    beforeEach(() => {
+      spyOn(window, 'Flash');
+
+      onError = BalsamiqViewer.onError();
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'onError');
+
+    it('should instantiate Flash', () => {
+      expect(window.Flash).toHaveBeenCalledWith('Balsamiq file could not be loaded.');
+    });
+
+    it('should return Flash', () => {
+      expect(onError).toEqual(jasmine.any(window.Flash));
+    });
+  });
+});
diff --git a/spec/javascripts/spinner_spec.js b/spec/javascripts/spinner_spec.js
new file mode 100644
index 0000000000000..ec62503659c20
--- /dev/null
+++ b/spec/javascripts/spinner_spec.js
@@ -0,0 +1,97 @@
+import Spinner from '~/spinner';
+import ClassSpecHelper from './helpers/class_spec_helper';
+
+describe('Spinner', () => {
+  let renderable;
+  let container;
+  let spinner;
+
+  describe('class constructor', () => {
+    beforeEach(() => {
+      renderable = {};
+      container = {};
+
+      spyOn(Spinner, 'createContainer').and.returnValue(container);
+
+      spinner = new Spinner(renderable);
+    });
+
+    it('should set .renderable', () => {
+      expect(spinner.renderable).toBe(renderable);
+    });
+
+    it('should call Spinner.createContainer', () => {
+      expect(Spinner.createContainer).toHaveBeenCalled();
+    });
+
+    it('should set .container', () => {
+      expect(spinner.container).toBe(container);
+    });
+  });
+
+  describe('start', () => {
+    beforeEach(() => {
+      renderable = jasmine.createSpyObj('renderable', ['prepend']);
+      container = {};
+
+      spinner = {
+        renderable,
+        container,
+      };
+
+      Spinner.prototype.start.call(spinner);
+    });
+
+    it('should call .prepend', () => {
+      expect(renderable.prepend).toHaveBeenCalledWith(container);
+    });
+  });
+
+  describe('stop', () => {
+    beforeEach(() => {
+      container = jasmine.createSpyObj('container', ['remove']);
+
+      spinner = {
+        container,
+      };
+
+      Spinner.prototype.stop.call(spinner);
+    });
+
+    it('should call .remove', () => {
+      expect(container.remove).toHaveBeenCalled();
+    });
+  });
+
+  describe('createContainer', () => {
+    let createContainer;
+
+    beforeEach(() => {
+      container = {
+        classList: jasmine.createSpyObj('classList', ['add']),
+      };
+
+      spyOn(document, 'createElement').and.returnValue(container);
+
+      createContainer = Spinner.createContainer();
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(Spinner, 'createContainer');
+
+    it('should call document.createElement', () => {
+      expect(document.createElement).toHaveBeenCalledWith('div');
+    });
+
+    it('should call classList.add', () => {
+      expect(container.classList.add).toHaveBeenCalledWith('loading');
+    });
+
+    it('should return the container element', () => {
+      expect(createContainer).toBe(container);
+    });
+
+    it('should set the container .innerHTML to Spinner.TEMPLATE', () => {
+      expect(container.innerHTML).toBe(Spinner.TEMPLATE);
+    });
+  });
+});
-- 
GitLab


From b72e18b04fb6a054368a065c4c612505e842b526 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 11 Apr 2017 10:09:47 +0100
Subject: [PATCH 021/363] Added feature test

---
 .../projects/blobs/balsamiq_preview_spec.rb   | 24 +++++++++++++++++++
 1 file changed, 24 insertions(+)
 create mode 100644 spec/features/projects/blobs/balsamiq_preview_spec.rb

diff --git a/spec/features/projects/blobs/balsamiq_preview_spec.rb b/spec/features/projects/blobs/balsamiq_preview_spec.rb
new file mode 100644
index 0000000000000..856ec3271a0f0
--- /dev/null
+++ b/spec/features/projects/blobs/balsamiq_preview_spec.rb
@@ -0,0 +1,24 @@
+require 'spec_helper'
+
+feature 'Balsamiq preview', :feature, :js do
+  include TreeHelper
+
+  let(:user) { create(:user) }
+  let(:project) { create(:project) }
+  let(:branch) { 'add-balsamiq-file' }
+  let(:path) { 'files/images/balsamiq.bmpr' }
+
+  before do
+    project.add_master(user)
+
+    login_as user
+
+    p namespace_project_blob_path(project.namespace, project, tree_join(branch, path))
+
+    visit namespace_project_blob_path(project.namespace, project, tree_join(branch, path))
+  end
+
+  it 'should' do
+    screenshot_and_open_image
+  end
+end
-- 
GitLab


From 897d8d547c5888bc63d3c3ecc0d0dd971d70e6c0 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 11 Apr 2017 11:28:17 +0100
Subject: [PATCH 022/363] Further review changes

---
 .../blob/balsamiq/balsamiq_viewer.js          |  28 +++--
 .../blob/balsamiq/balsamiq_viewer_spec.js     | 110 +++++++++++++-----
 2 files changed, 98 insertions(+), 40 deletions(-)

diff --git a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
index d0c161f2aad55..3885b0f43b2bb 100644
--- a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
+++ b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
@@ -1,6 +1,7 @@
 /* global Flash */
 
 import sqljs from 'sql.js';
+import _ from 'underscore';
 import Spinner from '../../spinner';
 
 class BalsamiqViewer {
@@ -52,6 +53,10 @@ class BalsamiqViewer {
     return thumbnails[0].values.map(BalsamiqViewer.parsePreview);
   }
 
+  getTitle(resourceID) {
+    return this.database.exec(`SELECT * FROM resources WHERE id = '${resourceID}'`);
+  }
+
   renderPreview(preview) {
     const previewElement = document.createElement('li');
 
@@ -62,13 +67,14 @@ class BalsamiqViewer {
   }
 
   renderTemplate(preview) {
-    let template = BalsamiqViewer.PREVIEW_TEMPLATE;
-
-    const title = this.database.exec(`SELECT * FROM resources WHERE id = '${preview.resourceID}'`);
-    const name = JSON.parse(title[0].values[0][2]).name;
+    const title = this.getTitle(preview.resourceID);
+    const name = BalsamiqViewer.parseTitle(title);
     const image = preview.image;
 
-    template = template.replace(/{{name}}/g, name).replace(/{{image}}/g, image);
+    const template = BalsamiqViewer.PREVIEW_TEMPLATE({
+      name,
+      image,
+    });
 
     return template;
   }
@@ -77,6 +83,10 @@ class BalsamiqViewer {
     return JSON.parse(preview[1]);
   }
 
+  static parseTitle(title) {
+    return JSON.parse(title[0].values[0][2]).name;
+  }
+
   static onError() {
     const flash = new Flash('Balsamiq file could not be loaded.');
 
@@ -84,13 +94,13 @@ class BalsamiqViewer {
   }
 }
 
-BalsamiqViewer.PREVIEW_TEMPLATE = `
+BalsamiqViewer.PREVIEW_TEMPLATE = _.template(`
   <div class="panel panel-default">
-    <div class="panel-heading">{{name}}</div>
+    <div class="panel-heading"><%- name %></div>
     <div class="panel-body">
-      <img class="img-thumbnail" src="data:image/png;base64,{{image}}"/>
+      <img class="img-thumbnail" src="data:image/png;base64,<%- image %>"/>
     </div>
   </div>
-`;
+`);
 
 export default BalsamiqViewer;
diff --git a/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js b/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
index 10db4175ca4e3..557eb721a2b8c 100644
--- a/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
+++ b/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
@@ -216,6 +216,35 @@ describe('BalsamiqViewer', () => {
     });
   });
 
+  describe('getTitle', () => {
+    let database;
+    let resourceID;
+    let resource;
+    let getTitle;
+
+    beforeEach(() => {
+      database = jasmine.createSpyObj('database', ['exec']);
+      resourceID = 4;
+      resource = 'resource';
+
+      balsamiqViewer = {
+        database,
+      };
+
+      database.exec.and.returnValue(resource);
+
+      getTitle = BalsamiqViewer.prototype.getTitle.call(balsamiqViewer, resourceID);
+    });
+
+    it('should call database.exec', () => {
+      expect(database.exec).toHaveBeenCalledWith(`SELECT * FROM resources WHERE id = '${resourceID}'`);
+    });
+
+    it('should return the selected resource', () => {
+      expect(getTitle).toBe(resource);
+    });
+  });
+
   describe('renderPreview', () => {
     let previewElement;
     let innerHTML;
@@ -261,54 +290,50 @@ describe('BalsamiqViewer', () => {
 
   describe('renderTemplate', () => {
     let preview;
-    let database;
+    let name;
     let title;
+    let template;
     let renderTemplate;
 
     beforeEach(() => {
-      preview = { reosourceID: 1, image: 'image' };
-      title = [{ values: [['{}', '{}', '{ "name": "name" }']] }];
-      database = jasmine.createSpyObj('database', ['exec']);
-
-      database.exec.and.returnValue(title);
+      preview = { resourceID: 1, image: 'image' };
+      name = 'name';
+      title = 'title';
+      template = `
+        <div class="panel panel-default">
+          <div class="panel-heading">name</div>
+          <div class="panel-body">
+            <img class="img-thumbnail" src=""/>
+          </div>
+        </div>
+      `;
 
-      balsamiqViewer = {
-        database,
-      };
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['getTitle']);
 
-      spyOn(JSON, 'parse').and.callThrough();
-      spyOn(String.prototype, 'replace').and.callThrough();
+      spyOn(BalsamiqViewer, 'parseTitle').and.returnValue(name);
+      spyOn(BalsamiqViewer, 'PREVIEW_TEMPLATE').and.returnValue(template);
+      balsamiqViewer.getTitle.and.returnValue(title);
 
       renderTemplate = BalsamiqViewer.prototype.renderTemplate.call(balsamiqViewer, preview);
     });
 
-    it('should call database.exec', () => {
-      expect(database.exec).toHaveBeenCalledWith(`SELECT * FROM resources WHERE id = '${preview.resourceID}'`);
+    it('should call .getTitle', () => {
+      expect(balsamiqViewer.getTitle).toHaveBeenCalledWith(preview.resourceID);
     });
 
-    it('should call JSON.parse', () => {
-      expect(JSON.parse).toHaveBeenCalledWith(title[0].values[0][2]);
+    it('should call .parseTitle', () => {
+      expect(BalsamiqViewer.parseTitle).toHaveBeenCalledWith(title);
     });
 
-    it('should call String.prototype.replace', () => {
-      const allArgs = String.prototype.replace.calls.allArgs();
-
-      expect(allArgs.length).toBe(2);
-      expect(allArgs[0]).toEqual([/{{name}}/g, 'name']);
-      expect(allArgs[1]).toEqual([/{{image}}/g, 'image']);
+    it('should call .PREVIEW_TEMPLATE', () => {
+      expect(BalsamiqViewer.PREVIEW_TEMPLATE).toHaveBeenCalledWith({
+        name,
+        image: preview.image,
+      });
     });
 
     it('should return the template string', function () {
-      const template = `
-        <div class="panel panel-default">
-          <div class="panel-heading">name</div>
-          <div class="panel-body">
-            <img class="img-thumbnail" src=""/>
-          </div>
-        </div>
-      `;
-
-      expect(renderTemplate.replace(/\s/g, '')).toEqual(template.replace(/\s/g, ''));
+      expect(renderTemplate.trim()).toBe(template.trim());
     });
   });
 
@@ -335,6 +360,29 @@ describe('BalsamiqViewer', () => {
     });
   });
 
+  describe('parseTitle', () => {
+    let title;
+    let parseTitle;
+
+    beforeEach(() => {
+      title = [{ values: [['{}', '{}', '{"name":"name"}']] }];
+
+      spyOn(JSON, 'parse').and.callThrough();
+
+      parseTitle = BalsamiqViewer.parseTitle(title);
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'parsePreview');
+
+    it('should call JSON.parse', () => {
+      expect(JSON.parse).toHaveBeenCalledWith(title[0].values[0][2]);
+    });
+
+    it('should return the name value', () => {
+      expect(parseTitle).toBe('name');
+    });
+  });
+
   describe('onError', () => {
     let onError;
 
-- 
GitLab


From fbed2909091b98f614ae51c5d6503cdd40a74eb5 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 11 Apr 2017 11:42:47 +0100
Subject: [PATCH 023/363] Removed prepend in favour of clean and appendChild

---
 app/assets/javascripts/spinner.js |  3 ++-
 spec/javascripts/spinner_spec.js  | 10 +++++++---
 2 files changed, 9 insertions(+), 4 deletions(-)

diff --git a/app/assets/javascripts/spinner.js b/app/assets/javascripts/spinner.js
index b7bfe1a2572de..6b5ac89a57641 100644
--- a/app/assets/javascripts/spinner.js
+++ b/app/assets/javascripts/spinner.js
@@ -6,7 +6,8 @@ class Spinner {
   }
 
   start() {
-    this.renderable.prepend(this.container);
+    this.renderable.innerHTML = '';
+    this.renderable.appendChild(this.container);
   }
 
   stop() {
diff --git a/spec/javascripts/spinner_spec.js b/spec/javascripts/spinner_spec.js
index ec62503659c20..f550285e0f76d 100644
--- a/spec/javascripts/spinner_spec.js
+++ b/spec/javascripts/spinner_spec.js
@@ -31,7 +31,7 @@ describe('Spinner', () => {
 
   describe('start', () => {
     beforeEach(() => {
-      renderable = jasmine.createSpyObj('renderable', ['prepend']);
+      renderable = jasmine.createSpyObj('renderable', ['appendChild']);
       container = {};
 
       spinner = {
@@ -42,8 +42,12 @@ describe('Spinner', () => {
       Spinner.prototype.start.call(spinner);
     });
 
-    it('should call .prepend', () => {
-      expect(renderable.prepend).toHaveBeenCalledWith(container);
+    it('should set .innerHTML to an empty string', () => {
+      expect(renderable.innerHTML).toEqual('');
+    });
+
+    it('should call .appendChild', () => {
+      expect(renderable.appendChild).toHaveBeenCalledWith(container);
     });
   });
 
-- 
GitLab


From 18751fb1356935f38bf1ff0f00350b276d77169d Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 11 Apr 2017 13:06:32 +0100
Subject: [PATCH 024/363] Only import template function from underscore

---
 app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
index 3885b0f43b2bb..5bcd7d5eccc07 100644
--- a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
+++ b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
@@ -1,7 +1,7 @@
 /* global Flash */
 
 import sqljs from 'sql.js';
-import _ from 'underscore';
+import { template as _template } from 'underscore';
 import Spinner from '../../spinner';
 
 class BalsamiqViewer {
@@ -94,7 +94,7 @@ class BalsamiqViewer {
   }
 }
 
-BalsamiqViewer.PREVIEW_TEMPLATE = _.template(`
+BalsamiqViewer.PREVIEW_TEMPLATE = _template(`
   <div class="panel panel-default">
     <div class="panel-heading"><%- name %></div>
     <div class="panel-body">
-- 
GitLab


From cf678945044f657111997a92330420a0110d03fc Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 11 Apr 2017 13:06:45 +0100
Subject: [PATCH 025/363] Finished feature specs

---
 app/views/projects/blob/_bmpr.html.haml               |  2 +-
 spec/features/projects/blobs/balsamiq_preview_spec.rb | 11 +++++++----
 spec/support/test_env.rb                              |  1 +
 3 files changed, 9 insertions(+), 5 deletions(-)

diff --git a/app/views/projects/blob/_bmpr.html.haml b/app/views/projects/blob/_bmpr.html.haml
index 36ec0cbcce8bb..573b24ae44f7f 100644
--- a/app/views/projects/blob/_bmpr.html.haml
+++ b/app/views/projects/blob/_bmpr.html.haml
@@ -1,4 +1,4 @@
 - content_for :page_specific_javascripts do
   = page_specific_javascript_bundle_tag('balsamiq_viewer')
 
-.file-content#js-balsamiq-viewer{ data: { endpoint: namespace_project_raw_path(@project.namespace, @project, @id) } }
+.file-content.balsamiq-viewer#js-balsamiq-viewer{ data: { endpoint: namespace_project_raw_path(@project.namespace, @project, @id) } }
diff --git a/spec/features/projects/blobs/balsamiq_preview_spec.rb b/spec/features/projects/blobs/balsamiq_preview_spec.rb
index 856ec3271a0f0..503f1f94915f6 100644
--- a/spec/features/projects/blobs/balsamiq_preview_spec.rb
+++ b/spec/features/projects/blobs/balsamiq_preview_spec.rb
@@ -7,18 +7,21 @@
   let(:project) { create(:project) }
   let(:branch) { 'add-balsamiq-file' }
   let(:path) { 'files/images/balsamiq.bmpr' }
+  let(:file_content) { find('.file-content') }
 
   before do
     project.add_master(user)
 
     login_as user
 
-    p namespace_project_blob_path(project.namespace, project, tree_join(branch, path))
-
     visit namespace_project_blob_path(project.namespace, project, tree_join(branch, path))
   end
 
-  it 'should' do
-    screenshot_and_open_image
+  it 'should show a loading icon' do
+    expect(file_content).to have_selector('.loading')
+  end
+
+  it 'should show a viewer container' do
+    expect(page).to have_selector('.balsamiq-viewer')
   end
 end
diff --git a/spec/support/test_env.rb b/spec/support/test_env.rb
index 1b5cb71a6b052..5765a31ca18e0 100644
--- a/spec/support/test_env.rb
+++ b/spec/support/test_env.rb
@@ -27,6 +27,7 @@ module TestEnv
     'expand-collapse-files'              => '025db92',
     'expand-collapse-lines'              => '238e82d',
     'video'                              => '8879059',
+    'add-balsamiq-file'                  => 'b89b56d',
     'crlf-diff'                          => '5938907',
     'conflict-start'                     => '824be60',
     'conflict-resolvable'                => '1450cd6',
-- 
GitLab


From ccba2f445dc0efd670a492cef954169a117eb166 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 12 Apr 2017 07:22:52 +0100
Subject: [PATCH 026/363] Spec review changes

---
 spec/features/projects/blobs/balsamiq_preview_spec.rb | 2 --
 1 file changed, 2 deletions(-)

diff --git a/spec/features/projects/blobs/balsamiq_preview_spec.rb b/spec/features/projects/blobs/balsamiq_preview_spec.rb
index 503f1f94915f6..551d2045744c3 100644
--- a/spec/features/projects/blobs/balsamiq_preview_spec.rb
+++ b/spec/features/projects/blobs/balsamiq_preview_spec.rb
@@ -11,9 +11,7 @@
 
   before do
     project.add_master(user)
-
     login_as user
-
     visit namespace_project_blob_path(project.namespace, project, tree_join(branch, path))
   end
 
-- 
GitLab


From 04eaed8088a398aee0954935752a99ac7721bb4f Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 12 Apr 2017 08:11:12 +0100
Subject: [PATCH 027/363] Used underscore to template list children to utilize
 their simple escaped interpolation

---
 app/assets/javascripts/droplab/constants.js      |  2 ++
 app/assets/javascripts/droplab/drop_down.js      |  2 +-
 app/assets/javascripts/droplab/utils.js          | 16 ++++++++--------
 .../javascripts/filtered_search/dropdown_hint.js |  2 +-
 4 files changed, 12 insertions(+), 10 deletions(-)

diff --git a/app/assets/javascripts/droplab/constants.js b/app/assets/javascripts/droplab/constants.js
index a23d914772a89..34b1aee73c7a9 100644
--- a/app/assets/javascripts/droplab/constants.js
+++ b/app/assets/javascripts/droplab/constants.js
@@ -2,10 +2,12 @@ const DATA_TRIGGER = 'data-dropdown-trigger';
 const DATA_DROPDOWN = 'data-dropdown';
 const SELECTED_CLASS = 'droplab-item-selected';
 const ACTIVE_CLASS = 'droplab-item-active';
+const TEMPLATE_REGEX = /\{\{(.+?)\}\}/g;
 
 export {
   DATA_TRIGGER,
   DATA_DROPDOWN,
   SELECTED_CLASS,
   ACTIVE_CLASS,
+  TEMPLATE_REGEX,
 };
diff --git a/app/assets/javascripts/droplab/drop_down.js b/app/assets/javascripts/droplab/drop_down.js
index 9588921ebcd8d..084d57e2e1f3d 100644
--- a/app/assets/javascripts/droplab/drop_down.js
+++ b/app/assets/javascripts/droplab/drop_down.js
@@ -93,7 +93,7 @@ Object.assign(DropDown.prototype, {
   },
 
   renderChildren: function(data) {
-    var html = utils.t(this.templateString, data);
+    var html = utils.template(this.templateString, data);
     var template = document.createElement('div');
 
     template.innerHTML = html;
diff --git a/app/assets/javascripts/droplab/utils.js b/app/assets/javascripts/droplab/utils.js
index c149a33a1e9ed..4da7344604eda 100644
--- a/app/assets/javascripts/droplab/utils.js
+++ b/app/assets/javascripts/droplab/utils.js
@@ -1,19 +1,19 @@
 /* eslint-disable */
 
-import { DATA_TRIGGER, DATA_DROPDOWN } from './constants';
+import { template as _template } from 'underscore';
+import { DATA_TRIGGER, DATA_DROPDOWN, TEMPLATE_REGEX } from './constants';
 
 const utils = {
   toCamelCase(attr) {
     return this.camelize(attr.split('-').slice(1).join(' '));
   },
 
-  t(s, d) {
-    for (const p in d) {
-      if (Object.prototype.hasOwnProperty.call(d, p)) {
-        s = s.replace(new RegExp(`{{${p}}}`, 'g'), d[p]);
-      }
-    }
-    return s;
+  template(templateString, data) {
+    const template = _template(templateString, {
+      escape: TEMPLATE_REGEX,
+    });
+
+    return template(data);
   },
 
   camelize(str) {
diff --git a/app/assets/javascripts/filtered_search/dropdown_hint.js b/app/assets/javascripts/filtered_search/dropdown_hint.js
index 381c40c03d802..5b7b059666a3b 100644
--- a/app/assets/javascripts/filtered_search/dropdown_hint.js
+++ b/app/assets/javascripts/filtered_search/dropdown_hint.js
@@ -63,7 +63,7 @@ require('./filtered_search_dropdown');
             Object.assign({
               icon: `fa-${icon}`,
               hint,
-              tag: `&lt;${tag}&gt;`,
+              tag: `<${tag}>`,
             }, type && { type }),
           );
         }
-- 
GitLab


From 606275da6f1245e7e2dd790a2679653aea6b9a11 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 12 Apr 2017 09:18:22 +0100
Subject: [PATCH 028/363] Update specs

---
 spec/javascripts/droplab/constants_spec.js | 6 ++++++
 spec/javascripts/droplab/drop_down_spec.js | 4 ++--
 2 files changed, 8 insertions(+), 2 deletions(-)

diff --git a/spec/javascripts/droplab/constants_spec.js b/spec/javascripts/droplab/constants_spec.js
index 35239e4fb8eff..c526e98748ebd 100644
--- a/spec/javascripts/droplab/constants_spec.js
+++ b/spec/javascripts/droplab/constants_spec.js
@@ -26,4 +26,10 @@ describe('constants', function () {
       expect(constants.ACTIVE_CLASS).toBe('droplab-item-active');
     });
   });
+
+  describe('TEMPLATE_REGEX', function () {
+    it('should be a handlebars templating syntax regex', function() {
+      expect(constants.TEMPLATE_REGEX).toEqual(/\{\{(.+?)\}\}/g);
+    });
+  });
 });
diff --git a/spec/javascripts/droplab/drop_down_spec.js b/spec/javascripts/droplab/drop_down_spec.js
index 802e2435672ea..1b658aa670125 100644
--- a/spec/javascripts/droplab/drop_down_spec.js
+++ b/spec/javascripts/droplab/drop_down_spec.js
@@ -429,7 +429,7 @@ describe('DropDown', function () {
       this.html = 'html';
       this.template = { firstChild: { outerHTML: 'outerHTML', style: {} } };
 
-      spyOn(utils, 't').and.returnValue(this.html);
+      spyOn(utils, 'template').and.returnValue(this.html);
       spyOn(document, 'createElement').and.returnValue(this.template);
       spyOn(this.dropdown, 'setImagesSrc');
 
@@ -437,7 +437,7 @@ describe('DropDown', function () {
     });
 
     it('should call utils.t with .templateString and data', function () {
-      expect(utils.t).toHaveBeenCalledWith(this.templateString, this.data);
+      expect(utils.template).toHaveBeenCalledWith(this.templateString, this.data);
     });
 
     it('should call document.createElement', function () {
-- 
GitLab


From 0453d6d7ae72f0179d02ff5190122a86e304e1bb Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 12 Apr 2017 09:50:54 +0100
Subject: [PATCH 029/363] BE review changes

---
 app/models/blob.rb                                    | 2 +-
 spec/features/projects/blobs/balsamiq_preview_spec.rb | 3 +--
 2 files changed, 2 insertions(+), 3 deletions(-)

diff --git a/app/models/blob.rb b/app/models/blob.rb
index 91cf171a6880b..82333b6f3696c 100644
--- a/app/models/blob.rb
+++ b/app/models/blob.rb
@@ -63,7 +63,7 @@ def sketch?
   end
 
   def balsamiq?
-    binary? && extname.downcase.delete('.') == 'bmpr'
+    binary? && extension == 'bmpr'
   end
 
   def stl?
diff --git a/spec/features/projects/blobs/balsamiq_preview_spec.rb b/spec/features/projects/blobs/balsamiq_preview_spec.rb
index 551d2045744c3..59475525b813f 100644
--- a/spec/features/projects/blobs/balsamiq_preview_spec.rb
+++ b/spec/features/projects/blobs/balsamiq_preview_spec.rb
@@ -7,7 +7,6 @@
   let(:project) { create(:project) }
   let(:branch) { 'add-balsamiq-file' }
   let(:path) { 'files/images/balsamiq.bmpr' }
-  let(:file_content) { find('.file-content') }
 
   before do
     project.add_master(user)
@@ -16,7 +15,7 @@
   end
 
   it 'should show a loading icon' do
-    expect(file_content).to have_selector('.loading')
+    expect(find('.file-content')).to have_selector('.loading')
   end
 
   it 'should show a viewer container' do
-- 
GitLab


From b09465f38d66d7ff6074843177bcdb7d72caf07f Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 12 Apr 2017 11:26:18 +0200
Subject: [PATCH 030/363] Implement new rule for manual actions in policies

---
 app/policies/ci/build_policy.rb       | 14 +++++++
 spec/policies/ci/build_policy_spec.rb | 53 +++++++++++++++++++++++++++
 2 files changed, 67 insertions(+)

diff --git a/app/policies/ci/build_policy.rb b/app/policies/ci/build_policy.rb
index 8b25332b73cee..0522cbdb33142 100644
--- a/app/policies/ci/build_policy.rb
+++ b/app/policies/ci/build_policy.rb
@@ -8,6 +8,20 @@ def rules
       %w[read create update admin].each do |rule|
         cannot! :"#{rule}_commit_status" unless can? :"#{rule}_build"
       end
+
+      can! :play_build if can_play_action?
+    end
+
+    private
+
+    alias_method :build, :subject
+
+    def can_play_action?
+      return false unless build.playable?
+
+      ::Gitlab::UserAccess
+        .new(user, project: build.project)
+        .can_push_to_branch?(build.ref)
     end
   end
 end
diff --git a/spec/policies/ci/build_policy_spec.rb b/spec/policies/ci/build_policy_spec.rb
index 0f280f32eac91..e4693cdcef0e0 100644
--- a/spec/policies/ci/build_policy_spec.rb
+++ b/spec/policies/ci/build_policy_spec.rb
@@ -89,5 +89,58 @@
         end
       end
     end
+
+    describe 'rules for manual actions' do
+      let(:project) { create(:project) }
+
+      before do
+        project.add_developer(user)
+      end
+
+      context 'when branch build is assigned to is protected' do
+        before do
+          create(:protected_branch, :no_one_can_push,
+                 name: 'some-ref', project: project)
+        end
+
+        context 'when build is a manual action' do
+          let(:build) do
+            create(:ci_build, :manual, ref: 'some-ref', pipeline: pipeline)
+          end
+
+          it 'does not include ability to play build' do
+            expect(policies).not_to include :play_build
+          end
+        end
+
+        context 'when build is not a manual action' do
+          let(:build) do
+            create(:ci_build, ref: 'some-ref', pipeline: pipeline)
+          end
+
+          it 'does not include ability to play build' do
+            expect(policies).not_to include :play_build
+          end
+        end
+      end
+
+      context 'when branch build is assigned to is not protected' do
+        context 'when build is a manual action' do
+          let(:build) { create(:ci_build, :manual, pipeline: pipeline) }
+
+          it 'includes ability to play build' do
+            expect(policies).to include :play_build
+          end
+        end
+
+        context 'when build is not a manual action' do
+          let(:build) { create(:ci_build,  pipeline: pipeline) }
+
+          it 'does not include ability to play build' do
+            expect(policies).not_to include :play_build
+          end
+        end
+      end
+    end
   end
 end
-- 
GitLab


From 6c6bc400d1d8a96f6e443788cd0b2c14addd88e3 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 12 Apr 2017 11:46:24 +0200
Subject: [PATCH 031/363] Move code for playing an action to separate service

---
 app/models/ci/build.rb                | 15 +++------------
 app/policies/ci/build_policy.rb       |  2 +-
 app/services/ci/play_build_service.rb | 17 +++++++++++++++++
 3 files changed, 21 insertions(+), 13 deletions(-)
 create mode 100644 app/services/ci/play_build_service.rb

diff --git a/app/models/ci/build.rb b/app/models/ci/build.rb
index 159b3b2e10153..9edc4cd96b9a8 100644
--- a/app/models/ci/build.rb
+++ b/app/models/ci/build.rb
@@ -122,18 +122,9 @@ def can_play?(current_user)
     end
 
     def play(current_user)
-      unless can_play?(current_user)
-        raise Gitlab::Access::AccessDeniedError
-      end
-
-      # Try to queue a current build
-      if self.enqueue
-        self.update(user: current_user)
-        self
-      else
-        # Otherwise we need to create a duplicate
-        Ci::Build.retry(self, current_user)
-      end
+      Ci::PlayBuildService
+        .new(project, current_user)
+        .execute(self)
     end
 
     def cancelable?
diff --git a/app/policies/ci/build_policy.rb b/app/policies/ci/build_policy.rb
index 0522cbdb33142..2c39d31488f05 100644
--- a/app/policies/ci/build_policy.rb
+++ b/app/policies/ci/build_policy.rb
@@ -17,7 +17,7 @@ def rules
     alias_method :build, :subject
 
     def can_play_action?
-      return false unless build.playable?
+      return false unless build.action?
 
       ::Gitlab::UserAccess
         .new(user, project: build.project)
diff --git a/app/services/ci/play_build_service.rb b/app/services/ci/play_build_service.rb
new file mode 100644
index 0000000000000..c9ed45408f223
--- /dev/null
+++ b/app/services/ci/play_build_service.rb
@@ -0,0 +1,17 @@
+module Ci
+  class PlayBuildService < ::BaseService
+    def execute(build)
+      unless can?(current_user, :play_build, build)
+        raise Gitlab::Access::AccessDeniedError
+      end
+
+      # Try to enqueue thebuild, otherwise create a duplicate.
+      #
+      if build.enqueue
+        build.tap { |action| action.update(user: current_user) }
+      else
+        Ci::Build.retry(build, current_user)
+      end
+    end
+  end
+end
-- 
GitLab


From 7fc6b5b6ff23e2faba7f06a1362ada31f6f3436a Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 12 Apr 2017 12:19:39 +0200
Subject: [PATCH 032/363] Do not inherit build policy in pipeline policy

---
 app/policies/base_policy.rb        | 4 ++++
 app/policies/ci/pipeline_policy.rb | 5 ++++-
 2 files changed, 8 insertions(+), 1 deletion(-)

diff --git a/app/policies/base_policy.rb b/app/policies/base_policy.rb
index 8890409d0565e..623424c63e087 100644
--- a/app/policies/base_policy.rb
+++ b/app/policies/base_policy.rb
@@ -97,6 +97,10 @@ def anonymous_rules
     rules
   end
 
+  def rules
+    raise NotImplementedError
+  end
+
   def delegate!(new_subject)
     @rule_set.merge(Ability.allowed(@user, new_subject))
   end
diff --git a/app/policies/ci/pipeline_policy.rb b/app/policies/ci/pipeline_policy.rb
index 3d2eef1c50cf9..10aa2d3e72a5d 100644
--- a/app/policies/ci/pipeline_policy.rb
+++ b/app/policies/ci/pipeline_policy.rb
@@ -1,4 +1,7 @@
 module Ci
-  class PipelinePolicy < BuildPolicy
+  class PipelinePolicy < BasePolicy
+    def rules
+      delegate! @subject.project
+    end
   end
 end
-- 
GitLab


From 55aa727eff50a9472405b302645abb54f28bdba0 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 12 Apr 2017 13:36:35 +0200
Subject: [PATCH 033/363] Extract build play specs and extend test examples

---
 spec/models/ci/build_spec.rb                |  36 +------
 spec/services/ci/play_build_service_spec.rb | 105 ++++++++++++++++++++
 2 files changed, 107 insertions(+), 34 deletions(-)
 create mode 100644 spec/services/ci/play_build_service_spec.rb

diff --git a/spec/models/ci/build_spec.rb b/spec/models/ci/build_spec.rb
index 8124d263fd417..09e9f5bd8ba77 100644
--- a/spec/models/ci/build_spec.rb
+++ b/spec/models/ci/build_spec.rb
@@ -959,40 +959,8 @@ def create_mr(build, pipeline, factory: :merge_request, created_at: Time.now)
       project.add_developer(user)
     end
 
-    context 'when user does not have ability to trigger action' do
-      before do
-        create(:protected_branch, :no_one_can_push,
-               name: build.ref, project: project)
-      end
-
-      it 'raises an error' do
-        expect { build.play(user) }
-          .to raise_error Gitlab::Access::AccessDeniedError
-      end
-    end
-
-    context 'when user has ability to trigger manual action' do
-      context 'when build is manual' do
-        it 'enqueues a build' do
-          new_build = build.play(user)
-
-          expect(new_build).to be_pending
-          expect(new_build).to eq(build)
-        end
-      end
-
-      context 'when build is not manual' do
-        before do
-          build.update(status: 'success')
-        end
-
-        it 'creates a new build' do
-          new_build = build.play(user)
-
-          expect(new_build).to be_pending
-          expect(new_build).not_to eq(build)
-        end
-      end
+    it 'enqueues the build' do
+      expect(build.play(user)).to be_pending
     end
   end
 
diff --git a/spec/services/ci/play_build_service_spec.rb b/spec/services/ci/play_build_service_spec.rb
new file mode 100644
index 0000000000000..d6f9fa42045b3
--- /dev/null
+++ b/spec/services/ci/play_build_service_spec.rb
@@ -0,0 +1,105 @@
+require 'spec_helper'
+
+describe Ci::PlayBuildService, '#execute', :services do
+  let(:user) { create(:user) }
+  let(:project) { create(:empty_project) }
+  let(:pipeline) { create(:ci_pipeline, project: project) }
+  let(:build) { create(:ci_build, :manual, pipeline: pipeline) }
+
+  let(:service) do
+    described_class.new(project, user)
+  end
+
+  context 'when project does not have repository yet' do
+    let(:project) { create(:empty_project) }
+
+    it 'allows user with master role to play build' do
+      project.add_master(user)
+
+      service.execute(build)
+
+      expect(build.reload).to be_pending
+    end
+
+    it 'does not allow user with developer role to play build' do
+      project.add_developer(user)
+
+      expect { service.execute(build) }
+        .to raise_error Gitlab::Access::AccessDeniedError
+    end
+  end
+
+  context 'when project has repository' do
+    let(:project) { create(:project) }
+
+    it 'allows user with developer role to play a build' do
+      project.add_developer(user)
+
+      service.execute(build)
+
+      expect(build.reload).to be_pending
+    end
+  end
+
+  context 'when build is a playable manual action' do
+    let(:build) { create(:ci_build, :manual, pipeline: pipeline) }
+
+    before do
+      project.add_master(user)
+    end
+
+    it 'enqueues the build' do
+      expect(service.execute(build)).to eq build
+      expect(build.reload).to be_pending
+    end
+
+    it 'reassignes build user correctly' do
+      service.execute(build)
+
+      expect(build.reload.user).to eq user
+    end
+  end
+
+  context 'when build is not a playable manual action' do
+    let(:build) { create(:ci_build, when: :manual, pipeline: pipeline) }
+
+    before do
+      project.add_master(user)
+    end
+
+    it 'duplicates the build' do
+      duplicate = service.execute(build)
+
+      expect(duplicate).not_to eq build
+      expect(duplicate).to be_pending
+    end
+
+    it 'assigns users correctly' do
+      duplicate = service.execute(build)
+
+      expect(build.user).not_to eq user
+      expect(duplicate.user).to eq user
+    end
+  end
+
+  context 'when build is not action' do
+    let(:build) { create(:ci_build, :success, pipeline: pipeline) }
+
+    it 'raises an error' do
+      expect { service.execute(build) }
+        .to raise_error Gitlab::Access::AccessDeniedError
+    end
+  end
+
+  context 'when user does not have ability to trigger action' do
+    before do
+      create(:protected_branch, :no_one_can_push,
+             name: build.ref, project: project)
+    end
+
+    it 'raises an error' do
+      expect { service.execute(build) }
+        .to raise_error Gitlab::Access::AccessDeniedError
+    end
+  end
+end
-- 
GitLab


From 2aa211fa69ffd02ba11757e06e19d34f6ca51865 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 12 Apr 2017 13:48:43 +0200
Subject: [PATCH 034/363] Use build policy to determine if user can play build

---
 app/models/ci/build.rb                        |  6 -----
 app/models/environment.rb                     |  6 -----
 app/serializers/build_action_entity.rb        |  2 +-
 app/serializers/build_entity.rb               |  2 +-
 app/services/ci/stop_environments_service.rb  |  3 ++-
 app/views/projects/ci/builds/_build.html.haml |  2 +-
 lib/gitlab/ci/status/build/play.rb            |  2 +-
 spec/models/ci/build_spec.rb                  | 27 -------------------
 spec/models/environment_spec.rb               | 25 -----------------
 9 files changed, 6 insertions(+), 69 deletions(-)

diff --git a/app/models/ci/build.rb b/app/models/ci/build.rb
index 9edc4cd96b9a8..b3acb25b9ce01 100644
--- a/app/models/ci/build.rb
+++ b/app/models/ci/build.rb
@@ -115,12 +115,6 @@ def has_commands?
       commands.present?
     end
 
-    def can_play?(current_user)
-      ::Gitlab::UserAccess
-        .new(current_user, project: project)
-        .can_push_to_branch?(ref)
-    end
-
     def play(current_user)
       Ci::PlayBuildService
         .new(project, current_user)
diff --git a/app/models/environment.rb b/app/models/environment.rb
index f8b9a21c08ea3..bf33010fd21f2 100644
--- a/app/models/environment.rb
+++ b/app/models/environment.rb
@@ -122,12 +122,6 @@ def stop_action?
     available? && stop_action.present?
   end
 
-  def can_trigger_stop_action?(current_user)
-    return false unless stop_action?
-
-    stop_action.can_play?(current_user)
-  end
-
   def stop_with_action!(current_user)
     return unless available?
 
diff --git a/app/serializers/build_action_entity.rb b/app/serializers/build_action_entity.rb
index 3d12c64b88aa4..0bb7e5610731f 100644
--- a/app/serializers/build_action_entity.rb
+++ b/app/serializers/build_action_entity.rb
@@ -19,6 +19,6 @@ class BuildActionEntity < Grape::Entity
   alias_method :build, :object
 
   def playable?
-    build.playable? && build.can_play?(request.user)
+    can?(request.user, :play_build, build) && build.playable?
   end
 end
diff --git a/app/serializers/build_entity.rb b/app/serializers/build_entity.rb
index 401a277dadc53..f301900c43c43 100644
--- a/app/serializers/build_entity.rb
+++ b/app/serializers/build_entity.rb
@@ -26,7 +26,7 @@ class BuildEntity < Grape::Entity
   alias_method :build, :object
 
   def playable?
-    build.playable? && build.can_play?(request.user)
+    can?(request.user, :play_build, build) && build.playable?
   end
 
   def detailed_status
diff --git a/app/services/ci/stop_environments_service.rb b/app/services/ci/stop_environments_service.rb
index d1e341bc9b509..bd9735fc0accf 100644
--- a/app/services/ci/stop_environments_service.rb
+++ b/app/services/ci/stop_environments_service.rb
@@ -9,7 +9,8 @@ def execute(branch_name)
       return unless can?(current_user, :create_deployment, project)
 
       environments.each do |environment|
-        next unless environment.can_trigger_stop_action?(current_user)
+        next unless environment.stop_action?
+        next unless can?(current_user, :play_build, environment.stop_action)
 
         environment.stop_with_action!(current_user)
       end
diff --git a/app/views/projects/ci/builds/_build.html.haml b/app/views/projects/ci/builds/_build.html.haml
index ceec36e24407c..769640c484279 100644
--- a/app/views/projects/ci/builds/_build.html.haml
+++ b/app/views/projects/ci/builds/_build.html.haml
@@ -101,7 +101,7 @@
           = link_to cancel_namespace_project_build_path(build.project.namespace, build.project, build, return_to: request.original_url), method: :post, title: 'Cancel', class: 'btn btn-build' do
             = icon('remove', class: 'cred')
         - elsif allow_retry
-          - if build.playable? && !admin && build.can_play?(current_user)
+          - if build.playable? && !admin && can?(current_user, :play_build, build)
             = link_to play_namespace_project_build_path(build.project.namespace, build.project, build, return_to: request.original_url), method: :post, title: 'Play', class: 'btn btn-build' do
               = custom_icon('icon_play')
           - elsif build.retryable?
diff --git a/lib/gitlab/ci/status/build/play.rb b/lib/gitlab/ci/status/build/play.rb
index 29d0558a265c5..4c893f62925e3 100644
--- a/lib/gitlab/ci/status/build/play.rb
+++ b/lib/gitlab/ci/status/build/play.rb
@@ -10,7 +10,7 @@ def label
           end
 
           def has_action?
-            can?(user, :update_build, subject) && subject.can_play?(user)
+            can?(user, :play_build, subject)
           end
 
           def action_icon
diff --git a/spec/models/ci/build_spec.rb b/spec/models/ci/build_spec.rb
index 09e9f5bd8ba77..cdceca975e5a9 100644
--- a/spec/models/ci/build_spec.rb
+++ b/spec/models/ci/build_spec.rb
@@ -925,33 +925,6 @@ def create_mr(build, pipeline, factory: :merge_request, created_at: Time.now)
     end
   end
 
-  describe '#can_play?' do
-    before do
-      project.add_developer(user)
-    end
-
-    let(:build) do
-      create(:ci_build, ref: 'some-ref', pipeline: pipeline)
-    end
-
-    context 'when branch build is running for is protected' do
-      before do
-        create(:protected_branch, :no_one_can_push,
-               name: 'some-ref', project: project)
-      end
-
-      it 'indicates that user can not trigger an action' do
-        expect(build.can_play?(user)).to be_falsey
-      end
-    end
-
-    context 'when branch build is running for is not protected' do
-      it 'indicates that user can trigger an action' do
-        expect(build.can_play?(user)).to be_truthy
-      end
-    end
-  end
-
   describe '#play' do
     let(:build) { create(:ci_build, :manual, pipeline: pipeline) }
 
diff --git a/spec/models/environment_spec.rb b/spec/models/environment_spec.rb
index fb7cee47ae8e2..9e00f2247e8b1 100644
--- a/spec/models/environment_spec.rb
+++ b/spec/models/environment_spec.rb
@@ -155,31 +155,6 @@
     end
   end
 
-  describe '#can_trigger_stop_action?' do
-    let(:user) { create(:user) }
-    let(:project) { create(:project) }
-
-    let(:environment) do
-      create(:environment, :with_review_app, project: project)
-    end
-
-    context 'when user can trigger stop action' do
-      before do
-        project.add_developer(user)
-      end
-
-      it 'returns value that evaluates to true' do
-        expect(environment.can_trigger_stop_action?(user)).to be_truthy
-      end
-    end
-
-    context 'when user is not allowed to trigger stop action' do
-      it 'returns value that evaluates to false' do
-        expect(environment.can_trigger_stop_action?(user)).to be_falsey
-      end
-    end
-  end
-
   describe '#stop_with_action!' do
     let(:user) { create(:admin) }
 
-- 
GitLab


From 0b75541559c0550bd3e195ca5aca55016fa614ef Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 12 Apr 2017 15:22:32 +0200
Subject: [PATCH 035/363] Fix manual action entity specs

---
 spec/serializers/build_action_entity_spec.rb | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/spec/serializers/build_action_entity_spec.rb b/spec/serializers/build_action_entity_spec.rb
index 54ac17447b13d..059deba54163d 100644
--- a/spec/serializers/build_action_entity_spec.rb
+++ b/spec/serializers/build_action_entity_spec.rb
@@ -2,9 +2,10 @@
 
 describe BuildActionEntity do
   let(:build) { create(:ci_build, name: 'test_build') }
+  let(:request) { double('request') }
 
   let(:entity) do
-    described_class.new(build, request: double)
+    described_class.new(build, request: spy('request'))
   end
 
   describe '#as_json' do
-- 
GitLab


From 1ebcc96e3a14c5bee675df918cdecbe1a7d04b63 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 12 Apr 2017 22:41:46 -0500
Subject: [PATCH 036/363] Setup gettext libraries

---
 Gemfile                        |  4 ++++
 Gemfile.lock                   | 12 +++++++++++-
 locale/de/gitlab.edit.po       | 19 +++++++++++++++++++
 locale/de/gitlab.po            | 18 ++++++++++++++++++
 locale/de/gitlab.po.time_stamp |  0
 locale/en/gitlab.edit.po       | 19 +++++++++++++++++++
 locale/en/gitlab.po            | 18 ++++++++++++++++++
 locale/en/gitlab.po.time_stamp |  0
 locale/es/gitlab.edit.po       | 19 +++++++++++++++++++
 locale/es/gitlab.po            | 18 ++++++++++++++++++
 locale/es/gitlab.po.time_stamp |  0
 locale/gitlab.pot              | 19 +++++++++++++++++++
 12 files changed, 145 insertions(+), 1 deletion(-)
 create mode 100644 locale/de/gitlab.edit.po
 create mode 100644 locale/de/gitlab.po
 create mode 100644 locale/de/gitlab.po.time_stamp
 create mode 100644 locale/en/gitlab.edit.po
 create mode 100644 locale/en/gitlab.po
 create mode 100644 locale/en/gitlab.po.time_stamp
 create mode 100644 locale/es/gitlab.edit.po
 create mode 100644 locale/es/gitlab.po
 create mode 100644 locale/es/gitlab.po.time_stamp
 create mode 100644 locale/gitlab.pot

diff --git a/Gemfile b/Gemfile
index d4b2ade424382..e98c127d2f57b 100644
--- a/Gemfile
+++ b/Gemfile
@@ -251,6 +251,10 @@ gem 'sentry-raven', '~> 2.4.0'
 
 gem 'premailer-rails', '~> 1.9.0'
 
+# I18n
+gem 'gettext_i18n_rails', '~> 1.8.0'
+gem 'gettext', '~> 3.2.2', require: false, group: :development
+
 # Metrics
 group :metrics do
   gem 'allocations', '~> 1.0', require: false, platform: :mri
diff --git a/Gemfile.lock b/Gemfile.lock
index 965c888ca7927..eeec4b67764fb 100644
--- a/Gemfile.lock
+++ b/Gemfile.lock
@@ -200,6 +200,7 @@ GEM
     faraday_middleware-multi_json (0.0.6)
       faraday_middleware
       multi_json
+    fast_gettext (1.4.0)
     ffaker (2.4.0)
     ffi (1.9.10)
     flay (2.8.1)
@@ -253,6 +254,11 @@ GEM
     gemojione (3.0.1)
       json
     get_process_mem (0.2.0)
+    gettext (3.2.2)
+      locale (>= 2.0.5)
+      text (>= 1.3.0)
+    gettext_i18n_rails (1.8.0)
+      fast_gettext (>= 0.9.0)
     gherkin-ruby (0.3.2)
     gitaly (0.5.0)
       google-protobuf (~> 3.1)
@@ -422,6 +428,7 @@ GEM
     licensee (8.7.0)
       rugged (~> 0.24)
     little-plugger (1.1.4)
+    locale (2.1.2)
     logging (2.1.0)
       little-plugger (~> 1.1)
       multi_json (~> 1.10)
@@ -776,6 +783,7 @@ GEM
     temple (0.7.7)
     test_after_commit (1.1.0)
       activerecord (>= 3.2)
+    text (1.3.1)
     thin (1.7.0)
       daemons (~> 1.0, >= 1.0.9)
       eventmachine (~> 1.0, >= 1.0.4)
@@ -902,6 +910,8 @@ DEPENDENCIES
   fuubar (~> 2.0.0)
   gemnasium-gitlab-service (~> 0.2)
   gemojione (~> 3.0)
+  gettext (~> 3.2.2)
+  gettext_i18n_rails (~> 1.8.0)
   gitaly (~> 0.5.0)
   github-linguist (~> 4.7.0)
   gitlab-flowdock-git-hook (~> 1.0.1)
@@ -1035,4 +1045,4 @@ DEPENDENCIES
   wikicloth (= 0.8.1)
 
 BUNDLED WITH
-   1.14.5
+   1.14.6
diff --git a/locale/de/gitlab.edit.po b/locale/de/gitlab.edit.po
new file mode 100644
index 0000000000000..70ac9a35be79f
--- /dev/null
+++ b/locale/de/gitlab.edit.po
@@ -0,0 +1,19 @@
+# German translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2017-04-12 22:36-0500\n"
+"PO-Revision-Date: 2017-04-12 22:37-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: German\n"
+"Language: de\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"\n"
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
new file mode 100644
index 0000000000000..5eb4a86d528e1
--- /dev/null
+++ b/locale/de/gitlab.po
@@ -0,0 +1,18 @@
+# German translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"PO-Revision-Date: 2017-04-12 22:37-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: German\n"
+"Language: de\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"\n"
diff --git a/locale/de/gitlab.po.time_stamp b/locale/de/gitlab.po.time_stamp
new file mode 100644
index 0000000000000..e69de29bb2d1d
diff --git a/locale/en/gitlab.edit.po b/locale/en/gitlab.edit.po
new file mode 100644
index 0000000000000..f9e8c3e67094c
--- /dev/null
+++ b/locale/en/gitlab.edit.po
@@ -0,0 +1,19 @@
+# English translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2017-04-12 22:36-0500\n"
+"PO-Revision-Date: 2017-04-12 22:36-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: English\n"
+"Language: en\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"\n"
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
new file mode 100644
index 0000000000000..5dcd2983f30c2
--- /dev/null
+++ b/locale/en/gitlab.po
@@ -0,0 +1,18 @@
+# English translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"PO-Revision-Date: 2017-04-12 22:36-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: English\n"
+"Language: en\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"\n"
diff --git a/locale/en/gitlab.po.time_stamp b/locale/en/gitlab.po.time_stamp
new file mode 100644
index 0000000000000..e69de29bb2d1d
diff --git a/locale/es/gitlab.edit.po b/locale/es/gitlab.edit.po
new file mode 100644
index 0000000000000..38a2d10f6e845
--- /dev/null
+++ b/locale/es/gitlab.edit.po
@@ -0,0 +1,19 @@
+# Spanish translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2017-04-12 22:36-0500\n"
+"PO-Revision-Date: 2017-04-12 22:37-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: Spanish\n"
+"Language: es\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"\n"
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
new file mode 100644
index 0000000000000..893eb7e15a3e3
--- /dev/null
+++ b/locale/es/gitlab.po
@@ -0,0 +1,18 @@
+# Spanish translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"PO-Revision-Date: 2017-04-12 22:37-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: Spanish\n"
+"Language: es\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"\n"
diff --git a/locale/es/gitlab.po.time_stamp b/locale/es/gitlab.po.time_stamp
new file mode 100644
index 0000000000000..e69de29bb2d1d
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
new file mode 100644
index 0000000000000..7c4f303a5d035
--- /dev/null
+++ b/locale/gitlab.pot
@@ -0,0 +1,19 @@
+# SOME DESCRIPTIVE TITLE.
+# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
+#
+#, fuzzy
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2017-04-12 22:36-0500\n"
+"PO-Revision-Date: 2017-04-12 22:36-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: LANGUAGE <LL@li.org>\n"
+"Language: \n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
-- 
GitLab


From acc807cd30a4b0005b552d67583b20538474a1f2 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 12 Apr 2017 23:04:26 -0500
Subject: [PATCH 037/363] Add preferred_language column to users

---
 ...170413035209_add_preferred_language_to_users.rb | 14 ++++++++++++++
 db/schema.rb                                       |  3 ++-
 2 files changed, 16 insertions(+), 1 deletion(-)
 create mode 100644 db/migrate/20170413035209_add_preferred_language_to_users.rb

diff --git a/db/migrate/20170413035209_add_preferred_language_to_users.rb b/db/migrate/20170413035209_add_preferred_language_to_users.rb
new file mode 100644
index 0000000000000..5dc128dbaea0a
--- /dev/null
+++ b/db/migrate/20170413035209_add_preferred_language_to_users.rb
@@ -0,0 +1,14 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddPreferredLanguageToUsers < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  disable_ddl_transaction!
+
+  def change
+    add_column_with_default :users, :preferred_language, :string, default: 'en'
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 5689f7331dcfd..03ce12570885a 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170408033905) do
+ActiveRecord::Schema.define(version: 20170413035209) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -1304,6 +1304,7 @@
     t.boolean "notified_of_own_activity"
     t.boolean "require_two_factor_authentication_from_group", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
+    t.string "preferred_language"
   end
 
   add_index "users", ["admin"], name: "index_users_on_admin", using: :btree
-- 
GitLab


From bd86796dd0edae7e5db2bfbb887d3196498ebd49 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 12 Apr 2017 23:23:02 -0500
Subject: [PATCH 038/363] Add support to change language in profile form

---
 app/controllers/profiles_controller.rb | 3 ++-
 app/views/profiles/show.html.haml      | 3 +++
 lib/gitlab/i18n.rb                     | 9 +++++++++
 3 files changed, 14 insertions(+), 1 deletion(-)
 create mode 100644 lib/gitlab/i18n.rb

diff --git a/app/controllers/profiles_controller.rb b/app/controllers/profiles_controller.rb
index 987b95e89b9e6..57e23cea00eb5 100644
--- a/app/controllers/profiles_controller.rb
+++ b/app/controllers/profiles_controller.rb
@@ -85,7 +85,8 @@ def user_params
       :twitter,
       :username,
       :website_url,
-      :organization
+      :organization,
+      :preferred_language
     )
   end
 end
diff --git a/app/views/profiles/show.html.haml b/app/views/profiles/show.html.haml
index c74b3249a1305..dc71a04cbf048 100644
--- a/app/views/profiles/show.html.haml
+++ b/app/views/profiles/show.html.haml
@@ -72,6 +72,9 @@
         = f.label :public_email, class: "label-light"
         = f.select :public_email, options_for_select(@user.all_emails, selected: @user.public_email), { include_blank: 'Do not show on profile' }, class: "select2"
         %span.help-block This email will be displayed on your public profile.
+      .form-group
+        = f.label :preferred_language, class: "label-light"
+        = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES, {}, class: "select2"
       .form-group
         = f.label :skype, class: "label-light"
         = f.text_field :skype, class: "form-control"
diff --git a/lib/gitlab/i18n.rb b/lib/gitlab/i18n.rb
new file mode 100644
index 0000000000000..8c578aa0b4650
--- /dev/null
+++ b/lib/gitlab/i18n.rb
@@ -0,0 +1,9 @@
+module Gitlab
+  module I18n
+    AVAILABLE_LANGUAGES = [
+      [_('English'), 'en'],
+      [_('Spanish'), 'es'],
+      [_('Deutsch'), 'de']
+    ]
+  end
+end
-- 
GitLab


From 020e12a6ae99e9b9dfaa3cfbd640ecdf31d7424d Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 12 Apr 2017 23:24:29 -0500
Subject: [PATCH 039/363] Add missing initializer for FastGettext

---
 config/initializers/fast_gettext.rb | 3 +++
 1 file changed, 3 insertions(+)
 create mode 100644 config/initializers/fast_gettext.rb

diff --git a/config/initializers/fast_gettext.rb b/config/initializers/fast_gettext.rb
new file mode 100644
index 0000000000000..cfc3427b5ae69
--- /dev/null
+++ b/config/initializers/fast_gettext.rb
@@ -0,0 +1,3 @@
+FastGettext.add_text_domain 'gitlab', path: 'locale', type: :po
+FastGettext.default_available_locales = ['en', 'es','de']
+FastGettext.default_text_domain = 'gitlab'
-- 
GitLab


From 73d0730d09b5f9a9b68f158cc72ad30c7a2b35d0 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 13 Apr 2017 01:03:47 -0500
Subject: [PATCH 040/363] Set locale through controller filter

---
 app/controllers/application_controller.rb |  6 ++++++
 app/controllers/profiles_controller.rb    |  1 +
 app/views/profiles/show.html.haml         |  3 ++-
 lib/gitlab/i18n.rb                        |  6 +++---
 locale/de/gitlab.edit.po                  | 11 ++++++++++-
 locale/de/gitlab.po                       |  9 +++++++++
 locale/en/gitlab.edit.po                  | 11 ++++++++++-
 locale/en/gitlab.po                       |  9 +++++++++
 locale/es/gitlab.edit.po                  | 17 +++++++++++++----
 locale/es/gitlab.po                       | 15 ++++++++++++---
 locale/gitlab.pot                         | 13 +++++++++++--
 11 files changed, 86 insertions(+), 15 deletions(-)

diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index e77094fe2a826..5a3bd4040cc03 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -265,4 +265,10 @@ def gitlab_project_import_enabled?
   def u2f_app_id
     request.base_url
   end
+
+  def set_locale
+    requested_locale = current_user&.preferred_language || request.env['HTTP_ACCEPT_LANGUAGE'] || I18n.default_locale
+    locale = FastGettext.set_locale(requested_locale)
+    I18n.locale = locale
+  end
 end
diff --git a/app/controllers/profiles_controller.rb b/app/controllers/profiles_controller.rb
index 57e23cea00eb5..0f01bf7e7067a 100644
--- a/app/controllers/profiles_controller.rb
+++ b/app/controllers/profiles_controller.rb
@@ -3,6 +3,7 @@ class ProfilesController < Profiles::ApplicationController
 
   before_action :user
   before_action :authorize_change_username!, only: :update_username
+  before_action :set_locale, only: :show
   skip_before_action :require_email, only: [:show, :update]
 
   def show
diff --git a/app/views/profiles/show.html.haml b/app/views/profiles/show.html.haml
index dc71a04cbf048..d8ef64ceb7288 100644
--- a/app/views/profiles/show.html.haml
+++ b/app/views/profiles/show.html.haml
@@ -74,7 +74,8 @@
         %span.help-block This email will be displayed on your public profile.
       .form-group
         = f.label :preferred_language, class: "label-light"
-        = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES, {}, class: "select2"
+        = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES.map { |lang| [_(lang[0]), lang[1]] },
+          {}, class: "select2"
       .form-group
         = f.label :skype, class: "label-light"
         = f.text_field :skype, class: "form-control"
diff --git a/lib/gitlab/i18n.rb b/lib/gitlab/i18n.rb
index 8c578aa0b4650..85e527afd7148 100644
--- a/lib/gitlab/i18n.rb
+++ b/lib/gitlab/i18n.rb
@@ -1,9 +1,9 @@
 module Gitlab
   module I18n
     AVAILABLE_LANGUAGES = [
-      [_('English'), 'en'],
-      [_('Spanish'), 'es'],
-      [_('Deutsch'), 'de']
+      ['English', 'en'],
+      ['Spanish', 'es'],
+      ['Deutsch', 'de']
     ]
   end
 end
diff --git a/locale/de/gitlab.edit.po b/locale/de/gitlab.edit.po
index 70ac9a35be79f..e4c8caef1647e 100644
--- a/locale/de/gitlab.edit.po
+++ b/locale/de/gitlab.edit.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-12 22:36-0500\n"
+"POT-Creation-Date: 2017-04-13 00:01-0500\n"
 "PO-Revision-Date: 2017-04-12 22:37-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: German\n"
@@ -17,3 +17,12 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
+
+msgid "Deutsch"
+msgstr ""
+
+msgid "English"
+msgstr ""
+
+msgid "Spanish"
+msgstr ""
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index 5eb4a86d528e1..7f32771b80f74 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -16,3 +16,12 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
+
+msgid "Deutsch"
+msgstr ""
+
+msgid "English"
+msgstr ""
+
+msgid "Spanish"
+msgstr ""
diff --git a/locale/en/gitlab.edit.po b/locale/en/gitlab.edit.po
index f9e8c3e67094c..048f531de4b0d 100644
--- a/locale/en/gitlab.edit.po
+++ b/locale/en/gitlab.edit.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-12 22:36-0500\n"
+"POT-Creation-Date: 2017-04-13 00:01-0500\n"
 "PO-Revision-Date: 2017-04-12 22:36-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: English\n"
@@ -17,3 +17,12 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
+
+msgid "Deutsch"
+msgstr ""
+
+msgid "English"
+msgstr ""
+
+msgid "Spanish"
+msgstr ""
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 5dcd2983f30c2..f93f438b42416 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -16,3 +16,12 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
+
+msgid "Deutsch"
+msgstr ""
+
+msgid "English"
+msgstr ""
+
+msgid "Spanish"
+msgstr ""
diff --git a/locale/es/gitlab.edit.po b/locale/es/gitlab.edit.po
index 38a2d10f6e845..b5b6c0a671f0d 100644
--- a/locale/es/gitlab.edit.po
+++ b/locale/es/gitlab.edit.po
@@ -7,13 +7,22 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-12 22:36-0500\n"
-"PO-Revision-Date: 2017-04-12 22:37-0500\n"
-"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"PO-Revision-Date: 2017-04-13 00:07-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
-"\n"
+"POT-Creation-Date: 2017-04-13 00:01-0500\n"
+"Last-Translator: \n"
+"X-Generator: Poedit 2.0.1\n"
+
+msgid "Deutsch"
+msgstr "Alemán"
+
+msgid "English"
+msgstr "Inglés"
+
+msgid "Spanish"
+msgstr "Español"
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 893eb7e15a3e3..e2d03d8834774 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,12 +7,21 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-04-12 22:37-0500\n"
-"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"PO-Revision-Date: 2017-04-13 00:07-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
-"\n"
+"Last-Translator: \n"
+"X-Generator: Poedit 2.0.1\n"
+
+msgid "Deutsch"
+msgstr "Alemán"
+
+msgid "English"
+msgstr "Inglés"
+
+msgid "Spanish"
+msgstr "Español"
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index 7c4f303a5d035..a90fc69d94bf0 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-12 22:36-0500\n"
-"PO-Revision-Date: 2017-04-12 22:36-0500\n"
+"POT-Creation-Date: 2017-04-13 00:01-0500\n"
+"PO-Revision-Date: 2017-04-13 00:01-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -17,3 +17,12 @@ msgstr ""
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
+
+msgid "Deutsch"
+msgstr ""
+
+msgid "English"
+msgstr ""
+
+msgid "Spanish"
+msgstr ""
-- 
GitLab


From 72fefba4c18e2464725231b3426ec2e3b0eda9c4 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 13 Apr 2017 01:08:44 -0500
Subject: [PATCH 041/363] Ignore some files required to the translation process

---
 .gitignore               |  3 +++
 locale/de/gitlab.edit.po | 28 ----------------------------
 locale/en/gitlab.edit.po | 28 ----------------------------
 locale/es/gitlab.edit.po | 28 ----------------------------
 4 files changed, 3 insertions(+), 84 deletions(-)
 delete mode 100644 locale/de/gitlab.edit.po
 delete mode 100644 locale/en/gitlab.edit.po
 delete mode 100644 locale/es/gitlab.edit.po

diff --git a/.gitignore b/.gitignore
index 51b4d06b01b2a..40948f2e89c72 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,5 +1,7 @@
 *.log
 *.swp
+*.mo
+*.edit.po
 .DS_Store
 .bundle
 .chef
@@ -53,3 +55,4 @@ eslint-report.html
 /shared/*
 /.gitlab_workhorse_secret
 /webpack-report/
+/locale/**/LC_MESSAGES
diff --git a/locale/de/gitlab.edit.po b/locale/de/gitlab.edit.po
deleted file mode 100644
index e4c8caef1647e..0000000000000
--- a/locale/de/gitlab.edit.po
+++ /dev/null
@@ -1,28 +0,0 @@
-# German translations for gitlab package.
-# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
-# This file is distributed under the same license as the gitlab package.
-# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
-#
-msgid ""
-msgstr ""
-"Project-Id-Version: gitlab 1.0.0\n"
-"Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-13 00:01-0500\n"
-"PO-Revision-Date: 2017-04-12 22:37-0500\n"
-"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
-"Language-Team: German\n"
-"Language: de\n"
-"MIME-Version: 1.0\n"
-"Content-Type: text/plain; charset=UTF-8\n"
-"Content-Transfer-Encoding: 8bit\n"
-"Plural-Forms: nplurals=2; plural=n != 1;\n"
-"\n"
-
-msgid "Deutsch"
-msgstr ""
-
-msgid "English"
-msgstr ""
-
-msgid "Spanish"
-msgstr ""
diff --git a/locale/en/gitlab.edit.po b/locale/en/gitlab.edit.po
deleted file mode 100644
index 048f531de4b0d..0000000000000
--- a/locale/en/gitlab.edit.po
+++ /dev/null
@@ -1,28 +0,0 @@
-# English translations for gitlab package.
-# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
-# This file is distributed under the same license as the gitlab package.
-# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
-#
-msgid ""
-msgstr ""
-"Project-Id-Version: gitlab 1.0.0\n"
-"Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-13 00:01-0500\n"
-"PO-Revision-Date: 2017-04-12 22:36-0500\n"
-"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
-"Language-Team: English\n"
-"Language: en\n"
-"MIME-Version: 1.0\n"
-"Content-Type: text/plain; charset=UTF-8\n"
-"Content-Transfer-Encoding: 8bit\n"
-"Plural-Forms: nplurals=2; plural=n != 1;\n"
-"\n"
-
-msgid "Deutsch"
-msgstr ""
-
-msgid "English"
-msgstr ""
-
-msgid "Spanish"
-msgstr ""
diff --git a/locale/es/gitlab.edit.po b/locale/es/gitlab.edit.po
deleted file mode 100644
index b5b6c0a671f0d..0000000000000
--- a/locale/es/gitlab.edit.po
+++ /dev/null
@@ -1,28 +0,0 @@
-# Spanish translations for gitlab package.
-# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
-# This file is distributed under the same license as the gitlab package.
-# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
-#
-msgid ""
-msgstr ""
-"Project-Id-Version: gitlab 1.0.0\n"
-"Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-04-13 00:07-0500\n"
-"Language-Team: Spanish\n"
-"Language: es\n"
-"MIME-Version: 1.0\n"
-"Content-Type: text/plain; charset=UTF-8\n"
-"Content-Transfer-Encoding: 8bit\n"
-"Plural-Forms: nplurals=2; plural=n != 1;\n"
-"POT-Creation-Date: 2017-04-13 00:01-0500\n"
-"Last-Translator: \n"
-"X-Generator: Poedit 2.0.1\n"
-
-msgid "Deutsch"
-msgstr "Alemán"
-
-msgid "English"
-msgstr "Inglés"
-
-msgid "Spanish"
-msgstr "Español"
-- 
GitLab


From 57f8f2d7ff851fc4f5d1c81a28a023855f1985b7 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 13 Apr 2017 15:06:45 +0200
Subject: [PATCH 042/363] Fix detailed build status specs for manual actions

---
 spec/lib/gitlab/ci/status/build/play_spec.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/lib/gitlab/ci/status/build/play_spec.rb b/spec/lib/gitlab/ci/status/build/play_spec.rb
index 48aeb89e11f44..95f1388a9b971 100644
--- a/spec/lib/gitlab/ci/status/build/play_spec.rb
+++ b/spec/lib/gitlab/ci/status/build/play_spec.rb
@@ -12,7 +12,7 @@
 
   describe 'action details' do
     let(:user) { create(:user) }
-    let(:build) { create(:ci_build) }
+    let(:build) { create(:ci_build, :manual) }
     let(:status) { Gitlab::Ci::Status::Core.new(build, user) }
 
     describe '#has_action?' do
-- 
GitLab


From cfd3d0fd377c3438c6ce8bc2f20b11f86b43a785 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 4 Apr 2017 14:58:45 +0100
Subject: [PATCH 043/363] [ci skip] Remove loadscript class in favour of
 backend conditional

---
 app/assets/javascripts/application.js         |  266 --
 .../javascripts/lib/utils/load_script.js.es6  |   26 -
 app/assets/javascripts/raven/index.js         |   10 +
 app/assets/javascripts/raven/raven_config.js  |   46 +
 app/assets/javascripts/raven_config.js.es6    |   66 -
 app/helpers/sentry_helper.rb                  |    2 +
 app/views/layouts/application.html.haml       |    2 +
 app/views/layouts/devise.html.haml            |    2 +
 app/views/layouts/devise_empty.html.haml      |    2 +
 config/webpack.config.js                      |    1 +
 lib/gitlab/gon_helper.rb                      |    1 -
 package.json                                  |    1 +
 spec/features/raven_js_spec.rb                |    8 +-
 spec/javascripts/class_spec_helper.js.es6     |   10 -
 .../javascripts/class_spec_helper_spec.js.es6 |   35 -
 .../lib/utils/load_script_spec.js.es6         |   95 -
 spec/javascripts/raven/index_spec.js          |   11 +
 spec/javascripts/raven/raven_config_spec.js   |  137 +
 spec/javascripts/raven_config_spec.js.es6     |  142 -
 vendor/assets/javascripts/raven.js            | 2547 -----------------
 yarn.lock                                     |    8 +-
 21 files changed, 225 insertions(+), 3193 deletions(-)
 delete mode 100644 app/assets/javascripts/application.js
 delete mode 100644 app/assets/javascripts/lib/utils/load_script.js.es6
 create mode 100644 app/assets/javascripts/raven/index.js
 create mode 100644 app/assets/javascripts/raven/raven_config.js
 delete mode 100644 app/assets/javascripts/raven_config.js.es6
 delete mode 100644 spec/javascripts/class_spec_helper.js.es6
 delete mode 100644 spec/javascripts/class_spec_helper_spec.js.es6
 delete mode 100644 spec/javascripts/lib/utils/load_script_spec.js.es6
 create mode 100644 spec/javascripts/raven/index_spec.js
 create mode 100644 spec/javascripts/raven/raven_config_spec.js
 delete mode 100644 spec/javascripts/raven_config_spec.js.es6
 delete mode 100644 vendor/assets/javascripts/raven.js

diff --git a/app/assets/javascripts/application.js b/app/assets/javascripts/application.js
deleted file mode 100644
index 94902e560a8b4..0000000000000
--- a/app/assets/javascripts/application.js
+++ /dev/null
@@ -1,266 +0,0 @@
-/* eslint-disable func-names, space-before-function-paren, no-var, quotes, consistent-return, prefer-arrow-callback, comma-dangle, object-shorthand, no-new, max-len */
-/* global bp */
-/* global Cookies */
-/* global Flash */
-/* global ConfirmDangerModal */
-/* global AwardsHandler */
-/* global Aside */
-
-// This is a manifest file that'll be compiled into including all the files listed below.
-// Add new JavaScript code in separate files in this directory and they'll automatically
-// be included in the compiled file accessible from http://example.com/assets/application.js
-// It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the
-// the compiled file.
-//
-/*= require jquery2 */
-/*= require jquery-ui/autocomplete */
-/*= require jquery-ui/datepicker */
-/*= require jquery-ui/draggable */
-/*= require jquery-ui/effect-highlight */
-/*= require jquery-ui/sortable */
-/*= require jquery_ujs */
-/*= require jquery.endless-scroll */
-/*= require jquery.highlight */
-/*= require jquery.waitforimages */
-/*= require jquery.atwho */
-/*= require jquery.scrollTo */
-/*= require jquery.turbolinks */
-/*= require js.cookie */
-/*= require turbolinks */
-/*= require autosave */
-/*= require bootstrap/affix */
-/*= require bootstrap/alert */
-/*= require bootstrap/button */
-/*= require bootstrap/collapse */
-/*= require bootstrap/dropdown */
-/*= require bootstrap/modal */
-/*= require bootstrap/scrollspy */
-/*= require bootstrap/tab */
-/*= require bootstrap/transition */
-/*= require bootstrap/tooltip */
-/*= require bootstrap/popover */
-/*= require select2 */
-/*= require underscore */
-/*= require dropzone */
-/*= require mousetrap */
-/*= require mousetrap/pause */
-/*= require shortcuts */
-/*= require shortcuts_navigation */
-/*= require shortcuts_dashboard_navigation */
-/*= require shortcuts_issuable */
-/*= require shortcuts_network */
-/*= require jquery.nicescroll */
-/*= require date.format */
-/*= require_directory ./behaviors */
-/*= require_directory ./blob */
-/*= require_directory ./templates */
-/*= require_directory ./commit */
-/*= require_directory ./extensions */
-/*= require_directory ./lib/utils */
-/*= require_directory ./u2f */
-/*= require_directory ./droplab */
-/*= require_directory . */
-/*= require fuzzaldrin-plus */
-/*= require es6-promise.auto */
-/*= require raven_config */
-
-(function () {
-  document.addEventListener('page:fetch', function () {
-    // Unbind scroll events
-    $(document).off('scroll');
-    // Close any open tooltips
-    $('.has-tooltip, [data-toggle="tooltip"]').tooltip('destroy');
-  });
-
-  window.addEventListener('hashchange', gl.utils.handleLocationHash);
-  window.addEventListener('load', function onLoad() {
-    window.removeEventListener('load', onLoad, false);
-    gl.utils.handleLocationHash();
-  }, false);
-
-  $(function () {
-    var $body = $('body');
-    var $document = $(document);
-    var $window = $(window);
-    var $sidebarGutterToggle = $('.js-sidebar-toggle');
-    var $flash = $('.flash-container');
-    var bootstrapBreakpoint = bp.getBreakpointSize();
-    var checkInitialSidebarSize;
-    var fitSidebarForSize;
-
-    // Set the default path for all cookies to GitLab's root directory
-    Cookies.defaults.path = gon.relative_url_root || '/';
-
-    // `hashchange` is not triggered when link target is already in window.location
-    $body.on('click', 'a[href^="#"]', function() {
-      var href = this.getAttribute('href');
-      if (href.substr(1) === gl.utils.getLocationHash()) {
-        setTimeout(gl.utils.handleLocationHash, 1);
-      }
-    });
-
-    // prevent default action for disabled buttons
-    $('.btn').click(function(e) {
-      if ($(this).hasClass('disabled')) {
-        e.preventDefault();
-        e.stopImmediatePropagation();
-        return false;
-      }
-    });
-
-    $('.nav-sidebar').niceScroll({
-      cursoropacitymax: '0.4',
-      cursorcolor: '#FFF',
-      cursorborder: '1px solid #FFF'
-    });
-    $('.js-select-on-focus').on('focusin', function () {
-      return $(this).select().one('mouseup', function (e) {
-        return e.preventDefault();
-      });
-    // Click a .js-select-on-focus field, select the contents
-    // Prevent a mouseup event from deselecting the input
-    });
-    $('.remove-row').bind('ajax:success', function () {
-      $(this).tooltip('destroy')
-        .closest('li')
-        .fadeOut();
-    });
-    $('.js-remove-tr').bind('ajax:before', function () {
-      return $(this).hide();
-    });
-    $('.js-remove-tr').bind('ajax:success', function () {
-      return $(this).closest('tr').fadeOut();
-    });
-    $('select.select2').select2({
-      width: 'resolve',
-      // Initialize select2 selects
-      dropdownAutoWidth: true
-    });
-    $('.js-select2').bind('select2-close', function () {
-      return setTimeout((function () {
-        $('.select2-container-active').removeClass('select2-container-active');
-        return $(':focus').blur();
-      }), 1);
-    // Close select2 on escape
-    });
-    // Initialize tooltips
-    $.fn.tooltip.Constructor.DEFAULTS.trigger = 'hover';
-    $body.tooltip({
-      selector: '.has-tooltip, [data-toggle="tooltip"]',
-      placement: function (_, el) {
-        return $(el).data('placement') || 'bottom';
-      }
-    });
-    $('.trigger-submit').on('change', function () {
-      return $(this).parents('form').submit();
-    // Form submitter
-    });
-    gl.utils.localTimeAgo($('abbr.timeago, .js-timeago'), true);
-    // Flash
-    if ($flash.length > 0) {
-      $flash.click(function () {
-        return $(this).fadeOut();
-      });
-      $flash.show();
-    }
-    // Disable form buttons while a form is submitting
-    $body.on('ajax:complete, ajax:beforeSend, submit', 'form', function (e) {
-      var buttons;
-      buttons = $('[type="submit"]', this);
-      switch (e.type) {
-        case 'ajax:beforeSend':
-        case 'submit':
-          return buttons.disable();
-        default:
-          return buttons.enable();
-      }
-    });
-    $(document).ajaxError(function (e, xhrObj) {
-      var ref = xhrObj.status;
-      if (xhrObj.status === 401) {
-        return new Flash('You need to be logged in.', 'alert');
-      } else if (ref === 404 || ref === 500) {
-        return new Flash('Something went wrong on our end.', 'alert');
-      }
-    });
-    $('.account-box').hover(function () {
-      // Show/Hide the profile menu when hovering the account box
-      return $(this).toggleClass('hover');
-    });
-    $document.on('click', '.diff-content .js-show-suppressed-diff', function () {
-      var $container;
-      $container = $(this).parent();
-      $container.next('table').show();
-      return $container.remove();
-    // Commit show suppressed diff
-    });
-    $('.navbar-toggle').on('click', function () {
-      $('.header-content .title').toggle();
-      $('.header-content .header-logo').toggle();
-      $('.header-content .navbar-collapse').toggle();
-      return $('.navbar-toggle').toggleClass('active');
-    });
-    // Show/hide comments on diff
-    $body.on('click', '.js-toggle-diff-comments', function (e) {
-      var $this = $(this);
-      var notesHolders = $this.closest('.diff-file').find('.notes_holder');
-      $this.toggleClass('active');
-      if ($this.hasClass('active')) {
-        notesHolders.show().find('.hide').show();
-      } else {
-        notesHolders.hide();
-      }
-      $this.trigger('blur');
-      return e.preventDefault();
-    });
-    $document.off('click', '.js-confirm-danger');
-    $document.on('click', '.js-confirm-danger', function (e) {
-      var btn = $(e.target);
-      var form = btn.closest('form');
-      var text = btn.data('confirm-danger-message');
-      e.preventDefault();
-      return new ConfirmDangerModal(form, text);
-    });
-    $('input[type="search"]').each(function () {
-      var $this = $(this);
-      $this.attr('value', $this.val());
-    });
-    $document.off('keyup', 'input[type="search"]').on('keyup', 'input[type="search"]', function () {
-      var $this;
-      $this = $(this);
-      return $this.attr('value', $this.val());
-    });
-    $document.off('breakpoint:change').on('breakpoint:change', function (e, breakpoint) {
-      var $gutterIcon;
-      if (breakpoint === 'sm' || breakpoint === 'xs') {
-        $gutterIcon = $sidebarGutterToggle.find('i');
-        if ($gutterIcon.hasClass('fa-angle-double-right')) {
-          return $sidebarGutterToggle.trigger('click');
-        }
-      }
-    });
-    fitSidebarForSize = function () {
-      var oldBootstrapBreakpoint;
-      oldBootstrapBreakpoint = bootstrapBreakpoint;
-      bootstrapBreakpoint = bp.getBreakpointSize();
-      if (bootstrapBreakpoint !== oldBootstrapBreakpoint) {
-        return $document.trigger('breakpoint:change', [bootstrapBreakpoint]);
-      }
-    };
-    checkInitialSidebarSize = function () {
-      bootstrapBreakpoint = bp.getBreakpointSize();
-      if (bootstrapBreakpoint === 'xs' || 'sm') {
-        return $document.trigger('breakpoint:change', [bootstrapBreakpoint]);
-      }
-    };
-    $window.off('resize.app').on('resize.app', function () {
-      return fitSidebarForSize();
-    });
-    gl.awardsHandler = new AwardsHandler();
-    checkInitialSidebarSize();
-    new Aside();
-
-    // bind sidebar events
-    new gl.Sidebar();
-  });
-}).call(this);
diff --git a/app/assets/javascripts/lib/utils/load_script.js.es6 b/app/assets/javascripts/lib/utils/load_script.js.es6
deleted file mode 100644
index 351d96530eddf..0000000000000
--- a/app/assets/javascripts/lib/utils/load_script.js.es6
+++ /dev/null
@@ -1,26 +0,0 @@
-(() => {
-  const global = window.gl || (window.gl = {});
-
-  class LoadScript {
-    static load(source, id = '') {
-      if (!source) return Promise.reject('source url must be defined');
-      if (id && document.querySelector(`#${id}`)) return Promise.reject('script id already exists');
-      return new Promise((resolve, reject) => this.appendScript(source, id, resolve, reject));
-    }
-
-    static appendScript(source, id, resolve, reject) {
-      const scriptElement = document.createElement('script');
-      scriptElement.type = 'text/javascript';
-      if (id) scriptElement.id = id;
-      scriptElement.onload = resolve;
-      scriptElement.onerror = reject;
-      scriptElement.src = source;
-
-      document.body.appendChild(scriptElement);
-    }
-  }
-
-  global.LoadScript = LoadScript;
-
-  return global.LoadScript;
-})();
diff --git a/app/assets/javascripts/raven/index.js b/app/assets/javascripts/raven/index.js
new file mode 100644
index 0000000000000..6cc81248e6b23
--- /dev/null
+++ b/app/assets/javascripts/raven/index.js
@@ -0,0 +1,10 @@
+import RavenConfig from './raven_config';
+
+RavenConfig.init({
+  sentryDsn: gon.sentry_dsn,
+  currentUserId: gon.current_user_id,
+  whitelistUrls: [gon.gitlab_url],
+  isProduction: gon.is_production,
+});
+
+export default RavenConfig;
diff --git a/app/assets/javascripts/raven/raven_config.js b/app/assets/javascripts/raven/raven_config.js
new file mode 100644
index 0000000000000..5510dd2752d58
--- /dev/null
+++ b/app/assets/javascripts/raven/raven_config.js
@@ -0,0 +1,46 @@
+import Raven from 'raven-js';
+
+class RavenConfig {
+  static init(options = {}) {
+    this.options = options;
+
+    this.configure();
+    this.bindRavenErrors();
+    if (this.options.currentUserId) this.setUser();
+  }
+
+  static configure() {
+    Raven.config(this.options.sentryDsn, {
+      whitelistUrls: this.options.whitelistUrls,
+      environment: this.options.isProduction ? 'production' : 'development',
+    }).install();
+  }
+
+  static setUser() {
+    Raven.setUserContext({
+      id: this.options.currentUserId,
+    });
+  }
+
+  static bindRavenErrors() {
+    $(document).on('ajaxError.raven', this.handleRavenErrors);
+  }
+
+  static handleRavenErrors(event, req, config, err) {
+    const error = err || req.statusText;
+
+    Raven.captureMessage(error, {
+      extra: {
+        type: config.type,
+        url: config.url,
+        data: config.data,
+        status: req.status,
+        response: req.responseText.substring(0, 100),
+        error,
+        event,
+      },
+    });
+  }
+}
+
+export default RavenConfig;
diff --git a/app/assets/javascripts/raven_config.js.es6 b/app/assets/javascripts/raven_config.js.es6
deleted file mode 100644
index e15eeb9f9cd75..0000000000000
--- a/app/assets/javascripts/raven_config.js.es6
+++ /dev/null
@@ -1,66 +0,0 @@
-/* global Raven */
-
-/*= require lib/utils/load_script */
-
-(() => {
-  const global = window.gl || (window.gl = {});
-
-  class RavenConfig {
-    static init(options = {}) {
-      this.options = options;
-      if (!this.options.sentryDsn || !this.options.ravenAssetUrl) return Promise.reject('sentry dsn and raven asset url is required');
-      return global.LoadScript.load(this.options.ravenAssetUrl, 'raven-js')
-        .then(() => {
-          this.configure();
-          this.bindRavenErrors();
-          if (this.options.currentUserId) this.setUser();
-        });
-    }
-
-    static configure() {
-      Raven.config(this.options.sentryDsn, {
-        whitelistUrls: this.options.whitelistUrls,
-        environment: this.options.isProduction ? 'production' : 'development',
-      }).install();
-    }
-
-    static setUser() {
-      Raven.setUserContext({
-        id: this.options.currentUserId,
-      });
-    }
-
-    static bindRavenErrors() {
-      $(document).on('ajaxError.raven', this.handleRavenErrors);
-    }
-
-    static handleRavenErrors(event, req, config, err) {
-      const error = err || req.statusText;
-      Raven.captureMessage(error, {
-        extra: {
-          type: config.type,
-          url: config.url,
-          data: config.data,
-          status: req.status,
-          response: req.responseText.substring(0, 100),
-          error,
-          event,
-        },
-      });
-    }
-  }
-
-  global.RavenConfig = RavenConfig;
-
-  document.addEventListener('DOMContentLoaded', () => {
-    if (!window.gon) return;
-
-    global.RavenConfig.init({
-      sentryDsn: gon.sentry_dsn,
-      ravenAssetUrl: gon.raven_asset_url,
-      currentUserId: gon.current_user_id,
-      whitelistUrls: [gon.gitlab_url],
-      isProduction: gon.is_production,
-    }).catch($.noop);
-  });
-})();
diff --git a/app/helpers/sentry_helper.rb b/app/helpers/sentry_helper.rb
index 19de38ac52db6..4b07f71bcea02 100644
--- a/app/helpers/sentry_helper.rb
+++ b/app/helpers/sentry_helper.rb
@@ -9,7 +9,9 @@ def sentry_context
 
   def sentry_dsn_public
     sentry_dsn = ApplicationSetting.current.sentry_dsn
+
     return unless sentry_dsn
+
     uri = URI.parse(sentry_dsn)
     uri.password = nil
     uri.to_s
diff --git a/app/views/layouts/application.html.haml b/app/views/layouts/application.html.haml
index 36543edc040f9..cfd9481e4b23a 100644
--- a/app/views/layouts/application.html.haml
+++ b/app/views/layouts/application.html.haml
@@ -9,3 +9,5 @@
 
     = yield :scripts_body
     = render "layouts/init_auto_complete" if @gfm_form
+
+    = javascript_include_tag(*webpack_asset_paths("raven")) if sentry_enabled?
diff --git a/app/views/layouts/devise.html.haml b/app/views/layouts/devise.html.haml
index 3368a9beb2964..6274f6340ab86 100644
--- a/app/views/layouts/devise.html.haml
+++ b/app/views/layouts/devise.html.haml
@@ -35,3 +35,5 @@
           = link_to "Explore", explore_root_path
           = link_to "Help", help_path
           = link_to "About GitLab", "https://about.gitlab.com/"
+
+      = javascript_include_tag(*webpack_asset_paths("raven")) if sentry_enabled?
diff --git a/app/views/layouts/devise_empty.html.haml b/app/views/layouts/devise_empty.html.haml
index 7466423a934ec..120f7299fc96c 100644
--- a/app/views/layouts/devise_empty.html.haml
+++ b/app/views/layouts/devise_empty.html.haml
@@ -16,3 +16,5 @@
         = link_to "Explore", explore_root_path
         = link_to "Help", help_path
         = link_to "About GitLab", "https://about.gitlab.com/"
+
+    = javascript_include_tag(*webpack_asset_paths("raven")) if sentry_enabled?
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 70d98b022c1ff..62118522606a0 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -45,6 +45,7 @@ var config = {
     u2f:                  ['vendor/u2f'],
     users:                './users/users_bundle.js',
     vue_pipelines:        './vue_pipelines_index/index.js',
+    raven:                './raven/index.js',
   },
 
   output: {
diff --git a/lib/gitlab/gon_helper.rb b/lib/gitlab/gon_helper.rb
index 956e80b4a09cf..4de504e9bf984 100644
--- a/lib/gitlab/gon_helper.rb
+++ b/lib/gitlab/gon_helper.rb
@@ -13,7 +13,6 @@ def add_gon_variables
       gon.katex_css_url          = ActionController::Base.helpers.asset_path('katex.css')
       gon.katex_js_url           = ActionController::Base.helpers.asset_path('katex.js')
       gon.sentry_dsn             = sentry_dsn_public if sentry_enabled?
-      gon.raven_asset_url        = ActionController::Base.helpers.asset_path('raven.js') if sentry_enabled?
       gon.gitlab_url             = Gitlab.config.gitlab.url
       gon.is_production          = Rails.env.production?
 
diff --git a/package.json b/package.json
index 7b6c4556e2c4a..0b24c5b8b0434 100644
--- a/package.json
+++ b/package.json
@@ -30,6 +30,7 @@
     "mousetrap": "^1.4.6",
     "pikaday": "^1.5.1",
     "raphael": "^2.2.7",
+    "raven-js": "^3.14.0",
     "raw-loader": "^0.5.1",
     "select2": "3.5.2-browserify",
     "stats-webpack-plugin": "^0.4.3",
diff --git a/spec/features/raven_js_spec.rb b/spec/features/raven_js_spec.rb
index e64da1e3a974f..74df52d80a73e 100644
--- a/spec/features/raven_js_spec.rb
+++ b/spec/features/raven_js_spec.rb
@@ -1,7 +1,7 @@
 require 'spec_helper'
 
-feature 'RavenJS', feature: true, js: true do
-  let(:raven_path) { '/raven.js' }
+feature 'RavenJS', :feature, :js do
+  let(:raven_path) { '/raven.bundle.js' }
 
   it 'should not load raven if sentry is disabled' do
     visit new_user_session_path
@@ -10,8 +10,8 @@
   end
 
   it 'should load raven if sentry is enabled' do
-    allow_any_instance_of(ApplicationController).to receive_messages(sentry_dsn_public: 'https://mock:sentry@dsn/path',
-                                                                     sentry_enabled?: true)
+    allow_any_instance_of(SentryHelper).to receive_messages(sentry_dsn_public: 'https://key@domain.com/id',
+                                                            sentry_enabled?: true)
 
     visit new_user_session_path
 
diff --git a/spec/javascripts/class_spec_helper.js.es6 b/spec/javascripts/class_spec_helper.js.es6
deleted file mode 100644
index 3a04e170924d5..0000000000000
--- a/spec/javascripts/class_spec_helper.js.es6
+++ /dev/null
@@ -1,10 +0,0 @@
-/* eslint-disable no-unused-vars */
-
-class ClassSpecHelper {
-  static itShouldBeAStaticMethod(base, method) {
-    return it('should be a static method', () => {
-      expect(base[method]).toBeDefined();
-      expect(base.prototype[method]).toBeUndefined();
-    });
-  }
-}
diff --git a/spec/javascripts/class_spec_helper_spec.js.es6 b/spec/javascripts/class_spec_helper_spec.js.es6
deleted file mode 100644
index d1155f1bd1edd..0000000000000
--- a/spec/javascripts/class_spec_helper_spec.js.es6
+++ /dev/null
@@ -1,35 +0,0 @@
-/* global ClassSpecHelper */
-//= require ./class_spec_helper
-
-describe('ClassSpecHelper', () => {
-  describe('.itShouldBeAStaticMethod', function () {
-    beforeEach(() => {
-      class TestClass {
-        instanceMethod() { this.prop = 'val'; }
-        static staticMethod() {}
-      }
-
-      this.TestClass = TestClass;
-    });
-
-    ClassSpecHelper.itShouldBeAStaticMethod(ClassSpecHelper, 'itShouldBeAStaticMethod');
-
-    it('should have a defined spec', () => {
-      expect(ClassSpecHelper.itShouldBeAStaticMethod(this.TestClass, 'staticMethod').description).toBe('should be a static method');
-    });
-
-    it('should pass for a static method', () => {
-      const spec = ClassSpecHelper.itShouldBeAStaticMethod(this.TestClass, 'staticMethod');
-      expect(spec.status()).toBe('passed');
-    });
-
-    it('should fail for an instance method', (done) => {
-      const spec = ClassSpecHelper.itShouldBeAStaticMethod(this.TestClass, 'instanceMethod');
-      spec.resultCallback = (result) => {
-        expect(result.status).toBe('failed');
-        done();
-      };
-      spec.execute();
-    });
-  });
-});
diff --git a/spec/javascripts/lib/utils/load_script_spec.js.es6 b/spec/javascripts/lib/utils/load_script_spec.js.es6
deleted file mode 100644
index 52c53327695c7..0000000000000
--- a/spec/javascripts/lib/utils/load_script_spec.js.es6
+++ /dev/null
@@ -1,95 +0,0 @@
-/* global ClassSpecHelper */
-
-/*= require lib/utils/load_script */
-/*= require class_spec_helper */
-
-describe('LoadScript', () => {
-  const global = window.gl || (window.gl = {});
-  const LoadScript = global.LoadScript;
-
-  it('should be defined in the global scope', () => {
-    expect(LoadScript).toBeDefined();
-  });
-
-  describe('.load', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(LoadScript, 'load');
-
-    it('should reject if no source argument is provided', () => {
-      spyOn(Promise, 'reject');
-      LoadScript.load();
-      expect(Promise.reject).toHaveBeenCalledWith('source url must be defined');
-    });
-
-    it('should reject if the script id already exists', () => {
-      spyOn(Promise, 'reject');
-      spyOn(document, 'querySelector').and.returnValue({});
-      LoadScript.load('src.js', 'src-id');
-
-      expect(Promise.reject).toHaveBeenCalledWith('script id already exists');
-    });
-
-    it('should return a promise on completion', () => {
-      expect(LoadScript.load('src.js')).toEqual(jasmine.any(Promise));
-    });
-
-    it('should call appendScript when the promise is constructed', () => {
-      spyOn(LoadScript, 'appendScript');
-      LoadScript.load('src.js', 'src-id');
-
-      expect(LoadScript.appendScript).toHaveBeenCalledWith('src.js', 'src-id', jasmine.any(Promise.resolve.constructor), jasmine.any(Promise.reject.constructor));
-    });
-  });
-
-  describe('.appendScript', () => {
-    beforeEach(() => {
-      spyOn(document.body, 'appendChild');
-    });
-
-    ClassSpecHelper.itShouldBeAStaticMethod(LoadScript, 'appendScript');
-
-    describe('when called', () => {
-      let mockScriptTag;
-
-      beforeEach(() => {
-        mockScriptTag = {};
-        spyOn(document, 'createElement').and.returnValue(mockScriptTag);
-        LoadScript.appendScript('src.js', 'src-id', () => {}, () => {});
-      });
-
-      it('should create a script tag', () => {
-        expect(document.createElement).toHaveBeenCalledWith('script');
-      });
-
-      it('should set the MIME type', () => {
-        expect(mockScriptTag.type).toBe('text/javascript');
-      });
-
-      it('should set the script id', () => {
-        expect(mockScriptTag.id).toBe('src-id');
-      });
-
-      it('should set an onload handler', () => {
-        expect(mockScriptTag.onload).toEqual(jasmine.any(Function));
-      });
-
-      it('should set an onerror handler', () => {
-        expect(mockScriptTag.onerror).toEqual(jasmine.any(Function));
-      });
-
-      it('should set the src attribute', () => {
-        expect(mockScriptTag.src).toBe('src.js');
-      });
-
-      it('should append the script tag to the body element', () => {
-        expect(document.body.appendChild).toHaveBeenCalledWith(mockScriptTag);
-      });
-    });
-
-    it('should not set the script id if no id is provided', () => {
-      const mockScriptTag = {};
-      spyOn(document, 'createElement').and.returnValue(mockScriptTag);
-      LoadScript.appendScript('src.js', undefined);
-      expect(mockScriptTag.id).toBeUndefined();
-    });
-  });
-});
diff --git a/spec/javascripts/raven/index_spec.js b/spec/javascripts/raven/index_spec.js
new file mode 100644
index 0000000000000..51e84a6dbeea4
--- /dev/null
+++ b/spec/javascripts/raven/index_spec.js
@@ -0,0 +1,11 @@
+import RavenConfig from '~/raven/index';
+
+describe('RavenConfig options', () => {
+  it('should set sentryDsn');
+
+  it('should set currentUserId');
+
+  it('should set whitelistUrls');
+
+  it('should set isProduction');
+});
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
new file mode 100644
index 0000000000000..2b63f56d71904
--- /dev/null
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -0,0 +1,137 @@
+import Raven from 'raven-js';
+import RavenConfig from '~/raven/raven_config';
+import ClassSpecHelper from '../helpers/class_spec_helper';
+
+fdescribe('RavenConfig', () => {
+  describe('init', () => {
+    beforeEach(() => {
+      spyOn(RavenConfig, 'configure');
+      spyOn(RavenConfig, 'bindRavenErrors');
+      spyOn(RavenConfig, 'setUser');
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'init');
+
+    describe('when called', () => {
+      let options;
+
+      beforeEach(() => {
+        options = {
+          sentryDsn: '//sentryDsn',
+          ravenAssetUrl: '//ravenAssetUrl',
+          currentUserId: 1,
+          whitelistUrls: ['//gitlabUrl'],
+          isProduction: true,
+        };
+
+        RavenConfig.init(options);
+      });
+
+      it('should set the options property', () => {
+        expect(RavenConfig.options).toEqual(options);
+      });
+
+      it('should call the configure method', () => {
+        expect(RavenConfig.configure).toHaveBeenCalled();
+      });
+
+      it('should call the error bindings method', () => {
+        expect(RavenConfig.bindRavenErrors).toHaveBeenCalled();
+      });
+
+      it('should call setUser', () => {
+        expect(RavenConfig.setUser).toHaveBeenCalled();
+      });
+    });
+
+    it('should not call setUser if there is no current user ID', () => {
+      RavenConfig.init({
+        sentryDsn: '//sentryDsn',
+        ravenAssetUrl: '//ravenAssetUrl',
+        currentUserId: undefined,
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      });
+
+      expect(RavenConfig.setUser).not.toHaveBeenCalled();
+    });
+  });
+
+  describe('configure', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'configure');
+
+    describe('when called', () => {
+      let options;
+      let raven;
+
+      beforeEach(() => {
+        options = {
+          sentryDsn: '//sentryDsn',
+          whitelistUrls: ['//gitlabUrl'],
+          isProduction: true,
+        };
+
+        raven = jasmine.createSpyObj('raven', ['install']);
+
+        spyOn(Raven, 'config').and.returnValue(raven);
+        spyOn(Raven, 'install');
+
+        RavenConfig.configure.call({
+          options,
+        });
+      });
+
+      it('should call Raven.config', () => {
+        expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
+          whitelistUrls: options.whitelistUrls,
+          environment: 'production',
+        });
+      });
+
+      it('should call Raven.install', () => {
+        expect(Raven.install).toHaveBeenCalled();
+      });
+
+      describe('if isProduction is false', () => {
+        beforeEach(() => {
+          options.isProduction = false;
+
+          RavenConfig.configure.call({
+            options,
+          });
+        });
+
+        it('should set .environment to development', () => {
+          expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
+            whitelistUrls: options.whitelistUrls,
+            environment: 'development',
+          });
+        });
+      });
+    });
+  });
+
+  describe('setUser', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'setUser');
+
+    describe('when called', () => {
+      beforeEach(() => {});
+    });
+  });
+
+  describe('bindRavenErrors', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'bindRavenErrors');
+
+    describe('when called', () => {
+      beforeEach(() => {});
+    });
+  });
+
+  describe('handleRavenErrors', () => {
+    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'handleRavenErrors');
+
+    describe('when called', () => {
+      beforeEach(() => {});
+    });
+  });
+});
diff --git a/spec/javascripts/raven_config_spec.js.es6 b/spec/javascripts/raven_config_spec.js.es6
deleted file mode 100644
index 25df2cec75f03..0000000000000
--- a/spec/javascripts/raven_config_spec.js.es6
+++ /dev/null
@@ -1,142 +0,0 @@
-/* global ClassSpecHelper */
-
-/*= require raven */
-/*= require lib/utils/load_script */
-/*= require raven_config */
-/*= require class_spec_helper */
-
-describe('RavenConfig', () => {
-  const global = window.gl || (window.gl = {});
-  const RavenConfig = global.RavenConfig;
-
-  it('should be defined in the global scope', () => {
-    expect(RavenConfig).toBeDefined();
-  });
-
-  describe('.init', () => {
-    beforeEach(() => {
-      spyOn(global.LoadScript, 'load').and.callThrough();
-      spyOn(document, 'querySelector').and.returnValue(undefined);
-      spyOn(RavenConfig, 'configure');
-      spyOn(RavenConfig, 'bindRavenErrors');
-      spyOn(RavenConfig, 'setUser');
-      spyOn(Promise, 'reject');
-    });
-
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'init');
-
-    describe('when called', () => {
-      let options;
-      let initPromise;
-
-      beforeEach(() => {
-        options = {
-          sentryDsn: '//sentryDsn',
-          ravenAssetUrl: '//ravenAssetUrl',
-          currentUserId: 1,
-          whitelistUrls: ['//gitlabUrl'],
-          isProduction: true,
-        };
-        initPromise = RavenConfig.init(options);
-      });
-
-      it('should set the options property', () => {
-        expect(RavenConfig.options).toEqual(options);
-      });
-
-      it('should load a #raven-js script with the raven asset URL', () => {
-        expect(global.LoadScript.load).toHaveBeenCalledWith(options.ravenAssetUrl, 'raven-js');
-      });
-
-      it('should return a promise', () => {
-        expect(initPromise).toEqual(jasmine.any(Promise));
-      });
-
-      it('should call the configure method', () => {
-        initPromise.then(() => {
-          expect(RavenConfig.configure).toHaveBeenCalled();
-        });
-      });
-
-      it('should call the error bindings method', () => {
-        initPromise.then(() => {
-          expect(RavenConfig.bindRavenErrors).toHaveBeenCalled();
-        });
-      });
-
-      it('should call setUser', () => {
-        initPromise.then(() => {
-          expect(RavenConfig.setUser).toHaveBeenCalled();
-        });
-      });
-    });
-
-    it('should not call setUser if there is no current user ID', () => {
-      RavenConfig.init({
-        sentryDsn: '//sentryDsn',
-        ravenAssetUrl: '//ravenAssetUrl',
-        currentUserId: undefined,
-        whitelistUrls: ['//gitlabUrl'],
-        isProduction: true,
-      });
-
-      expect(RavenConfig.setUser).not.toHaveBeenCalled();
-    });
-
-    it('should reject if there is no Sentry DSN', () => {
-      RavenConfig.init({
-        sentryDsn: undefined,
-        ravenAssetUrl: '//ravenAssetUrl',
-        currentUserId: 1,
-        whitelistUrls: ['//gitlabUrl'],
-        isProduction: true,
-      });
-
-      expect(Promise.reject).toHaveBeenCalledWith('sentry dsn and raven asset url is required');
-    });
-
-    it('should reject if there is no Raven asset URL', () => {
-      RavenConfig.init({
-        sentryDsn: '//sentryDsn',
-        ravenAssetUrl: undefined,
-        currentUserId: 1,
-        whitelistUrls: ['//gitlabUrl'],
-        isProduction: true,
-      });
-
-      expect(Promise.reject).toHaveBeenCalledWith('sentry dsn and raven asset url is required');
-    });
-  });
-
-  describe('.configure', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'configure');
-
-    describe('when called', () => {
-      beforeEach(() => {});
-    });
-  });
-
-  describe('.setUser', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'setUser');
-
-    describe('when called', () => {
-      beforeEach(() => {});
-    });
-  });
-
-  describe('.bindRavenErrors', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'bindRavenErrors');
-
-    describe('when called', () => {
-      beforeEach(() => {});
-    });
-  });
-
-  describe('.handleRavenErrors', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'handleRavenErrors');
-
-    describe('when called', () => {
-      beforeEach(() => {});
-    });
-  });
-});
diff --git a/vendor/assets/javascripts/raven.js b/vendor/assets/javascripts/raven.js
deleted file mode 100644
index ba416b26be14d..0000000000000
--- a/vendor/assets/javascripts/raven.js
+++ /dev/null
@@ -1,2547 +0,0 @@
-/*! Raven.js 3.9.1 (7bbae7d) | github.com/getsentry/raven-js */
-
-/*
- * Includes TraceKit
- * https://github.com/getsentry/TraceKit
- *
- * Copyright 2016 Matt Robenolt and other contributors
- * Released under the BSD license
- * https://github.com/getsentry/raven-js/blob/master/LICENSE
- *
- */
-
-(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.Raven = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(_dereq_,module,exports){
-exports = module.exports = stringify
-exports.getSerialize = serializer
-
-function stringify(obj, replacer, spaces, cycleReplacer) {
-  return JSON.stringify(obj, serializer(replacer, cycleReplacer), spaces)
-}
-
-function serializer(replacer, cycleReplacer) {
-  var stack = [], keys = []
-
-  if (cycleReplacer == null) cycleReplacer = function(key, value) {
-    if (stack[0] === value) return "[Circular ~]"
-    return "[Circular ~." + keys.slice(0, stack.indexOf(value)).join(".") + "]"
-  }
-
-  return function(key, value) {
-    if (stack.length > 0) {
-      var thisPos = stack.indexOf(this)
-      ~thisPos ? stack.splice(thisPos + 1) : stack.push(this)
-      ~thisPos ? keys.splice(thisPos, Infinity, key) : keys.push(key)
-      if (~stack.indexOf(value)) value = cycleReplacer.call(this, key, value)
-    }
-    else stack.push(value)
-
-    return replacer == null ? value : replacer.call(this, key, value)
-  }
-}
-
-},{}],2:[function(_dereq_,module,exports){
-'use strict';
-
-function RavenConfigError(message) {
-    this.name = 'RavenConfigError';
-    this.message = message;
-}
-RavenConfigError.prototype = new Error();
-RavenConfigError.prototype.constructor = RavenConfigError;
-
-module.exports = RavenConfigError;
-
-},{}],3:[function(_dereq_,module,exports){
-'use strict';
-
-var wrapMethod = function(console, level, callback) {
-    var originalConsoleLevel = console[level];
-    var originalConsole = console;
-
-    if (!(level in console)) {
-        return;
-    }
-
-    var sentryLevel = level === 'warn'
-        ? 'warning'
-        : level;
-
-    console[level] = function () {
-        var args = [].slice.call(arguments);
-
-        var msg = '' + args.join(' ');
-        var data = {level: sentryLevel, logger: 'console', extra: {'arguments': args}};
-        callback && callback(msg, data);
-
-        // this fails for some browsers. :(
-        if (originalConsoleLevel) {
-            // IE9 doesn't allow calling apply on console functions directly
-            // See: https://stackoverflow.com/questions/5472938/does-ie9-support-console-log-and-is-it-a-real-function#answer-5473193
-            Function.prototype.apply.call(
-                originalConsoleLevel,
-                originalConsole,
-                args
-            );
-        }
-    };
-};
-
-module.exports = {
-    wrapMethod: wrapMethod
-};
-
-},{}],4:[function(_dereq_,module,exports){
-(function (global){
-/*global XDomainRequest:false, __DEV__:false*/
-'use strict';
-
-var TraceKit = _dereq_(6);
-var RavenConfigError = _dereq_(2);
-var stringify = _dereq_(1);
-
-var wrapConsoleMethod = _dereq_(3).wrapMethod;
-
-var dsnKeys = 'source protocol user pass host port path'.split(' '),
-    dsnPattern = /^(?:(\w+):)?\/\/(?:(\w+)(:\w+)?@)?([\w\.-]+)(?::(\d+))?(\/.*)/;
-
-function now() {
-    return +new Date();
-}
-
-// This is to be defensive in environments where window does not exist (see https://github.com/getsentry/raven-js/pull/785)
-var _window = typeof window !== 'undefined' ? window
-            : typeof global !== 'undefined' ? global
-            : typeof self !== 'undefined' ? self
-            : {};
-var _document = _window.document;
-
-// First, check for JSON support
-// If there is no JSON, we no-op the core features of Raven
-// since JSON is required to encode the payload
-function Raven() {
-    this._hasJSON = !!(typeof JSON === 'object' && JSON.stringify);
-    // Raven can run in contexts where there's no document (react-native)
-    this._hasDocument = !isUndefined(_document);
-    this._lastCapturedException = null;
-    this._lastEventId = null;
-    this._globalServer = null;
-    this._globalKey = null;
-    this._globalProject = null;
-    this._globalContext = {};
-    this._globalOptions = {
-        logger: 'javascript',
-        ignoreErrors: [],
-        ignoreUrls: [],
-        whitelistUrls: [],
-        includePaths: [],
-        crossOrigin: 'anonymous',
-        collectWindowErrors: true,
-        maxMessageLength: 0,
-        stackTraceLimit: 50,
-        autoBreadcrumbs: true
-    };
-    this._ignoreOnError = 0;
-    this._isRavenInstalled = false;
-    this._originalErrorStackTraceLimit = Error.stackTraceLimit;
-    // capture references to window.console *and* all its methods first
-    // before the console plugin has a chance to monkey patch
-    this._originalConsole = _window.console || {};
-    this._originalConsoleMethods = {};
-    this._plugins = [];
-    this._startTime = now();
-    this._wrappedBuiltIns = [];
-    this._breadcrumbs = [];
-    this._lastCapturedEvent = null;
-    this._keypressTimeout;
-    this._location = _window.location;
-    this._lastHref = this._location && this._location.href;
-
-    for (var method in this._originalConsole) {  // eslint-disable-line guard-for-in
-      this._originalConsoleMethods[method] = this._originalConsole[method];
-    }
-}
-
-/*
- * The core Raven singleton
- *
- * @this {Raven}
- */
-
-Raven.prototype = {
-    // Hardcode version string so that raven source can be loaded directly via
-    // webpack (using a build step causes webpack #1617). Grunt verifies that
-    // this value matches package.json during build.
-    //   See: https://github.com/getsentry/raven-js/issues/465
-    VERSION: '3.9.1',
-
-    debug: false,
-
-    TraceKit: TraceKit, // alias to TraceKit
-
-    /*
-     * Configure Raven with a DSN and extra options
-     *
-     * @param {string} dsn The public Sentry DSN
-     * @param {object} options Optional set of of global options [optional]
-     * @return {Raven}
-     */
-    config: function(dsn, options) {
-        var self = this;
-
-        if (self._globalServer) {
-                this._logDebug('error', 'Error: Raven has already been configured');
-            return self;
-        }
-        if (!dsn) return self;
-
-        var globalOptions = self._globalOptions;
-
-        // merge in options
-        if (options) {
-            each(options, function(key, value){
-                // tags and extra are special and need to be put into context
-                if (key === 'tags' || key === 'extra' || key === 'user') {
-                    self._globalContext[key] = value;
-                } else {
-                    globalOptions[key] = value;
-                }
-            });
-        }
-
-        self.setDSN(dsn);
-
-        // "Script error." is hard coded into browsers for errors that it can't read.
-        // this is the result of a script being pulled in from an external domain and CORS.
-        globalOptions.ignoreErrors.push(/^Script error\.?$/);
-        globalOptions.ignoreErrors.push(/^Javascript error: Script error\.? on line 0$/);
-
-        // join regexp rules into one big rule
-        globalOptions.ignoreErrors = joinRegExp(globalOptions.ignoreErrors);
-        globalOptions.ignoreUrls = globalOptions.ignoreUrls.length ? joinRegExp(globalOptions.ignoreUrls) : false;
-        globalOptions.whitelistUrls = globalOptions.whitelistUrls.length ? joinRegExp(globalOptions.whitelistUrls) : false;
-        globalOptions.includePaths = joinRegExp(globalOptions.includePaths);
-        globalOptions.maxBreadcrumbs = Math.max(0, Math.min(globalOptions.maxBreadcrumbs || 100, 100)); // default and hard limit is 100
-
-        var autoBreadcrumbDefaults = {
-            xhr: true,
-            console: true,
-            dom: true,
-            location: true
-        };
-
-        var autoBreadcrumbs = globalOptions.autoBreadcrumbs;
-        if ({}.toString.call(autoBreadcrumbs) === '[object Object]') {
-            autoBreadcrumbs = objectMerge(autoBreadcrumbDefaults, autoBreadcrumbs);
-        } else if (autoBreadcrumbs !== false) {
-            autoBreadcrumbs = autoBreadcrumbDefaults;
-        }
-        globalOptions.autoBreadcrumbs = autoBreadcrumbs;
-
-        TraceKit.collectWindowErrors = !!globalOptions.collectWindowErrors;
-
-        // return for chaining
-        return self;
-    },
-
-    /*
-     * Installs a global window.onerror error handler
-     * to capture and report uncaught exceptions.
-     * At this point, install() is required to be called due
-     * to the way TraceKit is set up.
-     *
-     * @return {Raven}
-     */
-    install: function() {
-        var self = this;
-        if (self.isSetup() && !self._isRavenInstalled) {
-            TraceKit.report.subscribe(function () {
-                self._handleOnErrorStackInfo.apply(self, arguments);
-            });
-            self._instrumentTryCatch();
-            if (self._globalOptions.autoBreadcrumbs)
-                self._instrumentBreadcrumbs();
-
-            // Install all of the plugins
-            self._drainPlugins();
-
-            self._isRavenInstalled = true;
-        }
-
-        Error.stackTraceLimit = self._globalOptions.stackTraceLimit;
-        return this;
-    },
-
-    /*
-     * Set the DSN (can be called multiple time unlike config)
-     *
-     * @param {string} dsn The public Sentry DSN
-     */
-    setDSN: function(dsn) {
-        var self = this,
-            uri = self._parseDSN(dsn),
-          lastSlash = uri.path.lastIndexOf('/'),
-          path = uri.path.substr(1, lastSlash);
-
-        self._dsn = dsn;
-        self._globalKey = uri.user;
-        self._globalSecret = uri.pass && uri.pass.substr(1);
-        self._globalProject = uri.path.substr(lastSlash + 1);
-
-        self._globalServer = self._getGlobalServer(uri);
-
-        self._globalEndpoint = self._globalServer +
-            '/' + path + 'api/' + self._globalProject + '/store/';
-    },
-
-    /*
-     * Wrap code within a context so Raven can capture errors
-     * reliably across domains that is executed immediately.
-     *
-     * @param {object} options A specific set of options for this context [optional]
-     * @param {function} func The callback to be immediately executed within the context
-     * @param {array} args An array of arguments to be called with the callback [optional]
-     */
-    context: function(options, func, args) {
-        if (isFunction(options)) {
-            args = func || [];
-            func = options;
-            options = undefined;
-        }
-
-        return this.wrap(options, func).apply(this, args);
-    },
-
-    /*
-     * Wrap code within a context and returns back a new function to be executed
-     *
-     * @param {object} options A specific set of options for this context [optional]
-     * @param {function} func The function to be wrapped in a new context
-     * @param {function} func A function to call before the try/catch wrapper [optional, private]
-     * @return {function} The newly wrapped functions with a context
-     */
-    wrap: function(options, func, _before) {
-        var self = this;
-        // 1 argument has been passed, and it's not a function
-        // so just return it
-        if (isUndefined(func) && !isFunction(options)) {
-            return options;
-        }
-
-        // options is optional
-        if (isFunction(options)) {
-            func = options;
-            options = undefined;
-        }
-
-        // At this point, we've passed along 2 arguments, and the second one
-        // is not a function either, so we'll just return the second argument.
-        if (!isFunction(func)) {
-            return func;
-        }
-
-        // We don't wanna wrap it twice!
-        try {
-            if (func.__raven__) {
-                return func;
-            }
-
-            // If this has already been wrapped in the past, return that
-            if (func.__raven_wrapper__ ){
-                return func.__raven_wrapper__ ;
-            }
-        } catch (e) {
-            // Just accessing custom props in some Selenium environments
-            // can cause a "Permission denied" exception (see raven-js#495).
-            // Bail on wrapping and return the function as-is (defers to window.onerror).
-            return func;
-        }
-
-        function wrapped() {
-            var args = [], i = arguments.length,
-                deep = !options || options && options.deep !== false;
-
-            if (_before && isFunction(_before)) {
-                _before.apply(this, arguments);
-            }
-
-            // Recursively wrap all of a function's arguments that are
-            // functions themselves.
-            while(i--) args[i] = deep ? self.wrap(options, arguments[i]) : arguments[i];
-
-            try {
-                return func.apply(this, args);
-            } catch(e) {
-                self._ignoreNextOnError();
-                self.captureException(e, options);
-                throw e;
-            }
-        }
-
-        // copy over properties of the old function
-        for (var property in func) {
-            if (hasKey(func, property)) {
-                wrapped[property] = func[property];
-            }
-        }
-        wrapped.prototype = func.prototype;
-
-        func.__raven_wrapper__ = wrapped;
-        // Signal that this function has been wrapped already
-        // for both debugging and to prevent it to being wrapped twice
-        wrapped.__raven__ = true;
-        wrapped.__inner__ = func;
-
-        return wrapped;
-    },
-
-    /*
-     * Uninstalls the global error handler.
-     *
-     * @return {Raven}
-     */
-    uninstall: function() {
-        TraceKit.report.uninstall();
-
-        this._restoreBuiltIns();
-
-        Error.stackTraceLimit = this._originalErrorStackTraceLimit;
-        this._isRavenInstalled = false;
-
-        return this;
-    },
-
-    /*
-     * Manually capture an exception and send it over to Sentry
-     *
-     * @param {error} ex An exception to be logged
-     * @param {object} options A specific set of options for this error [optional]
-     * @return {Raven}
-     */
-    captureException: function(ex, options) {
-        // If not an Error is passed through, recall as a message instead
-        if (!isError(ex)) {
-            return this.captureMessage(ex, objectMerge({
-                trimHeadFrames: 1,
-                stacktrace: true // if we fall back to captureMessage, default to attempting a new trace
-            }, options));
-        }
-
-        // Store the raw exception object for potential debugging and introspection
-        this._lastCapturedException = ex;
-
-        // TraceKit.report will re-raise any exception passed to it,
-        // which means you have to wrap it in try/catch. Instead, we
-        // can wrap it here and only re-raise if TraceKit.report
-        // raises an exception different from the one we asked to
-        // report on.
-        try {
-            var stack = TraceKit.computeStackTrace(ex);
-            this._handleStackInfo(stack, options);
-        } catch(ex1) {
-            if(ex !== ex1) {
-                throw ex1;
-            }
-        }
-
-        return this;
-    },
-
-    /*
-     * Manually send a message to Sentry
-     *
-     * @param {string} msg A plain message to be captured in Sentry
-     * @param {object} options A specific set of options for this message [optional]
-     * @return {Raven}
-     */
-    captureMessage: function(msg, options) {
-        // config() automagically converts ignoreErrors from a list to a RegExp so we need to test for an
-        // early call; we'll error on the side of logging anything called before configuration since it's
-        // probably something you should see:
-        if (!!this._globalOptions.ignoreErrors.test && this._globalOptions.ignoreErrors.test(msg)) {
-            return;
-        }
-
-        options = options || {};
-
-        var data = objectMerge({
-            message: msg + ''  // Make sure it's actually a string
-        }, options);
-
-        if (this._globalOptions.stacktrace || (options && options.stacktrace)) {
-            var ex;
-            // create a stack trace from this point; just trim
-            // off extra frames so they don't include this function call (or
-            // earlier Raven.js library fn calls)
-            try {
-                throw new Error(msg);
-            } catch (ex1) {
-                ex = ex1;
-            }
-
-            // null exception name so `Error` isn't prefixed to msg
-            ex.name = null;
-
-            options = objectMerge({
-                // fingerprint on msg, not stack trace (legacy behavior, could be
-                // revisited)
-                fingerprint: msg,
-                trimHeadFrames: (options.trimHeadFrames || 0) + 1
-            }, options);
-
-            var stack = TraceKit.computeStackTrace(ex);
-            var frames = this._prepareFrames(stack, options);
-            data.stacktrace = {
-                // Sentry expects frames oldest to newest
-                frames: frames.reverse()
-            }
-        }
-
-        // Fire away!
-        this._send(data);
-
-        return this;
-    },
-
-    captureBreadcrumb: function (obj) {
-        var crumb = objectMerge({
-            timestamp: now() / 1000
-        }, obj);
-
-        if (isFunction(this._globalOptions.breadcrumbCallback)) {
-            var result = this._globalOptions.breadcrumbCallback(crumb);
-
-            if (isObject(result) && !isEmptyObject(result)) {
-                crumb = result;
-            } else if (result === false) {
-                return this;
-            }
-        }
-
-        this._breadcrumbs.push(crumb);
-        if (this._breadcrumbs.length > this._globalOptions.maxBreadcrumbs) {
-            this._breadcrumbs.shift();
-        }
-        return this;
-    },
-
-    addPlugin: function(plugin /*arg1, arg2, ... argN*/) {
-        var pluginArgs = [].slice.call(arguments, 1);
-
-        this._plugins.push([plugin, pluginArgs]);
-        if (this._isRavenInstalled) {
-            this._drainPlugins();
-        }
-
-        return this;
-    },
-
-    /*
-     * Set/clear a user to be sent along with the payload.
-     *
-     * @param {object} user An object representing user data [optional]
-     * @return {Raven}
-     */
-    setUserContext: function(user) {
-        // Intentionally do not merge here since that's an unexpected behavior.
-        this._globalContext.user = user;
-
-        return this;
-    },
-
-    /*
-     * Merge extra attributes to be sent along with the payload.
-     *
-     * @param {object} extra An object representing extra data [optional]
-     * @return {Raven}
-     */
-    setExtraContext: function(extra) {
-        this._mergeContext('extra', extra);
-
-        return this;
-    },
-
-    /*
-     * Merge tags to be sent along with the payload.
-     *
-     * @param {object} tags An object representing tags [optional]
-     * @return {Raven}
-     */
-    setTagsContext: function(tags) {
-        this._mergeContext('tags', tags);
-
-        return this;
-    },
-
-    /*
-     * Clear all of the context.
-     *
-     * @return {Raven}
-     */
-    clearContext: function() {
-        this._globalContext = {};
-
-        return this;
-    },
-
-    /*
-     * Get a copy of the current context. This cannot be mutated.
-     *
-     * @return {object} copy of context
-     */
-    getContext: function() {
-        // lol javascript
-        return JSON.parse(stringify(this._globalContext));
-    },
-
-
-    /*
-     * Set environment of application
-     *
-     * @param {string} environment Typically something like 'production'.
-     * @return {Raven}
-     */
-    setEnvironment: function(environment) {
-        this._globalOptions.environment = environment;
-
-        return this;
-    },
-
-    /*
-     * Set release version of application
-     *
-     * @param {string} release Typically something like a git SHA to identify version
-     * @return {Raven}
-     */
-    setRelease: function(release) {
-        this._globalOptions.release = release;
-
-        return this;
-    },
-
-    /*
-     * Set the dataCallback option
-     *
-     * @param {function} callback The callback to run which allows the
-     *                            data blob to be mutated before sending
-     * @return {Raven}
-     */
-    setDataCallback: function(callback) {
-        var original = this._globalOptions.dataCallback;
-        this._globalOptions.dataCallback = isFunction(callback)
-          ? function (data) { return callback(data, original); }
-          : callback;
-
-        return this;
-    },
-
-    /*
-     * Set the breadcrumbCallback option
-     *
-     * @param {function} callback The callback to run which allows filtering
-     *                            or mutating breadcrumbs
-     * @return {Raven}
-     */
-    setBreadcrumbCallback: function(callback) {
-        var original = this._globalOptions.breadcrumbCallback;
-        this._globalOptions.breadcrumbCallback = isFunction(callback)
-          ? function (data) { return callback(data, original); }
-          : callback;
-
-        return this;
-    },
-
-    /*
-     * Set the shouldSendCallback option
-     *
-     * @param {function} callback The callback to run which allows
-     *                            introspecting the blob before sending
-     * @return {Raven}
-     */
-    setShouldSendCallback: function(callback) {
-        var original = this._globalOptions.shouldSendCallback;
-        this._globalOptions.shouldSendCallback = isFunction(callback)
-            ? function (data) { return callback(data, original); }
-            : callback;
-
-        return this;
-    },
-
-    /**
-     * Override the default HTTP transport mechanism that transmits data
-     * to the Sentry server.
-     *
-     * @param {function} transport Function invoked instead of the default
-     *                             `makeRequest` handler.
-     *
-     * @return {Raven}
-     */
-    setTransport: function(transport) {
-        this._globalOptions.transport = transport;
-
-        return this;
-    },
-
-    /*
-     * Get the latest raw exception that was captured by Raven.
-     *
-     * @return {error}
-     */
-    lastException: function() {
-        return this._lastCapturedException;
-    },
-
-    /*
-     * Get the last event id
-     *
-     * @return {string}
-     */
-    lastEventId: function() {
-        return this._lastEventId;
-    },
-
-    /*
-     * Determine if Raven is setup and ready to go.
-     *
-     * @return {boolean}
-     */
-    isSetup: function() {
-        if (!this._hasJSON) return false;  // needs JSON support
-        if (!this._globalServer) {
-            if (!this.ravenNotConfiguredError) {
-              this.ravenNotConfiguredError = true;
-              this._logDebug('error', 'Error: Raven has not been configured.');
-            }
-            return false;
-        }
-        return true;
-    },
-
-    afterLoad: function () {
-        // TODO: remove window dependence?
-
-        // Attempt to initialize Raven on load
-        var RavenConfig = _window.RavenConfig;
-        if (RavenConfig) {
-            this.config(RavenConfig.dsn, RavenConfig.config).install();
-        }
-    },
-
-    showReportDialog: function (options) {
-        if (!_document) // doesn't work without a document (React native)
-            return;
-
-        options = options || {};
-
-        var lastEventId = options.eventId || this.lastEventId();
-        if (!lastEventId) {
-            throw new RavenConfigError('Missing eventId');
-        }
-
-        var dsn = options.dsn || this._dsn;
-        if (!dsn) {
-            throw new RavenConfigError('Missing DSN');
-        }
-
-        var encode = encodeURIComponent;
-        var qs = '';
-        qs += '?eventId=' + encode(lastEventId);
-        qs += '&dsn=' + encode(dsn);
-
-        var user = options.user || this._globalContext.user;
-        if (user) {
-            if (user.name)  qs += '&name=' + encode(user.name);
-            if (user.email) qs += '&email=' + encode(user.email);
-        }
-
-        var globalServer = this._getGlobalServer(this._parseDSN(dsn));
-
-        var script = _document.createElement('script');
-        script.async = true;
-        script.src = globalServer + '/api/embed/error-page/' + qs;
-        (_document.head || _document.body).appendChild(script);
-    },
-
-    /**** Private functions ****/
-    _ignoreNextOnError: function () {
-        var self = this;
-        this._ignoreOnError += 1;
-        setTimeout(function () {
-            // onerror should trigger before setTimeout
-            self._ignoreOnError -= 1;
-        });
-    },
-
-    _triggerEvent: function(eventType, options) {
-        // NOTE: `event` is a native browser thing, so let's avoid conflicting wiht it
-        var evt, key;
-
-        if (!this._hasDocument)
-            return;
-
-        options = options || {};
-
-        eventType = 'raven' + eventType.substr(0,1).toUpperCase() + eventType.substr(1);
-
-        if (_document.createEvent) {
-            evt = _document.createEvent('HTMLEvents');
-            evt.initEvent(eventType, true, true);
-        } else {
-            evt = _document.createEventObject();
-            evt.eventType = eventType;
-        }
-
-        for (key in options) if (hasKey(options, key)) {
-            evt[key] = options[key];
-        }
-
-        if (_document.createEvent) {
-            // IE9 if standards
-            _document.dispatchEvent(evt);
-        } else {
-            // IE8 regardless of Quirks or Standards
-            // IE9 if quirks
-            try {
-                _document.fireEvent('on' + evt.eventType.toLowerCase(), evt);
-            } catch(e) {
-                // Do nothing
-            }
-        }
-    },
-
-    /**
-     * Wraps addEventListener to capture UI breadcrumbs
-     * @param evtName the event name (e.g. "click")
-     * @returns {Function}
-     * @private
-     */
-    _breadcrumbEventHandler: function(evtName) {
-        var self = this;
-        return function (evt) {
-            // reset keypress timeout; e.g. triggering a 'click' after
-            // a 'keypress' will reset the keypress debounce so that a new
-            // set of keypresses can be recorded
-            self._keypressTimeout = null;
-
-            // It's possible this handler might trigger multiple times for the same
-            // event (e.g. event propagation through node ancestors). Ignore if we've
-            // already captured the event.
-            if (self._lastCapturedEvent === evt)
-                return;
-
-            self._lastCapturedEvent = evt;
-            var elem = evt.target;
-
-            var target;
-
-            // try/catch htmlTreeAsString because it's particularly complicated, and
-            // just accessing the DOM incorrectly can throw an exception in some circumstances.
-            try {
-                target = htmlTreeAsString(elem);
-            } catch (e) {
-                target = '<unknown>';
-            }
-
-            self.captureBreadcrumb({
-                category: 'ui.' + evtName, // e.g. ui.click, ui.input
-                message: target
-            });
-        };
-    },
-
-    /**
-     * Wraps addEventListener to capture keypress UI events
-     * @returns {Function}
-     * @private
-     */
-    _keypressEventHandler: function() {
-        var self = this,
-            debounceDuration = 1000; // milliseconds
-
-        // TODO: if somehow user switches keypress target before
-        //       debounce timeout is triggered, we will only capture
-        //       a single breadcrumb from the FIRST target (acceptable?)
-        return function (evt) {
-            var target = evt.target,
-                tagName = target && target.tagName;
-
-            // only consider keypress events on actual input elements
-            // this will disregard keypresses targeting body (e.g. tabbing
-            // through elements, hotkeys, etc)
-            if (!tagName || tagName !== 'INPUT' && tagName !== 'TEXTAREA' && !target.isContentEditable)
-                return;
-
-            // record first keypress in a series, but ignore subsequent
-            // keypresses until debounce clears
-            var timeout = self._keypressTimeout;
-            if (!timeout) {
-                self._breadcrumbEventHandler('input')(evt);
-            }
-            clearTimeout(timeout);
-            self._keypressTimeout = setTimeout(function () {
-                self._keypressTimeout = null;
-            }, debounceDuration);
-        };
-    },
-
-    /**
-     * Captures a breadcrumb of type "navigation", normalizing input URLs
-     * @param to the originating URL
-     * @param from the target URL
-     * @private
-     */
-    _captureUrlChange: function(from, to) {
-        var parsedLoc = parseUrl(this._location.href);
-        var parsedTo = parseUrl(to);
-        var parsedFrom = parseUrl(from);
-
-        // because onpopstate only tells you the "new" (to) value of location.href, and
-        // not the previous (from) value, we need to track the value of the current URL
-        // state ourselves
-        this._lastHref = to;
-
-        // Use only the path component of the URL if the URL matches the current
-        // document (almost all the time when using pushState)
-        if (parsedLoc.protocol === parsedTo.protocol && parsedLoc.host === parsedTo.host)
-            to = parsedTo.relative;
-        if (parsedLoc.protocol === parsedFrom.protocol && parsedLoc.host === parsedFrom.host)
-            from = parsedFrom.relative;
-
-        this.captureBreadcrumb({
-            category: 'navigation',
-            data: {
-                to: to,
-                from: from
-            }
-        });
-    },
-
-    /**
-     * Install any queued plugins
-     */
-    _instrumentTryCatch: function() {
-        var self = this;
-
-        var wrappedBuiltIns = self._wrappedBuiltIns;
-
-        function wrapTimeFn(orig) {
-            return function (fn, t) { // preserve arity
-                // Make a copy of the arguments to prevent deoptimization
-                // https://github.com/petkaantonov/bluebird/wiki/Optimization-killers#32-leaking-arguments
-                var args = new Array(arguments.length);
-                for(var i = 0; i < args.length; ++i) {
-                    args[i] = arguments[i];
-                }
-                var originalCallback = args[0];
-                if (isFunction(originalCallback)) {
-                    args[0] = self.wrap(originalCallback);
-                }
-
-                // IE < 9 doesn't support .call/.apply on setInterval/setTimeout, but it
-                // also supports only two arguments and doesn't care what this is, so we
-                // can just call the original function directly.
-                if (orig.apply) {
-                    return orig.apply(this, args);
-                } else {
-                    return orig(args[0], args[1]);
-                }
-            };
-        }
-
-        var autoBreadcrumbs = this._globalOptions.autoBreadcrumbs;
-
-        function wrapEventTarget(global) {
-            var proto = _window[global] && _window[global].prototype;
-            if (proto && proto.hasOwnProperty && proto.hasOwnProperty('addEventListener')) {
-                fill(proto, 'addEventListener', function(orig) {
-                    return function (evtName, fn, capture, secure) { // preserve arity
-                        try {
-                            if (fn && fn.handleEvent) {
-                                fn.handleEvent = self.wrap(fn.handleEvent);
-                            }
-                        } catch (err) {
-                            // can sometimes get 'Permission denied to access property "handle Event'
-                        }
-
-                        // More breadcrumb DOM capture ... done here and not in `_instrumentBreadcrumbs`
-                        // so that we don't have more than one wrapper function
-                        var before,
-                            clickHandler,
-                            keypressHandler;
-
-                        if (autoBreadcrumbs && autoBreadcrumbs.dom && (global === 'EventTarget' || global === 'Node')) {
-                            // NOTE: generating multiple handlers per addEventListener invocation, should
-                            //       revisit and verify we can just use one (almost certainly)
-                            clickHandler = self._breadcrumbEventHandler('click');
-                            keypressHandler = self._keypressEventHandler();
-                            before = function (evt) {
-                                // need to intercept every DOM event in `before` argument, in case that
-                                // same wrapped method is re-used for different events (e.g. mousemove THEN click)
-                                // see #724
-                                if (!evt) return;
-
-                                if (evt.type === 'click')
-                                    return clickHandler(evt);
-                                else if (evt.type === 'keypress')
-                                    return keypressHandler(evt);
-                            };
-                        }
-                        return orig.call(this, evtName, self.wrap(fn, undefined, before), capture, secure);
-                    };
-                }, wrappedBuiltIns);
-                fill(proto, 'removeEventListener', function (orig) {
-                    return function (evt, fn, capture, secure) {
-                        try {
-                            fn = fn && (fn.__raven_wrapper__ ? fn.__raven_wrapper__  : fn);
-                        } catch (e) {
-                            // ignore, accessing __raven_wrapper__ will throw in some Selenium environments
-                        }
-                        return orig.call(this, evt, fn, capture, secure);
-                    };
-                }, wrappedBuiltIns);
-            }
-        }
-
-        fill(_window, 'setTimeout', wrapTimeFn, wrappedBuiltIns);
-        fill(_window, 'setInterval', wrapTimeFn, wrappedBuiltIns);
-        if (_window.requestAnimationFrame) {
-            fill(_window, 'requestAnimationFrame', function (orig) {
-                return function (cb) {
-                    return orig(self.wrap(cb));
-                };
-            }, wrappedBuiltIns);
-        }
-
-        // event targets borrowed from bugsnag-js:
-        // https://github.com/bugsnag/bugsnag-js/blob/master/src/bugsnag.js#L666
-        var eventTargets = ['EventTarget', 'Window', 'Node', 'ApplicationCache', 'AudioTrackList', 'ChannelMergerNode', 'CryptoOperation', 'EventSource', 'FileReader', 'HTMLUnknownElement', 'IDBDatabase', 'IDBRequest', 'IDBTransaction', 'KeyOperation', 'MediaController', 'MessagePort', 'ModalWindow', 'Notification', 'SVGElementInstance', 'Screen', 'TextTrack', 'TextTrackCue', 'TextTrackList', 'WebSocket', 'WebSocketWorker', 'Worker', 'XMLHttpRequest', 'XMLHttpRequestEventTarget', 'XMLHttpRequestUpload'];
-        for (var i = 0; i < eventTargets.length; i++) {
-            wrapEventTarget(eventTargets[i]);
-        }
-
-        var $ = _window.jQuery || _window.$;
-        if ($ && $.fn && $.fn.ready) {
-            fill($.fn, 'ready', function (orig) {
-                return function (fn) {
-                    return orig.call(this, self.wrap(fn));
-                };
-            }, wrappedBuiltIns);
-        }
-    },
-
-
-    /**
-     * Instrument browser built-ins w/ breadcrumb capturing
-     *  - XMLHttpRequests
-     *  - DOM interactions (click/typing)
-     *  - window.location changes
-     *  - console
-     *
-     * Can be disabled or individually configured via the `autoBreadcrumbs` config option
-     */
-    _instrumentBreadcrumbs: function () {
-        var self = this;
-        var autoBreadcrumbs = this._globalOptions.autoBreadcrumbs;
-
-        var wrappedBuiltIns = self._wrappedBuiltIns;
-
-        function wrapProp(prop, xhr) {
-            if (prop in xhr && isFunction(xhr[prop])) {
-                fill(xhr, prop, function (orig) {
-                    return self.wrap(orig);
-                }); // intentionally don't track filled methods on XHR instances
-            }
-        }
-
-        if (autoBreadcrumbs.xhr && 'XMLHttpRequest' in _window) {
-            var xhrproto = XMLHttpRequest.prototype;
-            fill(xhrproto, 'open', function(origOpen) {
-                return function (method, url) { // preserve arity
-
-                    // if Sentry key appears in URL, don't capture
-                    if (isString(url) && url.indexOf(self._globalKey) === -1) {
-                        this.__raven_xhr = {
-                            method: method,
-                            url: url,
-                            status_code: null
-                        };
-                    }
-
-                    return origOpen.apply(this, arguments);
-                };
-            }, wrappedBuiltIns);
-
-            fill(xhrproto, 'send', function(origSend) {
-                return function (data) { // preserve arity
-                    var xhr = this;
-
-                    function onreadystatechangeHandler() {
-                        if (xhr.__raven_xhr && (xhr.readyState === 1 || xhr.readyState === 4)) {
-                            try {
-                                // touching statusCode in some platforms throws
-                                // an exception
-                                xhr.__raven_xhr.status_code = xhr.status;
-                            } catch (e) { /* do nothing */ }
-                            self.captureBreadcrumb({
-                                type: 'http',
-                                category: 'xhr',
-                                data: xhr.__raven_xhr
-                            });
-                        }
-                    }
-
-                    var props = ['onload', 'onerror', 'onprogress'];
-                    for (var j = 0; j < props.length; j++) {
-                        wrapProp(props[j], xhr);
-                    }
-
-                    if ('onreadystatechange' in xhr && isFunction(xhr.onreadystatechange)) {
-                        fill(xhr, 'onreadystatechange', function (orig) {
-                            return self.wrap(orig, undefined, onreadystatechangeHandler);
-                        } /* intentionally don't track this instrumentation */);
-                    } else {
-                        // if onreadystatechange wasn't actually set by the page on this xhr, we
-                        // are free to set our own and capture the breadcrumb
-                        xhr.onreadystatechange = onreadystatechangeHandler;
-                    }
-
-                    return origSend.apply(this, arguments);
-                };
-            }, wrappedBuiltIns);
-        }
-
-        if (autoBreadcrumbs.xhr && 'fetch' in _window) {
-            fill(_window, 'fetch', function(origFetch) {
-                return function (fn, t) { // preserve arity
-                    // Make a copy of the arguments to prevent deoptimization
-                    // https://github.com/petkaantonov/bluebird/wiki/Optimization-killers#32-leaking-arguments
-                    var args = new Array(arguments.length);
-                    for(var i = 0; i < args.length; ++i) {
-                        args[i] = arguments[i];
-                    }
-
-                    var method = 'GET';
-
-                    if (args[1] && args[1].method) {
-                        method = args[1].method;
-                    }
-
-                    var fetchData = {
-                        method: method,
-                        url: args[0],
-                        status_code: null
-                    };
-
-                    self.captureBreadcrumb({
-                        type: 'http',
-                        category: 'fetch',
-                        data: fetchData
-                    });
-
-                    return origFetch.apply(this, args).then(function (response) {
-                        fetchData.status_code = response.status;
-
-                        return response;
-                    });
-                };
-            }, wrappedBuiltIns);
-        }
-
-        // Capture breadcrumbs from any click that is unhandled / bubbled up all the way
-        // to the document. Do this before we instrument addEventListener.
-        if (autoBreadcrumbs.dom && this._hasDocument) {
-            if (_document.addEventListener) {
-                _document.addEventListener('click', self._breadcrumbEventHandler('click'), false);
-                _document.addEventListener('keypress', self._keypressEventHandler(), false);
-            }
-            else {
-                // IE8 Compatibility
-                _document.attachEvent('onclick', self._breadcrumbEventHandler('click'));
-                _document.attachEvent('onkeypress', self._keypressEventHandler());
-            }
-        }
-
-        // record navigation (URL) changes
-        // NOTE: in Chrome App environment, touching history.pushState, *even inside
-        //       a try/catch block*, will cause Chrome to output an error to console.error
-        // borrowed from: https://github.com/angular/angular.js/pull/13945/files
-        var chrome = _window.chrome;
-        var isChromePackagedApp = chrome && chrome.app && chrome.app.runtime;
-        var hasPushState = !isChromePackagedApp && _window.history && history.pushState;
-        if (autoBreadcrumbs.location && hasPushState) {
-            // TODO: remove onpopstate handler on uninstall()
-            var oldOnPopState = _window.onpopstate;
-            _window.onpopstate = function () {
-                var currentHref = self._location.href;
-                self._captureUrlChange(self._lastHref, currentHref);
-
-                if (oldOnPopState) {
-                    return oldOnPopState.apply(this, arguments);
-                }
-            };
-
-            fill(history, 'pushState', function (origPushState) {
-                // note history.pushState.length is 0; intentionally not declaring
-                // params to preserve 0 arity
-                return function (/* state, title, url */) {
-                    var url = arguments.length > 2 ? arguments[2] : undefined;
-
-                    // url argument is optional
-                    if (url) {
-                        // coerce to string (this is what pushState does)
-                        self._captureUrlChange(self._lastHref, url + '');
-                    }
-
-                    return origPushState.apply(this, arguments);
-                };
-            }, wrappedBuiltIns);
-        }
-
-        if (autoBreadcrumbs.console && 'console' in _window && console.log) {
-            // console
-            var consoleMethodCallback = function (msg, data) {
-                self.captureBreadcrumb({
-                    message: msg,
-                    level: data.level,
-                    category: 'console'
-                });
-            };
-
-            each(['debug', 'info', 'warn', 'error', 'log'], function (_, level) {
-                wrapConsoleMethod(console, level, consoleMethodCallback);
-            });
-        }
-
-    },
-
-    _restoreBuiltIns: function () {
-        // restore any wrapped builtins
-        var builtin;
-        while (this._wrappedBuiltIns.length) {
-            builtin = this._wrappedBuiltIns.shift();
-
-            var obj = builtin[0],
-              name = builtin[1],
-              orig = builtin[2];
-
-            obj[name] = orig;
-        }
-    },
-
-    _drainPlugins: function() {
-        var self = this;
-
-        // FIX ME TODO
-        each(this._plugins, function(_, plugin) {
-            var installer = plugin[0];
-            var args = plugin[1];
-            installer.apply(self, [self].concat(args));
-        });
-    },
-
-    _parseDSN: function(str) {
-        var m = dsnPattern.exec(str),
-            dsn = {},
-            i = 7;
-
-        try {
-            while (i--) dsn[dsnKeys[i]] = m[i] || '';
-        } catch(e) {
-            throw new RavenConfigError('Invalid DSN: ' + str);
-        }
-
-        if (dsn.pass && !this._globalOptions.allowSecretKey) {
-            throw new RavenConfigError('Do not specify your secret key in the DSN. See: http://bit.ly/raven-secret-key');
-        }
-
-        return dsn;
-    },
-
-    _getGlobalServer: function(uri) {
-        // assemble the endpoint from the uri pieces
-        var globalServer = '//' + uri.host +
-            (uri.port ? ':' + uri.port : '');
-
-        if (uri.protocol) {
-            globalServer = uri.protocol + ':' + globalServer;
-        }
-        return globalServer;
-    },
-
-    _handleOnErrorStackInfo: function() {
-        // if we are intentionally ignoring errors via onerror, bail out
-        if (!this._ignoreOnError) {
-            this._handleStackInfo.apply(this, arguments);
-        }
-    },
-
-    _handleStackInfo: function(stackInfo, options) {
-        var frames = this._prepareFrames(stackInfo, options);
-
-        this._triggerEvent('handle', {
-            stackInfo: stackInfo,
-            options: options
-        });
-
-        this._processException(
-            stackInfo.name,
-            stackInfo.message,
-            stackInfo.url,
-            stackInfo.lineno,
-            frames,
-            options
-        );
-    },
-
-    _prepareFrames: function(stackInfo, options) {
-        var self = this;
-        var frames = [];
-        if (stackInfo.stack && stackInfo.stack.length) {
-            each(stackInfo.stack, function(i, stack) {
-                var frame = self._normalizeFrame(stack);
-                if (frame) {
-                    frames.push(frame);
-                }
-            });
-
-            // e.g. frames captured via captureMessage throw
-            if (options && options.trimHeadFrames) {
-                for (var j = 0; j < options.trimHeadFrames && j < frames.length; j++) {
-                    frames[j].in_app = false;
-                }
-            }
-        }
-        frames = frames.slice(0, this._globalOptions.stackTraceLimit);
-        return frames;
-    },
-
-
-    _normalizeFrame: function(frame) {
-        if (!frame.url) return;
-
-        // normalize the frames data
-        var normalized = {
-            filename:   frame.url,
-            lineno:     frame.line,
-            colno:      frame.column,
-            'function': frame.func || '?'
-        };
-
-        normalized.in_app = !( // determine if an exception came from outside of our app
-            // first we check the global includePaths list.
-            !!this._globalOptions.includePaths.test && !this._globalOptions.includePaths.test(normalized.filename) ||
-            // Now we check for fun, if the function name is Raven or TraceKit
-            /(Raven|TraceKit)\./.test(normalized['function']) ||
-            // finally, we do a last ditch effort and check for raven.min.js
-            /raven\.(min\.)?js$/.test(normalized.filename)
-        );
-
-        return normalized;
-    },
-
-    _processException: function(type, message, fileurl, lineno, frames, options) {
-        var stacktrace;
-        if (!!this._globalOptions.ignoreErrors.test && this._globalOptions.ignoreErrors.test(message)) return;
-
-        message += '';
-
-        if (frames && frames.length) {
-            fileurl = frames[0].filename || fileurl;
-            // Sentry expects frames oldest to newest
-            // and JS sends them as newest to oldest
-            frames.reverse();
-            stacktrace = {frames: frames};
-        } else if (fileurl) {
-            stacktrace = {
-                frames: [{
-                    filename: fileurl,
-                    lineno: lineno,
-                    in_app: true
-                }]
-            };
-        }
-
-        if (!!this._globalOptions.ignoreUrls.test && this._globalOptions.ignoreUrls.test(fileurl)) return;
-        if (!!this._globalOptions.whitelistUrls.test && !this._globalOptions.whitelistUrls.test(fileurl)) return;
-
-        var data = objectMerge({
-            // sentry.interfaces.Exception
-            exception: {
-                values: [{
-                    type: type,
-                    value: message,
-                    stacktrace: stacktrace
-                }]
-            },
-            culprit: fileurl
-        }, options);
-
-        // Fire away!
-        this._send(data);
-    },
-
-    _trimPacket: function(data) {
-        // For now, we only want to truncate the two different messages
-        // but this could/should be expanded to just trim everything
-        var max = this._globalOptions.maxMessageLength;
-        if (data.message) {
-            data.message = truncate(data.message, max);
-        }
-        if (data.exception) {
-            var exception = data.exception.values[0];
-            exception.value = truncate(exception.value, max);
-        }
-
-        return data;
-    },
-
-    _getHttpData: function() {
-        if (!this._hasDocument || !_document.location || !_document.location.href) {
-            return;
-        }
-
-        var httpData = {
-            headers: {
-                'User-Agent': navigator.userAgent
-            }
-        };
-
-        httpData.url = _document.location.href;
-
-        if (_document.referrer) {
-            httpData.headers.Referer = _document.referrer;
-        }
-
-        return httpData;
-    },
-
-
-    _send: function(data) {
-        var globalOptions = this._globalOptions;
-
-        var baseData = {
-            project: this._globalProject,
-            logger: globalOptions.logger,
-            platform: 'javascript'
-        }, httpData = this._getHttpData();
-
-        if (httpData) {
-            baseData.request = httpData;
-        }
-
-        // HACK: delete `trimHeadFrames` to prevent from appearing in outbound payload
-        if (data.trimHeadFrames) delete data.trimHeadFrames;
-
-        data = objectMerge(baseData, data);
-
-        // Merge in the tags and extra separately since objectMerge doesn't handle a deep merge
-        data.tags = objectMerge(objectMerge({}, this._globalContext.tags), data.tags);
-        data.extra = objectMerge(objectMerge({}, this._globalContext.extra), data.extra);
-
-        // Send along our own collected metadata with extra
-        data.extra['session:duration'] = now() - this._startTime;
-
-        if (this._breadcrumbs && this._breadcrumbs.length > 0) {
-            // intentionally make shallow copy so that additions
-            // to breadcrumbs aren't accidentally sent in this request
-            data.breadcrumbs = {
-                values: [].slice.call(this._breadcrumbs, 0)
-            };
-        }
-
-        // If there are no tags/extra, strip the key from the payload alltogther.
-        if (isEmptyObject(data.tags)) delete data.tags;
-
-        if (this._globalContext.user) {
-            // sentry.interfaces.User
-            data.user = this._globalContext.user;
-        }
-
-        // Include the environment if it's defined in globalOptions
-        if (globalOptions.environment) data.environment = globalOptions.environment;
-
-        // Include the release if it's defined in globalOptions
-        if (globalOptions.release) data.release = globalOptions.release;
-
-        // Include server_name if it's defined in globalOptions
-        if (globalOptions.serverName) data.server_name = globalOptions.serverName;
-
-        if (isFunction(globalOptions.dataCallback)) {
-            data = globalOptions.dataCallback(data) || data;
-        }
-
-        // Why??????????
-        if (!data || isEmptyObject(data)) {
-            return;
-        }
-
-        // Check if the request should be filtered or not
-        if (isFunction(globalOptions.shouldSendCallback) && !globalOptions.shouldSendCallback(data)) {
-            return;
-        }
-
-        this._sendProcessedPayload(data);
-    },
-
-    _getUuid: function () {
-      return uuid4();
-    },
-
-    _sendProcessedPayload: function(data, callback) {
-        var self = this;
-        var globalOptions = this._globalOptions;
-
-        // Send along an event_id if not explicitly passed.
-        // This event_id can be used to reference the error within Sentry itself.
-        // Set lastEventId after we know the error should actually be sent
-        this._lastEventId = data.event_id || (data.event_id = this._getUuid());
-
-        // Try and clean up the packet before sending by truncating long values
-        data = this._trimPacket(data);
-
-        this._logDebug('debug', 'Raven about to send:', data);
-
-        if (!this.isSetup()) return;
-
-        var auth = {
-            sentry_version: '7',
-            sentry_client: 'raven-js/' + this.VERSION,
-            sentry_key: this._globalKey
-        };
-        if (this._globalSecret) {
-            auth.sentry_secret = this._globalSecret;
-        }
-
-        var exception = data.exception && data.exception.values[0];
-        this.captureBreadcrumb({
-            category: 'sentry',
-            message: exception
-                ? (exception.type ? exception.type + ': ' : '') + exception.value
-                : data.message,
-            event_id: data.event_id,
-            level: data.level || 'error' // presume error unless specified
-        });
-
-        var url = this._globalEndpoint;
-        (globalOptions.transport || this._makeRequest).call(this, {
-            url: url,
-            auth: auth,
-            data: data,
-            options: globalOptions,
-            onSuccess: function success() {
-                self._triggerEvent('success', {
-                    data: data,
-                    src: url
-                });
-                callback && callback();
-            },
-            onError: function failure(error) {
-                self._triggerEvent('failure', {
-                    data: data,
-                    src: url
-                });
-                error = error || new Error('Raven send failed (no additional details provided)');
-                callback && callback(error);
-            }
-        });
-    },
-
-    _makeRequest: function(opts) {
-        var request = new XMLHttpRequest();
-
-        // if browser doesn't support CORS (e.g. IE7), we are out of luck
-        var hasCORS =
-            'withCredentials' in request ||
-            typeof XDomainRequest !== 'undefined';
-
-        if (!hasCORS) return;
-
-        var url = opts.url;
-        function handler() {
-            if (request.status === 200) {
-                if (opts.onSuccess) {
-                    opts.onSuccess();
-                }
-            } else if (opts.onError) {
-                opts.onError(new Error('Sentry error code: ' + request.status));
-            }
-        }
-
-        if ('withCredentials' in request) {
-            request.onreadystatechange = function () {
-                if (request.readyState !== 4) {
-                    return;
-                }
-                handler();
-            };
-        } else {
-            request = new XDomainRequest();
-            // xdomainrequest cannot go http -> https (or vice versa),
-            // so always use protocol relative
-            url = url.replace(/^https?:/, '');
-
-            // onreadystatechange not supported by XDomainRequest
-            request.onload = handler;
-        }
-
-        // NOTE: auth is intentionally sent as part of query string (NOT as custom
-        //       HTTP header) so as to avoid preflight CORS requests
-        request.open('POST', url + '?' + urlencode(opts.auth));
-        request.send(stringify(opts.data));
-    },
-
-    _logDebug: function(level) {
-        if (this._originalConsoleMethods[level] && this.debug) {
-            // In IE<10 console methods do not have their own 'apply' method
-            Function.prototype.apply.call(
-                this._originalConsoleMethods[level],
-                this._originalConsole,
-                [].slice.call(arguments, 1)
-            );
-        }
-    },
-
-    _mergeContext: function(key, context) {
-        if (isUndefined(context)) {
-            delete this._globalContext[key];
-        } else {
-            this._globalContext[key] = objectMerge(this._globalContext[key] || {}, context);
-        }
-    }
-};
-
-/*------------------------------------------------
- * utils
- *
- * conditionally exported for test via Raven.utils
- =================================================
- */
-var objectPrototype = Object.prototype;
-
-function isUndefined(what) {
-    return what === void 0;
-}
-
-function isFunction(what) {
-    return typeof what === 'function';
-}
-
-function isString(what) {
-    return objectPrototype.toString.call(what) === '[object String]';
-}
-
-function isObject(what) {
-    return typeof what === 'object' && what !== null;
-}
-
-function isEmptyObject(what) {
-    for (var _ in what) return false;  // eslint-disable-line guard-for-in, no-unused-vars
-    return true;
-}
-
-// Sorta yanked from https://github.com/joyent/node/blob/aa3b4b4/lib/util.js#L560
-// with some tiny modifications
-function isError(what) {
-    var toString = objectPrototype.toString.call(what);
-    return isObject(what) &&
-        toString === '[object Error]' ||
-        toString === '[object Exception]' || // Firefox NS_ERROR_FAILURE Exceptions
-        what instanceof Error;
-}
-
-function each(obj, callback) {
-    var i, j;
-
-    if (isUndefined(obj.length)) {
-        for (i in obj) {
-            if (hasKey(obj, i)) {
-                callback.call(null, i, obj[i]);
-            }
-        }
-    } else {
-        j = obj.length;
-        if (j) {
-            for (i = 0; i < j; i++) {
-                callback.call(null, i, obj[i]);
-            }
-        }
-    }
-}
-
-function objectMerge(obj1, obj2) {
-    if (!obj2) {
-        return obj1;
-    }
-    each(obj2, function(key, value){
-        obj1[key] = value;
-    });
-    return obj1;
-}
-
-function truncate(str, max) {
-    return !max || str.length <= max ? str : str.substr(0, max) + '\u2026';
-}
-
-/**
- * hasKey, a better form of hasOwnProperty
- * Example: hasKey(MainHostObject, property) === true/false
- *
- * @param {Object} host object to check property
- * @param {string} key to check
- */
-function hasKey(object, key) {
-    return objectPrototype.hasOwnProperty.call(object, key);
-}
-
-function joinRegExp(patterns) {
-    // Combine an array of regular expressions and strings into one large regexp
-    // Be mad.
-    var sources = [],
-        i = 0, len = patterns.length,
-        pattern;
-
-    for (; i < len; i++) {
-        pattern = patterns[i];
-        if (isString(pattern)) {
-            // If it's a string, we need to escape it
-            // Taken from: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions
-            sources.push(pattern.replace(/([.*+?^=!:${}()|\[\]\/\\])/g, '\\$1'));
-        } else if (pattern && pattern.source) {
-            // If it's a regexp already, we want to extract the source
-            sources.push(pattern.source);
-        }
-        // Intentionally skip other cases
-    }
-    return new RegExp(sources.join('|'), 'i');
-}
-
-function urlencode(o) {
-    var pairs = [];
-    each(o, function(key, value) {
-        pairs.push(encodeURIComponent(key) + '=' + encodeURIComponent(value));
-    });
-    return pairs.join('&');
-}
-
-// borrowed from https://tools.ietf.org/html/rfc3986#appendix-B
-// intentionally using regex and not <a/> href parsing trick because React Native and other
-// environments where DOM might not be available
-function parseUrl(url) {
-    var match = url.match(/^(([^:\/?#]+):)?(\/\/([^\/?#]*))?([^?#]*)(\?([^#]*))?(#(.*))?$/);
-    if (!match) return {};
-
-    // coerce to undefined values to empty string so we don't get 'undefined'
-    var query = match[6] || '';
-    var fragment = match[8] || '';
-    return {
-        protocol: match[2],
-        host: match[4],
-        path: match[5],
-        relative: match[5] + query + fragment // everything minus origin
-    };
-}
-function uuid4() {
-    var crypto = _window.crypto || _window.msCrypto;
-
-    if (!isUndefined(crypto) && crypto.getRandomValues) {
-        // Use window.crypto API if available
-        var arr = new Uint16Array(8);
-        crypto.getRandomValues(arr);
-
-        // set 4 in byte 7
-        arr[3] = arr[3] & 0xFFF | 0x4000;
-        // set 2 most significant bits of byte 9 to '10'
-        arr[4] = arr[4] & 0x3FFF | 0x8000;
-
-        var pad = function(num) {
-            var v = num.toString(16);
-            while (v.length < 4) {
-                v = '0' + v;
-            }
-            return v;
-        };
-
-        return pad(arr[0]) + pad(arr[1]) + pad(arr[2]) + pad(arr[3]) + pad(arr[4]) +
-        pad(arr[5]) + pad(arr[6]) + pad(arr[7]);
-    } else {
-        // http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#2117523
-        return 'xxxxxxxxxxxx4xxxyxxxxxxxxxxxxxxx'.replace(/[xy]/g, function(c) {
-            var r = Math.random()*16|0,
-                v = c === 'x' ? r : r&0x3|0x8;
-            return v.toString(16);
-        });
-    }
-}
-
-/**
- * Given a child DOM element, returns a query-selector statement describing that
- * and its ancestors
- * e.g. [HTMLElement] => body > div > input#foo.btn[name=baz]
- * @param elem
- * @returns {string}
- */
-function htmlTreeAsString(elem) {
-    /* eslint no-extra-parens:0*/
-    var MAX_TRAVERSE_HEIGHT = 5,
-        MAX_OUTPUT_LEN = 80,
-        out = [],
-        height = 0,
-        len = 0,
-        separator = ' > ',
-        sepLength = separator.length,
-        nextStr;
-
-    while (elem && height++ < MAX_TRAVERSE_HEIGHT) {
-
-        nextStr = htmlElementAsString(elem);
-        // bail out if
-        // - nextStr is the 'html' element
-        // - the length of the string that would be created exceeds MAX_OUTPUT_LEN
-        //   (ignore this limit if we are on the first iteration)
-        if (nextStr === 'html' || height > 1 && len + (out.length * sepLength) + nextStr.length >= MAX_OUTPUT_LEN) {
-            break;
-        }
-
-        out.push(nextStr);
-
-        len += nextStr.length;
-        elem = elem.parentNode;
-    }
-
-    return out.reverse().join(separator);
-}
-
-/**
- * Returns a simple, query-selector representation of a DOM element
- * e.g. [HTMLElement] => input#foo.btn[name=baz]
- * @param HTMLElement
- * @returns {string}
- */
-function htmlElementAsString(elem) {
-    var out = [],
-        className,
-        classes,
-        key,
-        attr,
-        i;
-
-    if (!elem || !elem.tagName) {
-        return '';
-    }
-
-    out.push(elem.tagName.toLowerCase());
-    if (elem.id) {
-        out.push('#' + elem.id);
-    }
-
-    className = elem.className;
-    if (className && isString(className)) {
-        classes = className.split(' ');
-        for (i = 0; i < classes.length; i++) {
-            out.push('.' + classes[i]);
-        }
-    }
-    var attrWhitelist = ['type', 'name', 'title', 'alt'];
-    for (i = 0; i < attrWhitelist.length; i++) {
-        key = attrWhitelist[i];
-        attr = elem.getAttribute(key);
-        if (attr) {
-            out.push('[' + key + '="' + attr + '"]');
-        }
-    }
-    return out.join('');
-}
-
-/**
- * Polyfill a method
- * @param obj object e.g. `document`
- * @param name method name present on object e.g. `addEventListener`
- * @param replacement replacement function
- * @param track {optional} record instrumentation to an array
- */
-function fill(obj, name, replacement, track) {
-    var orig = obj[name];
-    obj[name] = replacement(orig);
-    if (track) {
-        track.push([obj, name, orig]);
-    }
-}
-
-if (typeof __DEV__ !== 'undefined' && __DEV__) {
-    Raven.utils = {
-        isUndefined: isUndefined,
-        isFunction: isFunction,
-        isString: isString,
-        isObject: isObject,
-        isEmptyObject: isEmptyObject,
-        isError: isError,
-        each: each,
-        objectMerge: objectMerge,
-        truncate: truncate,
-        hasKey: hasKey,
-        joinRegExp: joinRegExp,
-        urlencode: urlencode,
-        uuid4: uuid4,
-        htmlTreeAsString: htmlTreeAsString,
-        htmlElementAsString: htmlElementAsString,
-        parseUrl: parseUrl,
-        fill: fill
-    };
-};
-
-// Deprecations
-Raven.prototype.setUser = Raven.prototype.setUserContext;
-Raven.prototype.setReleaseContext = Raven.prototype.setRelease;
-
-module.exports = Raven;
-
-}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
-},{"1":1,"2":2,"3":3,"6":6}],5:[function(_dereq_,module,exports){
-(function (global){
-/**
- * Enforces a single instance of the Raven client, and the
- * main entry point for Raven. If you are a consumer of the
- * Raven library, you SHOULD load this file (vs raven.js).
- **/
-
-'use strict';
-
-var RavenConstructor = _dereq_(4);
-
-// This is to be defensive in environments where window does not exist (see https://github.com/getsentry/raven-js/pull/785)
-var _window = typeof window !== 'undefined' ? window
-            : typeof global !== 'undefined' ? global
-            : typeof self !== 'undefined' ? self
-            : {};
-var _Raven = _window.Raven;
-
-var Raven = new RavenConstructor();
-
-/*
- * Allow multiple versions of Raven to be installed.
- * Strip Raven from the global context and returns the instance.
- *
- * @return {Raven}
- */
-Raven.noConflict = function () {
-	_window.Raven = _Raven;
-	return Raven;
-};
-
-Raven.afterLoad();
-
-module.exports = Raven;
-
-}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
-},{"4":4}],6:[function(_dereq_,module,exports){
-(function (global){
-'use strict';
-
-/*
- TraceKit - Cross brower stack traces - github.com/occ/TraceKit
- MIT license
-*/
-
-var TraceKit = {
-    collectWindowErrors: true,
-    debug: false
-};
-
-// This is to be defensive in environments where window does not exist (see https://github.com/getsentry/raven-js/pull/785)
-var _window = typeof window !== 'undefined' ? window
-            : typeof global !== 'undefined' ? global
-            : typeof self !== 'undefined' ? self
-            : {};
-
-// global reference to slice
-var _slice = [].slice;
-var UNKNOWN_FUNCTION = '?';
-
-// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Error#Error_types
-var ERROR_TYPES_RE = /^(?:Uncaught (?:exception: )?)?((?:Eval|Internal|Range|Reference|Syntax|Type|URI)Error): ?(.*)$/;
-
-function getLocationHref() {
-    if (typeof document === 'undefined')
-        return '';
-
-    return document.location.href;
-}
-
-/**
- * TraceKit.report: cross-browser processing of unhandled exceptions
- *
- * Syntax:
- *   TraceKit.report.subscribe(function(stackInfo) { ... })
- *   TraceKit.report.unsubscribe(function(stackInfo) { ... })
- *   TraceKit.report(exception)
- *   try { ...code... } catch(ex) { TraceKit.report(ex); }
- *
- * Supports:
- *   - Firefox: full stack trace with line numbers, plus column number
- *              on top frame; column number is not guaranteed
- *   - Opera:   full stack trace with line and column numbers
- *   - Chrome:  full stack trace with line and column numbers
- *   - Safari:  line and column number for the top frame only; some frames
- *              may be missing, and column number is not guaranteed
- *   - IE:      line and column number for the top frame only; some frames
- *              may be missing, and column number is not guaranteed
- *
- * In theory, TraceKit should work on all of the following versions:
- *   - IE5.5+ (only 8.0 tested)
- *   - Firefox 0.9+ (only 3.5+ tested)
- *   - Opera 7+ (only 10.50 tested; versions 9 and earlier may require
- *     Exceptions Have Stacktrace to be enabled in opera:config)
- *   - Safari 3+ (only 4+ tested)
- *   - Chrome 1+ (only 5+ tested)
- *   - Konqueror 3.5+ (untested)
- *
- * Requires TraceKit.computeStackTrace.
- *
- * Tries to catch all unhandled exceptions and report them to the
- * subscribed handlers. Please note that TraceKit.report will rethrow the
- * exception. This is REQUIRED in order to get a useful stack trace in IE.
- * If the exception does not reach the top of the browser, you will only
- * get a stack trace from the point where TraceKit.report was called.
- *
- * Handlers receive a stackInfo object as described in the
- * TraceKit.computeStackTrace docs.
- */
-TraceKit.report = (function reportModuleWrapper() {
-    var handlers = [],
-        lastArgs = null,
-        lastException = null,
-        lastExceptionStack = null;
-
-    /**
-     * Add a crash handler.
-     * @param {Function} handler
-     */
-    function subscribe(handler) {
-        installGlobalHandler();
-        handlers.push(handler);
-    }
-
-    /**
-     * Remove a crash handler.
-     * @param {Function} handler
-     */
-    function unsubscribe(handler) {
-        for (var i = handlers.length - 1; i >= 0; --i) {
-            if (handlers[i] === handler) {
-                handlers.splice(i, 1);
-            }
-        }
-    }
-
-    /**
-     * Remove all crash handlers.
-     */
-    function unsubscribeAll() {
-        uninstallGlobalHandler();
-        handlers = [];
-    }
-
-    /**
-     * Dispatch stack information to all handlers.
-     * @param {Object.<string, *>} stack
-     */
-    function notifyHandlers(stack, isWindowError) {
-        var exception = null;
-        if (isWindowError && !TraceKit.collectWindowErrors) {
-          return;
-        }
-        for (var i in handlers) {
-            if (handlers.hasOwnProperty(i)) {
-                try {
-                    handlers[i].apply(null, [stack].concat(_slice.call(arguments, 2)));
-                } catch (inner) {
-                    exception = inner;
-                }
-            }
-        }
-
-        if (exception) {
-            throw exception;
-        }
-    }
-
-    var _oldOnerrorHandler, _onErrorHandlerInstalled;
-
-    /**
-     * Ensures all global unhandled exceptions are recorded.
-     * Supported by Gecko and IE.
-     * @param {string} message Error message.
-     * @param {string} url URL of script that generated the exception.
-     * @param {(number|string)} lineNo The line number at which the error
-     * occurred.
-     * @param {?(number|string)} colNo The column number at which the error
-     * occurred.
-     * @param {?Error} ex The actual Error object.
-     */
-    function traceKitWindowOnError(message, url, lineNo, colNo, ex) {
-        var stack = null;
-
-        if (lastExceptionStack) {
-            TraceKit.computeStackTrace.augmentStackTraceWithInitialElement(lastExceptionStack, url, lineNo, message);
-            processLastException();
-        } else if (ex) {
-            // New chrome and blink send along a real error object
-            // Let's just report that like a normal error.
-            // See: https://mikewest.org/2013/08/debugging-runtime-errors-with-window-onerror
-            stack = TraceKit.computeStackTrace(ex);
-            notifyHandlers(stack, true);
-        } else {
-            var location = {
-                'url': url,
-                'line': lineNo,
-                'column': colNo
-            };
-
-            var name = undefined;
-            var msg = message; // must be new var or will modify original `arguments`
-            var groups;
-            if ({}.toString.call(message) === '[object String]') {
-                var groups = message.match(ERROR_TYPES_RE);
-                if (groups) {
-                    name = groups[1];
-                    msg = groups[2];
-                }
-            }
-
-            location.func = UNKNOWN_FUNCTION;
-
-            stack = {
-                'name': name,
-                'message': msg,
-                'url': getLocationHref(),
-                'stack': [location]
-            };
-            notifyHandlers(stack, true);
-        }
-
-        if (_oldOnerrorHandler) {
-            return _oldOnerrorHandler.apply(this, arguments);
-        }
-
-        return false;
-    }
-
-    function installGlobalHandler ()
-    {
-        if (_onErrorHandlerInstalled) {
-            return;
-        }
-        _oldOnerrorHandler = _window.onerror;
-        _window.onerror = traceKitWindowOnError;
-        _onErrorHandlerInstalled = true;
-    }
-
-    function uninstallGlobalHandler ()
-    {
-        if (!_onErrorHandlerInstalled) {
-            return;
-        }
-        _window.onerror = _oldOnerrorHandler;
-        _onErrorHandlerInstalled = false;
-        _oldOnerrorHandler = undefined;
-    }
-
-    function processLastException() {
-        var _lastExceptionStack = lastExceptionStack,
-            _lastArgs = lastArgs;
-        lastArgs = null;
-        lastExceptionStack = null;
-        lastException = null;
-        notifyHandlers.apply(null, [_lastExceptionStack, false].concat(_lastArgs));
-    }
-
-    /**
-     * Reports an unhandled Error to TraceKit.
-     * @param {Error} ex
-     * @param {?boolean} rethrow If false, do not re-throw the exception.
-     * Only used for window.onerror to not cause an infinite loop of
-     * rethrowing.
-     */
-    function report(ex, rethrow) {
-        var args = _slice.call(arguments, 1);
-        if (lastExceptionStack) {
-            if (lastException === ex) {
-                return; // already caught by an inner catch block, ignore
-            } else {
-              processLastException();
-            }
-        }
-
-        var stack = TraceKit.computeStackTrace(ex);
-        lastExceptionStack = stack;
-        lastException = ex;
-        lastArgs = args;
-
-        // If the stack trace is incomplete, wait for 2 seconds for
-        // slow slow IE to see if onerror occurs or not before reporting
-        // this exception; otherwise, we will end up with an incomplete
-        // stack trace
-        setTimeout(function () {
-            if (lastException === ex) {
-                processLastException();
-            }
-        }, (stack.incomplete ? 2000 : 0));
-
-        if (rethrow !== false) {
-            throw ex; // re-throw to propagate to the top level (and cause window.onerror)
-        }
-    }
-
-    report.subscribe = subscribe;
-    report.unsubscribe = unsubscribe;
-    report.uninstall = unsubscribeAll;
-    return report;
-}());
-
-/**
- * TraceKit.computeStackTrace: cross-browser stack traces in JavaScript
- *
- * Syntax:
- *   s = TraceKit.computeStackTrace(exception) // consider using TraceKit.report instead (see below)
- * Returns:
- *   s.name              - exception name
- *   s.message           - exception message
- *   s.stack[i].url      - JavaScript or HTML file URL
- *   s.stack[i].func     - function name, or empty for anonymous functions (if guessing did not work)
- *   s.stack[i].args     - arguments passed to the function, if known
- *   s.stack[i].line     - line number, if known
- *   s.stack[i].column   - column number, if known
- *
- * Supports:
- *   - Firefox:  full stack trace with line numbers and unreliable column
- *               number on top frame
- *   - Opera 10: full stack trace with line and column numbers
- *   - Opera 9-: full stack trace with line numbers
- *   - Chrome:   full stack trace with line and column numbers
- *   - Safari:   line and column number for the topmost stacktrace element
- *               only
- *   - IE:       no line numbers whatsoever
- *
- * Tries to guess names of anonymous functions by looking for assignments
- * in the source code. In IE and Safari, we have to guess source file names
- * by searching for function bodies inside all page scripts. This will not
- * work for scripts that are loaded cross-domain.
- * Here be dragons: some function names may be guessed incorrectly, and
- * duplicate functions may be mismatched.
- *
- * TraceKit.computeStackTrace should only be used for tracing purposes.
- * Logging of unhandled exceptions should be done with TraceKit.report,
- * which builds on top of TraceKit.computeStackTrace and provides better
- * IE support by utilizing the window.onerror event to retrieve information
- * about the top of the stack.
- *
- * Note: In IE and Safari, no stack trace is recorded on the Error object,
- * so computeStackTrace instead walks its *own* chain of callers.
- * This means that:
- *  * in Safari, some methods may be missing from the stack trace;
- *  * in IE, the topmost function in the stack trace will always be the
- *    caller of computeStackTrace.
- *
- * This is okay for tracing (because you are likely to be calling
- * computeStackTrace from the function you want to be the topmost element
- * of the stack trace anyway), but not okay for logging unhandled
- * exceptions (because your catch block will likely be far away from the
- * inner function that actually caused the exception).
- *
- */
-TraceKit.computeStackTrace = (function computeStackTraceWrapper() {
-    /**
-     * Escapes special characters, except for whitespace, in a string to be
-     * used inside a regular expression as a string literal.
-     * @param {string} text The string.
-     * @return {string} The escaped string literal.
-     */
-    function escapeRegExp(text) {
-        return text.replace(/[\-\[\]{}()*+?.,\\\^$|#]/g, '\\$&');
-    }
-
-    /**
-     * Escapes special characters in a string to be used inside a regular
-     * expression as a string literal. Also ensures that HTML entities will
-     * be matched the same as their literal friends.
-     * @param {string} body The string.
-     * @return {string} The escaped string.
-     */
-    function escapeCodeAsRegExpForMatchingInsideHTML(body) {
-        return escapeRegExp(body).replace('<', '(?:<|&lt;)').replace('>', '(?:>|&gt;)').replace('&', '(?:&|&amp;)').replace('"', '(?:"|&quot;)').replace(/\s+/g, '\\s+');
-    }
-
-    // Contents of Exception in various browsers.
-    //
-    // SAFARI:
-    // ex.message = Can't find variable: qq
-    // ex.line = 59
-    // ex.sourceId = 580238192
-    // ex.sourceURL = http://...
-    // ex.expressionBeginOffset = 96
-    // ex.expressionCaretOffset = 98
-    // ex.expressionEndOffset = 98
-    // ex.name = ReferenceError
-    //
-    // FIREFOX:
-    // ex.message = qq is not defined
-    // ex.fileName = http://...
-    // ex.lineNumber = 59
-    // ex.columnNumber = 69
-    // ex.stack = ...stack trace... (see the example below)
-    // ex.name = ReferenceError
-    //
-    // CHROME:
-    // ex.message = qq is not defined
-    // ex.name = ReferenceError
-    // ex.type = not_defined
-    // ex.arguments = ['aa']
-    // ex.stack = ...stack trace...
-    //
-    // INTERNET EXPLORER:
-    // ex.message = ...
-    // ex.name = ReferenceError
-    //
-    // OPERA:
-    // ex.message = ...message... (see the example below)
-    // ex.name = ReferenceError
-    // ex.opera#sourceloc = 11  (pretty much useless, duplicates the info in ex.message)
-    // ex.stacktrace = n/a; see 'opera:config#UserPrefs|Exceptions Have Stacktrace'
-
-    /**
-     * Computes stack trace information from the stack property.
-     * Chrome and Gecko use this property.
-     * @param {Error} ex
-     * @return {?Object.<string, *>} Stack trace information.
-     */
-    function computeStackTraceFromStackProp(ex) {
-        if (typeof ex.stack === 'undefined' || !ex.stack) return;
-
-        var chrome = /^\s*at (.*?) ?\(((?:file|https?|blob|chrome-extension|native|eval|<anonymous>).*?)(?::(\d+))?(?::(\d+))?\)?\s*$/i,
-            gecko = /^\s*(.*?)(?:\((.*?)\))?(?:^|@)((?:file|https?|blob|chrome|\[native).*?)(?::(\d+))?(?::(\d+))?\s*$/i,
-            winjs = /^\s*at (?:((?:\[object object\])?.+) )?\(?((?:file|ms-appx|https?|blob):.*?):(\d+)(?::(\d+))?\)?\s*$/i,
-            lines = ex.stack.split('\n'),
-            stack = [],
-            parts,
-            element,
-            reference = /^(.*) is undefined$/.exec(ex.message);
-
-        for (var i = 0, j = lines.length; i < j; ++i) {
-            if ((parts = chrome.exec(lines[i]))) {
-                var isNative = parts[2] && parts[2].indexOf('native') !== -1;
-                element = {
-                    'url': !isNative ? parts[2] : null,
-                    'func': parts[1] || UNKNOWN_FUNCTION,
-                    'args': isNative ? [parts[2]] : [],
-                    'line': parts[3] ? +parts[3] : null,
-                    'column': parts[4] ? +parts[4] : null
-                };
-            } else if ( parts = winjs.exec(lines[i]) ) {
-                element = {
-                    'url': parts[2],
-                    'func': parts[1] || UNKNOWN_FUNCTION,
-                    'args': [],
-                    'line': +parts[3],
-                    'column': parts[4] ? +parts[4] : null
-                };
-            } else if ((parts = gecko.exec(lines[i]))) {
-                element = {
-                    'url': parts[3],
-                    'func': parts[1] || UNKNOWN_FUNCTION,
-                    'args': parts[2] ? parts[2].split(',') : [],
-                    'line': parts[4] ? +parts[4] : null,
-                    'column': parts[5] ? +parts[5] : null
-                };
-            } else {
-                continue;
-            }
-
-            if (!element.func && element.line) {
-                element.func = UNKNOWN_FUNCTION;
-            }
-
-            stack.push(element);
-        }
-
-        if (!stack.length) {
-            return null;
-        }
-
-        if (!stack[0].column && typeof ex.columnNumber !== 'undefined') {
-            // FireFox uses this awesome columnNumber property for its top frame
-            // Also note, Firefox's column number is 0-based and everything else expects 1-based,
-            // so adding 1
-            stack[0].column = ex.columnNumber + 1;
-        }
-
-        return {
-            'name': ex.name,
-            'message': ex.message,
-            'url': getLocationHref(),
-            'stack': stack
-        };
-    }
-
-    /**
-     * Adds information about the first frame to incomplete stack traces.
-     * Safari and IE require this to get complete data on the first frame.
-     * @param {Object.<string, *>} stackInfo Stack trace information from
-     * one of the compute* methods.
-     * @param {string} url The URL of the script that caused an error.
-     * @param {(number|string)} lineNo The line number of the script that
-     * caused an error.
-     * @param {string=} message The error generated by the browser, which
-     * hopefully contains the name of the object that caused the error.
-     * @return {boolean} Whether or not the stack information was
-     * augmented.
-     */
-    function augmentStackTraceWithInitialElement(stackInfo, url, lineNo, message) {
-        var initial = {
-            'url': url,
-            'line': lineNo
-        };
-
-        if (initial.url && initial.line) {
-            stackInfo.incomplete = false;
-
-            if (!initial.func) {
-                initial.func = UNKNOWN_FUNCTION;
-            }
-
-            if (stackInfo.stack.length > 0) {
-                if (stackInfo.stack[0].url === initial.url) {
-                    if (stackInfo.stack[0].line === initial.line) {
-                        return false; // already in stack trace
-                    } else if (!stackInfo.stack[0].line && stackInfo.stack[0].func === initial.func) {
-                        stackInfo.stack[0].line = initial.line;
-                        return false;
-                    }
-                }
-            }
-
-            stackInfo.stack.unshift(initial);
-            stackInfo.partial = true;
-            return true;
-        } else {
-            stackInfo.incomplete = true;
-        }
-
-        return false;
-    }
-
-    /**
-     * Computes stack trace information by walking the arguments.caller
-     * chain at the time the exception occurred. This will cause earlier
-     * frames to be missed but is the only way to get any stack trace in
-     * Safari and IE. The top frame is restored by
-     * {@link augmentStackTraceWithInitialElement}.
-     * @param {Error} ex
-     * @return {?Object.<string, *>} Stack trace information.
-     */
-    function computeStackTraceByWalkingCallerChain(ex, depth) {
-        var functionName = /function\s+([_$a-zA-Z\xA0-\uFFFF][_$a-zA-Z0-9\xA0-\uFFFF]*)?\s*\(/i,
-            stack = [],
-            funcs = {},
-            recursion = false,
-            parts,
-            item,
-            source;
-
-        for (var curr = computeStackTraceByWalkingCallerChain.caller; curr && !recursion; curr = curr.caller) {
-            if (curr === computeStackTrace || curr === TraceKit.report) {
-                // console.log('skipping internal function');
-                continue;
-            }
-
-            item = {
-                'url': null,
-                'func': UNKNOWN_FUNCTION,
-                'line': null,
-                'column': null
-            };
-
-            if (curr.name) {
-                item.func = curr.name;
-            } else if ((parts = functionName.exec(curr.toString()))) {
-                item.func = parts[1];
-            }
-
-            if (typeof item.func === 'undefined') {
-              try {
-                item.func = parts.input.substring(0, parts.input.indexOf('{'));
-              } catch (e) { }
-            }
-
-            if (funcs['' + curr]) {
-                recursion = true;
-            }else{
-                funcs['' + curr] = true;
-            }
-
-            stack.push(item);
-        }
-
-        if (depth) {
-            // console.log('depth is ' + depth);
-            // console.log('stack is ' + stack.length);
-            stack.splice(0, depth);
-        }
-
-        var result = {
-            'name': ex.name,
-            'message': ex.message,
-            'url': getLocationHref(),
-            'stack': stack
-        };
-        augmentStackTraceWithInitialElement(result, ex.sourceURL || ex.fileName, ex.line || ex.lineNumber, ex.message || ex.description);
-        return result;
-    }
-
-    /**
-     * Computes a stack trace for an exception.
-     * @param {Error} ex
-     * @param {(string|number)=} depth
-     */
-    function computeStackTrace(ex, depth) {
-        var stack = null;
-        depth = (depth == null ? 0 : +depth);
-
-        try {
-            stack = computeStackTraceFromStackProp(ex);
-            if (stack) {
-                return stack;
-            }
-        } catch (e) {
-            if (TraceKit.debug) {
-                throw e;
-            }
-        }
-
-        try {
-            stack = computeStackTraceByWalkingCallerChain(ex, depth + 1);
-            if (stack) {
-                return stack;
-            }
-        } catch (e) {
-            if (TraceKit.debug) {
-                throw e;
-            }
-        }
-
-        return {
-            'name': ex.name,
-            'message': ex.message,
-            'url': getLocationHref()
-        };
-    }
-
-    computeStackTrace.augmentStackTraceWithInitialElement = augmentStackTraceWithInitialElement;
-    computeStackTrace.computeStackTraceFromStackProp = computeStackTraceFromStackProp;
-
-    return computeStackTrace;
-}());
-
-module.exports = TraceKit;
-
-}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
-},{}]},{},[5])(5)
-});
diff --git a/yarn.lock b/yarn.lock
index f254668646c54..46f528f0bec5b 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -2730,7 +2730,7 @@ json-stable-stringify@^1.0.0, json-stable-stringify@^1.0.1:
   dependencies:
     jsonify "~0.0.0"
 
-json-stringify-safe@~5.0.1:
+json-stringify-safe@^5.0.1, json-stringify-safe@~5.0.1:
   version "5.0.1"
   resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb"
 
@@ -3604,6 +3604,12 @@ raphael@^2.2.7:
   dependencies:
     eve-raphael "0.5.0"
 
+raven-js@^3.14.0:
+  version "3.14.0"
+  resolved "https://registry.yarnpkg.com/raven-js/-/raven-js-3.14.0.tgz#94dda81d975fdc4a42f193db437cf70021d654e0"
+  dependencies:
+    json-stringify-safe "^5.0.1"
+
 raw-body@~2.2.0:
   version "2.2.0"
   resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.2.0.tgz#994976cf6a5096a41162840492f0bdc5d6e7fb96"
-- 
GitLab


From 13b60eb75b99abb23f41c0d899d6e40eefa641cc Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 13 Apr 2017 17:17:41 +0100
Subject: [PATCH 044/363] [ci skip] Index and singleton improvements with some
 more unit, more to come

---
 app/assets/javascripts/raven/index.js        |  6 ++-
 app/assets/javascripts/raven/raven_config.js | 25 +++++-----
 spec/helpers/sentry_helper_spec.rb           | 15 ++++--
 spec/javascripts/raven/index_spec.js         | 42 ++++++++++++++---
 spec/javascripts/raven/raven_config_spec.js  | 13 +-----
 spec/javascripts/spec_helper.js              | 49 --------------------
 6 files changed, 66 insertions(+), 84 deletions(-)
 delete mode 100644 spec/javascripts/spec_helper.js

diff --git a/app/assets/javascripts/raven/index.js b/app/assets/javascripts/raven/index.js
index 6cc81248e6b23..373f9f29c7980 100644
--- a/app/assets/javascripts/raven/index.js
+++ b/app/assets/javascripts/raven/index.js
@@ -1,10 +1,12 @@
 import RavenConfig from './raven_config';
 
-RavenConfig.init({
+const index = RavenConfig.init.bind(RavenConfig, {
   sentryDsn: gon.sentry_dsn,
   currentUserId: gon.current_user_id,
   whitelistUrls: [gon.gitlab_url],
   isProduction: gon.is_production,
 });
 
-export default RavenConfig;
+index();
+
+export default index;
diff --git a/app/assets/javascripts/raven/raven_config.js b/app/assets/javascripts/raven/raven_config.js
index 5510dd2752d58..1157a10d96f45 100644
--- a/app/assets/javascripts/raven/raven_config.js
+++ b/app/assets/javascripts/raven/raven_config.js
@@ -1,32 +1,35 @@
 import Raven from 'raven-js';
+import $ from 'jquery';
 
-class RavenConfig {
-  static init(options = {}) {
+const RavenConfig = {
+  init(options = {}) {
     this.options = options;
 
     this.configure();
     this.bindRavenErrors();
     if (this.options.currentUserId) this.setUser();
-  }
 
-  static configure() {
+    return this;
+  },
+
+  configure() {
     Raven.config(this.options.sentryDsn, {
       whitelistUrls: this.options.whitelistUrls,
       environment: this.options.isProduction ? 'production' : 'development',
     }).install();
-  }
+  },
 
-  static setUser() {
+  setUser() {
     Raven.setUserContext({
       id: this.options.currentUserId,
     });
-  }
+  },
 
-  static bindRavenErrors() {
+  bindRavenErrors() {
     $(document).on('ajaxError.raven', this.handleRavenErrors);
-  }
+  },
 
-  static handleRavenErrors(event, req, config, err) {
+  handleRavenErrors(event, req, config, err) {
     const error = err || req.statusText;
 
     Raven.captureMessage(error, {
@@ -40,7 +43,7 @@ class RavenConfig {
         event,
       },
     });
-  }
+  },
 }
 
 export default RavenConfig;
diff --git a/spec/helpers/sentry_helper_spec.rb b/spec/helpers/sentry_helper_spec.rb
index 35ecf9355e165..ff218235cd1d3 100644
--- a/spec/helpers/sentry_helper_spec.rb
+++ b/spec/helpers/sentry_helper_spec.rb
@@ -3,13 +3,20 @@
 describe SentryHelper do
   describe '#sentry_dsn_public' do
     it 'returns nil if no sentry_dsn is set' do
-      allow(ApplicationSetting.current).to receive(:sentry_dsn).and_return(nil)
-      expect(helper.sentry_dsn_public).to eq(nil)
+      mock_sentry_dsn(nil)
+
+      expect(helper.sentry_dsn_public).to eq nil
     end
 
     it 'returns the uri string with no password if sentry_dsn is set' do
-      allow(ApplicationSetting.current).to receive(:sentry_dsn).and_return('https://test:dsn@host/path')
-      expect(helper.sentry_dsn_public).to eq('https://test@host/path')
+      mock_sentry_dsn('https://test:dsn@host/path')
+
+      expect(helper.sentry_dsn_public).to eq 'https://test@host/path'
     end
   end
+
+  def mock_sentry_dsn(value)
+    allow_message_expectations_on_nil
+    allow(ApplicationSetting.current).to receive(:sentry_dsn).and_return(value)
+  end
 end
diff --git a/spec/javascripts/raven/index_spec.js b/spec/javascripts/raven/index_spec.js
index 51e84a6dbeea4..efde37b1f8ee6 100644
--- a/spec/javascripts/raven/index_spec.js
+++ b/spec/javascripts/raven/index_spec.js
@@ -1,11 +1,41 @@
-import RavenConfig from '~/raven/index';
+import RavenConfig from '~/raven/raven_config';
+import index from '~/raven/index';
 
-describe('RavenConfig options', () => {
-  it('should set sentryDsn');
+fdescribe('RavenConfig options', () => {
+  let sentryDsn;
+  let currentUserId;
+  let gitlabUrl;
+  let isProduction;
+  let indexReturnValue;
 
-  it('should set currentUserId');
+  beforeEach(() => {
+    sentryDsn = 'sentryDsn';
+    currentUserId = 'currentUserId';
+    gitlabUrl = 'gitlabUrl';
+    isProduction = 'isProduction';
 
-  it('should set whitelistUrls');
+    window.gon = {
+      sentry_dsn: sentryDsn,
+      current_user_id: currentUserId,
+      gitlab_url: gitlabUrl,
+      is_production: isProduction,
+    };
 
-  it('should set isProduction');
+    spyOn(RavenConfig.init, 'bind');
+
+    indexReturnValue = index();
+  });
+
+  it('should init with .sentryDsn, .currentUserId, .whitelistUrls and .isProduction', () => {
+    expect(RavenConfig.init.bind).toHaveBeenCalledWith(RavenConfig, {
+      sentryDsn,
+      currentUserId,
+      whitelistUrls: [gitlabUrl],
+      isProduction,
+    });
+  });
+
+  it('should return RavenConfig', () => {
+    expect(indexReturnValue).toBe(RavenConfig);
+  });
 });
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index 2b63f56d71904..64cf63702bc2c 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -1,8 +1,7 @@
 import Raven from 'raven-js';
 import RavenConfig from '~/raven/raven_config';
-import ClassSpecHelper from '../helpers/class_spec_helper';
 
-fdescribe('RavenConfig', () => {
+describe('RavenConfig', () => {
   describe('init', () => {
     beforeEach(() => {
       spyOn(RavenConfig, 'configure');
@@ -10,8 +9,6 @@ fdescribe('RavenConfig', () => {
       spyOn(RavenConfig, 'setUser');
     });
 
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'init');
-
     describe('when called', () => {
       let options;
 
@@ -58,8 +55,6 @@ fdescribe('RavenConfig', () => {
   });
 
   describe('configure', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'configure');
-
     describe('when called', () => {
       let options;
       let raven;
@@ -112,24 +107,18 @@ fdescribe('RavenConfig', () => {
   });
 
   describe('setUser', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'setUser');
-
     describe('when called', () => {
       beforeEach(() => {});
     });
   });
 
   describe('bindRavenErrors', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'bindRavenErrors');
-
     describe('when called', () => {
       beforeEach(() => {});
     });
   });
 
   describe('handleRavenErrors', () => {
-    ClassSpecHelper.itShouldBeAStaticMethod(RavenConfig, 'handleRavenErrors');
-
     describe('when called', () => {
       beforeEach(() => {});
     });
diff --git a/spec/javascripts/spec_helper.js b/spec/javascripts/spec_helper.js
deleted file mode 100644
index ee6e6279ac9fc..0000000000000
--- a/spec/javascripts/spec_helper.js
+++ /dev/null
@@ -1,49 +0,0 @@
-/* eslint-disable space-before-function-paren */
-// PhantomJS (Teaspoons default driver) doesn't have support for
-// Function.prototype.bind, which has caused confusion.  Use this polyfill to
-// avoid the confusion.
-/*= require support/bind-poly */
-
-// You can require your own javascript files here. By default this will include
-// everything in application, however you may get better load performance if you
-// require the specific files that are being used in the spec that tests them.
-/*= require jquery */
-/*= require jquery.turbolinks */
-/*= require bootstrap */
-/*= require underscore */
-/*= require es6-promise.auto */
-
-// Teaspoon includes some support files, but you can use anything from your own
-// support path too.
-// require support/jasmine-jquery-1.7.0
-// require support/jasmine-jquery-2.0.0
-/*= require support/jasmine-jquery-2.1.0 */
-
-// require support/sinon
-// require support/your-support-file
-// Deferring execution
-// If you're using CommonJS, RequireJS or some other asynchronous library you can
-// defer execution. Call Teaspoon.execute() after everything has been loaded.
-// Simple example of a timeout:
-// Teaspoon.defer = true
-// setTimeout(Teaspoon.execute, 1000)
-// Matching files
-// By default Teaspoon will look for files that match
-// _spec.{js,js.es6}. Add a filename_spec.js file in your spec path
-// and it'll be included in the default suite automatically. If you want to
-// customize suites, check out the configuration in teaspoon_env.rb
-// Manifest
-// If you'd rather require your spec files manually (to control order for
-// instance) you can disable the suite matcher in the configuration and use this
-// file as a manifest.
-// For more information: http://github.com/modeset/teaspoon
-
-// set our fixtures path
-jasmine.getFixtures().fixturesPath = '/teaspoon/fixtures';
-jasmine.getJSONFixtures().fixturesPath = '/teaspoon/fixtures';
-
-// defined in ActionDispatch::TestRequest
-// see https://github.com/rails/rails/blob/v4.2.7.1/actionpack/lib/action_dispatch/testing/test_request.rb#L7
-window.gl = window.gl || {};
-window.gl.TEST_HOST = 'http://test.host';
-window.gon = window.gon || {};
-- 
GitLab


From 067361327bba0cb7a8b28190485699da7deb22bb Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 14 Apr 2017 21:09:16 +0100
Subject: [PATCH 045/363] Updated units

---
 app/assets/javascripts/raven/index.js        |  16 +-
 app/assets/javascripts/raven/raven_config.js |   4 +-
 spec/javascripts/raven/index_spec.js         |   6 +-
 spec/javascripts/raven/raven_config_spec.js  | 154 ++++++++++---------
 4 files changed, 96 insertions(+), 84 deletions(-)

diff --git a/app/assets/javascripts/raven/index.js b/app/assets/javascripts/raven/index.js
index 373f9f29c7980..3c5656040b9e1 100644
--- a/app/assets/javascripts/raven/index.js
+++ b/app/assets/javascripts/raven/index.js
@@ -1,11 +1,15 @@
 import RavenConfig from './raven_config';
 
-const index = RavenConfig.init.bind(RavenConfig, {
-  sentryDsn: gon.sentry_dsn,
-  currentUserId: gon.current_user_id,
-  whitelistUrls: [gon.gitlab_url],
-  isProduction: gon.is_production,
-});
+const index = function index() {
+  RavenConfig.init({
+    sentryDsn: gon.sentry_dsn,
+    currentUserId: gon.current_user_id,
+    whitelistUrls: [gon.gitlab_url],
+    isProduction: gon.is_production,
+  });
+
+  return RavenConfig;
+};
 
 index();
 
diff --git a/app/assets/javascripts/raven/raven_config.js b/app/assets/javascripts/raven/raven_config.js
index 1157a10d96f45..bb9be7cb196fb 100644
--- a/app/assets/javascripts/raven/raven_config.js
+++ b/app/assets/javascripts/raven/raven_config.js
@@ -8,8 +8,6 @@ const RavenConfig = {
     this.configure();
     this.bindRavenErrors();
     if (this.options.currentUserId) this.setUser();
-
-    return this;
   },
 
   configure() {
@@ -44,6 +42,6 @@ const RavenConfig = {
       },
     });
   },
-}
+};
 
 export default RavenConfig;
diff --git a/spec/javascripts/raven/index_spec.js b/spec/javascripts/raven/index_spec.js
index efde37b1f8ee6..85ec1de4e4efa 100644
--- a/spec/javascripts/raven/index_spec.js
+++ b/spec/javascripts/raven/index_spec.js
@@ -1,7 +1,7 @@
 import RavenConfig from '~/raven/raven_config';
 import index from '~/raven/index';
 
-fdescribe('RavenConfig options', () => {
+describe('RavenConfig options', () => {
   let sentryDsn;
   let currentUserId;
   let gitlabUrl;
@@ -21,13 +21,13 @@ fdescribe('RavenConfig options', () => {
       is_production: isProduction,
     };
 
-    spyOn(RavenConfig.init, 'bind');
+    spyOn(RavenConfig, 'init');
 
     indexReturnValue = index();
   });
 
   it('should init with .sentryDsn, .currentUserId, .whitelistUrls and .isProduction', () => {
-    expect(RavenConfig.init.bind).toHaveBeenCalledWith(RavenConfig, {
+    expect(RavenConfig.init).toHaveBeenCalledWith({
       sentryDsn,
       currentUserId,
       whitelistUrls: [gitlabUrl],
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index 64cf63702bc2c..3885cfde6bf80 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -1,47 +1,46 @@
+import $ from 'jquery';
 import Raven from 'raven-js';
 import RavenConfig from '~/raven/raven_config';
 
-describe('RavenConfig', () => {
+fdescribe('RavenConfig', () => {
   describe('init', () => {
+    let options;
+
     beforeEach(() => {
+      options = {
+        sentryDsn: '//sentryDsn',
+        ravenAssetUrl: '//ravenAssetUrl',
+        currentUserId: 1,
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      };
+
       spyOn(RavenConfig, 'configure');
       spyOn(RavenConfig, 'bindRavenErrors');
       spyOn(RavenConfig, 'setUser');
-    });
-
-    describe('when called', () => {
-      let options;
 
-      beforeEach(() => {
-        options = {
-          sentryDsn: '//sentryDsn',
-          ravenAssetUrl: '//ravenAssetUrl',
-          currentUserId: 1,
-          whitelistUrls: ['//gitlabUrl'],
-          isProduction: true,
-        };
-
-        RavenConfig.init(options);
-      });
+      RavenConfig.init(options);
+    });
 
-      it('should set the options property', () => {
-        expect(RavenConfig.options).toEqual(options);
-      });
+    it('should set the options property', () => {
+      expect(RavenConfig.options).toEqual(options);
+    });
 
-      it('should call the configure method', () => {
-        expect(RavenConfig.configure).toHaveBeenCalled();
-      });
+    it('should call the configure method', () => {
+      expect(RavenConfig.configure).toHaveBeenCalled();
+    });
 
-      it('should call the error bindings method', () => {
-        expect(RavenConfig.bindRavenErrors).toHaveBeenCalled();
-      });
+    it('should call the error bindings method', () => {
+      expect(RavenConfig.bindRavenErrors).toHaveBeenCalled();
+    });
 
-      it('should call setUser', () => {
-        expect(RavenConfig.setUser).toHaveBeenCalled();
-      });
+    it('should call setUser', () => {
+      expect(RavenConfig.setUser).toHaveBeenCalled();
     });
 
     it('should not call setUser if there is no current user ID', () => {
+      RavenConfig.setUser.calls.reset();
+
       RavenConfig.init({
         sentryDsn: '//sentryDsn',
         ravenAssetUrl: '//ravenAssetUrl',
@@ -55,72 +54,83 @@ describe('RavenConfig', () => {
   });
 
   describe('configure', () => {
-    describe('when called', () => {
-      let options;
-      let raven;
+    let options;
+    let raven;
 
-      beforeEach(() => {
-        options = {
-          sentryDsn: '//sentryDsn',
-          whitelistUrls: ['//gitlabUrl'],
-          isProduction: true,
-        };
+    beforeEach(() => {
+      options = {
+        sentryDsn: '//sentryDsn',
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      };
 
-        raven = jasmine.createSpyObj('raven', ['install']);
+      raven = jasmine.createSpyObj('raven', ['install']);
 
-        spyOn(Raven, 'config').and.returnValue(raven);
-        spyOn(Raven, 'install');
+      spyOn(Raven, 'config').and.returnValue(raven);
 
-        RavenConfig.configure.call({
-          options,
-        });
+      RavenConfig.configure.call({
+        options,
       });
+    });
 
-      it('should call Raven.config', () => {
-        expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
-          whitelistUrls: options.whitelistUrls,
-          environment: 'production',
-        });
+    it('should call Raven.config', () => {
+      expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
+        whitelistUrls: options.whitelistUrls,
+        environment: 'production',
       });
+    });
 
-      it('should call Raven.install', () => {
-        expect(Raven.install).toHaveBeenCalled();
+    it('should call Raven.install', () => {
+      expect(raven.install).toHaveBeenCalled();
+    });
+
+    it('should set .environment to development if isProduction is false', () => {
+      options.isProduction = false;
+
+      RavenConfig.configure.call({
+        options,
       });
 
-      describe('if isProduction is false', () => {
-        beforeEach(() => {
-          options.isProduction = false;
-
-          RavenConfig.configure.call({
-            options,
-          });
-        });
-
-        it('should set .environment to development', () => {
-          expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
-            whitelistUrls: options.whitelistUrls,
-            environment: 'development',
-          });
-        });
+      expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
+        whitelistUrls: options.whitelistUrls,
+        environment: 'development',
       });
     });
   });
 
   describe('setUser', () => {
-    describe('when called', () => {
-      beforeEach(() => {});
+    let ravenConfig;
+
+    beforeEach(() => {
+      ravenConfig = { options: { currentUserId: 1 } };
+      spyOn(Raven, 'setUserContext');
+
+      RavenConfig.setUser.call(ravenConfig);
+    });
+
+    it('should call .setUserContext', function () {
+      expect(Raven.setUserContext).toHaveBeenCalledWith({
+        id: ravenConfig.options.currentUserId,
+      });
     });
   });
 
   describe('bindRavenErrors', () => {
-    describe('when called', () => {
-      beforeEach(() => {});
+    beforeEach(() => {
+      RavenConfig.bindRavenErrors();
+    });
+
+    it('should query for document using jquery', () => {
+      console.log($, 'or', $.fn);
+      // expect($).toHaveBeenCalledWith()
+    });
+
+    it('should call .on', function () {
+      // expect($document.on).toHaveBeenCalledWith('ajaxError.raven', RavenConfig.handleRavenErrors);
     });
   });
 
   describe('handleRavenErrors', () => {
-    describe('when called', () => {
-      beforeEach(() => {});
-    });
+    beforeEach(() => {});
   });
 });
-- 
GitLab


From e22bd9650d7bebfea0909fdaf5be7dcf746b53f7 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Sat, 15 Apr 2017 03:38:59 +0100
Subject: [PATCH 046/363] Updated specs, added rewire, updated layouts to move
 conditional raven and gon to head

---
 app/views/layouts/_head.html.haml           |  3 +
 app/views/layouts/application.html.haml     |  4 --
 app/views/layouts/devise.html.haml          |  3 -
 app/views/layouts/devise_empty.html.haml    |  3 -
 config/webpack.config.js                    |  2 +-
 package.json                                |  1 +
 spec/javascripts/.eslintrc                  | 15 ++++-
 spec/javascripts/raven/raven_config_spec.js | 70 ++++++++++++++++++---
 yarn.lock                                   |  4 ++
 9 files changed, 86 insertions(+), 19 deletions(-)

diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index f6d8bb08a646a..00de3b506b401 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -28,9 +28,12 @@
   = stylesheet_link_tag "application", media: "all"
   = stylesheet_link_tag "print",       media: "print"
 
+  = Gon::Base.render_data
+
   = javascript_include_tag(*webpack_asset_paths("runtime"))
   = javascript_include_tag(*webpack_asset_paths("common"))
   = javascript_include_tag(*webpack_asset_paths("main"))
+  = javascript_include_tag(*webpack_asset_paths("raven")) if sentry_enabled?
 
   - if content_for?(:page_specific_javascripts)
     = yield :page_specific_javascripts
diff --git a/app/views/layouts/application.html.haml b/app/views/layouts/application.html.haml
index cfd9481e4b23a..4c7f0b57d162e 100644
--- a/app/views/layouts/application.html.haml
+++ b/app/views/layouts/application.html.haml
@@ -2,12 +2,8 @@
 %html{ lang: "en", class: "#{page_class}" }
   = render "layouts/head"
   %body{ class: @body_class, data: { page: body_data_page, project: "#{@project.path if @project}", group: "#{@group.path if @group}" } }
-    = Gon::Base.render_data
-
     = render "layouts/header/default", title: header_title
     = render 'layouts/page', sidebar: sidebar, nav: nav
 
     = yield :scripts_body
     = render "layouts/init_auto_complete" if @gfm_form
-
-    = javascript_include_tag(*webpack_asset_paths("raven")) if sentry_enabled?
diff --git a/app/views/layouts/devise.html.haml b/app/views/layouts/devise.html.haml
index 6274f6340ab86..52fb46eb8c93f 100644
--- a/app/views/layouts/devise.html.haml
+++ b/app/views/layouts/devise.html.haml
@@ -3,7 +3,6 @@
   = render "layouts/head"
   %body.ui_charcoal.login-page.application.navless{ data: { page: body_data_page } }
     .page-wrap
-      = Gon::Base.render_data
       = render "layouts/header/empty"
       = render "layouts/broadcast"
       .container.navless-container
@@ -35,5 +34,3 @@
           = link_to "Explore", explore_root_path
           = link_to "Help", help_path
           = link_to "About GitLab", "https://about.gitlab.com/"
-
-      = javascript_include_tag(*webpack_asset_paths("raven")) if sentry_enabled?
diff --git a/app/views/layouts/devise_empty.html.haml b/app/views/layouts/devise_empty.html.haml
index 120f7299fc96c..ed6731bde9540 100644
--- a/app/views/layouts/devise_empty.html.haml
+++ b/app/views/layouts/devise_empty.html.haml
@@ -2,7 +2,6 @@
 %html{ lang: "en" }
   = render "layouts/head"
   %body.ui_charcoal.login-page.application.navless
-    = Gon::Base.render_data
     = render "layouts/header/empty"
     = render "layouts/broadcast"
     .container.navless-container
@@ -16,5 +15,3 @@
         = link_to "Explore", explore_root_path
         = link_to "Help", help_path
         = link_to "About GitLab", "https://about.gitlab.com/"
-
-    = javascript_include_tag(*webpack_asset_paths("raven")) if sentry_enabled?
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 62118522606a0..77d703e008a67 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -61,7 +61,7 @@ var config = {
       {
         test: /\.js$/,
         exclude: /(node_modules|vendor\/assets)/,
-        loader: 'babel-loader'
+        loader: 'babel-loader?plugins=rewire'
       },
       {
         test: /\.svg$/,
diff --git a/package.json b/package.json
index 0b24c5b8b0434..d05ae8f7658dc 100644
--- a/package.json
+++ b/package.json
@@ -44,6 +44,7 @@
   },
   "devDependencies": {
     "babel-plugin-istanbul": "^4.0.0",
+    "babel-plugin-rewire": "^1.1.0",
     "eslint": "^3.10.1",
     "eslint-config-airbnb-base": "^10.0.1",
     "eslint-import-resolver-webpack": "^0.8.1",
diff --git a/spec/javascripts/.eslintrc b/spec/javascripts/.eslintrc
index 3d922021978aa..a3a8bb050c95e 100644
--- a/spec/javascripts/.eslintrc
+++ b/spec/javascripts/.eslintrc
@@ -27,6 +27,19 @@
     "jasmine/no-suite-dupes": [1, "branch"],
     "jasmine/no-spec-dupes": [1, "branch"],
     "no-console": 0,
-    "prefer-arrow-callback": 0
+    "prefer-arrow-callback": 0,
+    "no-underscore-dangle": [
+      2,
+      {
+        "allow": [
+          "__GetDependency__",
+          "__Rewire__",
+          "__ResetDependency__",
+          "__get__",
+          "__set__",
+          "__RewireAPI__"
+        ]
+      }
+    ]
   }
 }
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index 3885cfde6bf80..b8bb558d22ea2 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -1,8 +1,7 @@
-import $ from 'jquery';
 import Raven from 'raven-js';
-import RavenConfig from '~/raven/raven_config';
+import RavenConfig, { __RewireAPI__ as RavenConfigRewire } from '~/raven/raven_config';
 
-fdescribe('RavenConfig', () => {
+describe('RavenConfig', () => {
   describe('init', () => {
     let options;
 
@@ -116,21 +115,78 @@ fdescribe('RavenConfig', () => {
   });
 
   describe('bindRavenErrors', () => {
+    let $document;
+    let $;
+
     beforeEach(() => {
+      $document = jasmine.createSpyObj('$document', ['on']);
+      $ = jasmine.createSpy('$').and.returnValue($document);
+
+      RavenConfigRewire.__set__('$', $);
+
       RavenConfig.bindRavenErrors();
     });
 
     it('should query for document using jquery', () => {
-      console.log($, 'or', $.fn);
-      // expect($).toHaveBeenCalledWith()
+      expect($).toHaveBeenCalledWith(document);
     });
 
     it('should call .on', function () {
-      // expect($document.on).toHaveBeenCalledWith('ajaxError.raven', RavenConfig.handleRavenErrors);
+      expect($document.on).toHaveBeenCalledWith('ajaxError.raven', RavenConfig.handleRavenErrors);
     });
   });
 
   describe('handleRavenErrors', () => {
-    beforeEach(() => {});
+    let event;
+    let req;
+    let config;
+    let err;
+
+    beforeEach(() => {
+      event = {};
+      req = { status: 'status', responseText: 'responseText', statusText: 'statusText' };
+      config = { type: 'type', url: 'url', data: 'data' };
+      err = {};
+
+      spyOn(Raven, 'captureMessage');
+
+      RavenConfig.handleRavenErrors(event, req, config, err);
+    });
+
+    it('should call Raven.captureMessage', () => {
+      expect(Raven.captureMessage).toHaveBeenCalledWith(err, {
+        extra: {
+          type: config.type,
+          url: config.url,
+          data: config.data,
+          status: req.status,
+          response: req.responseText.substring(0, 100),
+          error: err,
+          event,
+        },
+      });
+    });
+
+    describe('if no err is provided', () => {
+      beforeEach(() => {
+        Raven.captureMessage.calls.reset();
+
+        RavenConfig.handleRavenErrors(event, req, config);
+      });
+
+      it('should use req.statusText as the error value', () => {
+        expect(Raven.captureMessage).toHaveBeenCalledWith(req.statusText, {
+          extra: {
+            type: config.type,
+            url: config.url,
+            data: config.data,
+            status: req.status,
+            response: req.responseText.substring(0, 100),
+            error: req.statusText,
+            event,
+          },
+        });
+      });
+    });
   });
 });
diff --git a/yarn.lock b/yarn.lock
index 46f528f0bec5b..6e9b9bd9cec6b 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -406,6 +406,10 @@ babel-plugin-istanbul@^4.0.0:
     istanbul-lib-instrument "^1.4.2"
     test-exclude "^4.0.0"
 
+babel-plugin-rewire@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/babel-plugin-rewire/-/babel-plugin-rewire-1.1.0.tgz#a6b966d9d8c06c03d95dcda2eec4e2521519549b"
+
 babel-plugin-syntax-async-functions@^6.8.0:
   version "6.13.0"
   resolved "https://registry.yarnpkg.com/babel-plugin-syntax-async-functions/-/babel-plugin-syntax-async-functions-6.13.0.tgz#cad9cad1191b5ad634bf30ae0872391e0647be95"
-- 
GitLab


From 69e49a095ded719ccfb1287b6100646663de3f03 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Sat, 15 Apr 2017 11:52:37 +0100
Subject: [PATCH 047/363] Only rewire in dev

---
 config/webpack.config.js | 5 +++--
 1 file changed, 3 insertions(+), 2 deletions(-)

diff --git a/config/webpack.config.js b/config/webpack.config.js
index f034a8ae27d92..79c1d66a24214 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -15,6 +15,7 @@ var DEV_SERVER_HOST = process.env.DEV_SERVER_HOST || 'localhost';
 var DEV_SERVER_PORT = parseInt(process.env.DEV_SERVER_PORT, 10) || 3808;
 var DEV_SERVER_LIVERELOAD = process.env.DEV_SERVER_LIVERELOAD !== 'false';
 var WEBPACK_REPORT = process.env.WEBPACK_REPORT;
+var rewirePlugin = IS_PRODUCTION ? '?plugins=rewire' : '';
 
 var config = {
   context: path.join(ROOT_PATH, 'app/assets/javascripts'),
@@ -52,7 +53,7 @@ var config = {
     vue_pipelines:        './vue_pipelines_index/index.js',
     raven:                './raven/index.js',
     issue_show:           './issue_show/index.js',
-    group:                './group.js'
+    group:                './group.js',
   },
 
   output: {
@@ -68,7 +69,7 @@ var config = {
       {
         test: /\.js$/,
         exclude: /(node_modules|vendor\/assets)/,
-        loader: 'babel-loader?plugins=rewire',
+        loader: `babel-loader${rewirePlugin}`,
       },
       {
         test: /\.vue$/,
-- 
GitLab


From 86b4f49c8c76a322d2f12f61e5a2d27d3c5c4671 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Sat, 15 Apr 2017 12:29:46 +0100
Subject: [PATCH 048/363] Removed rewire and fixed tests

---
 app/assets/javascripts/raven/raven_config.js |  6 ++--
 config/webpack.config.js                     |  3 +-
 package.json                                 |  1 -
 spec/javascripts/.eslintrc                   | 15 +---------
 spec/javascripts/raven/raven_config_spec.js  | 30 ++++++++++++++++++--
 yarn.lock                                    |  4 ---
 6 files changed, 32 insertions(+), 27 deletions(-)

diff --git a/app/assets/javascripts/raven/raven_config.js b/app/assets/javascripts/raven/raven_config.js
index bb9be7cb196fb..9c756a4130b74 100644
--- a/app/assets/javascripts/raven/raven_config.js
+++ b/app/assets/javascripts/raven/raven_config.js
@@ -1,5 +1,4 @@
 import Raven from 'raven-js';
-import $ from 'jquery';
 
 const RavenConfig = {
   init(options = {}) {
@@ -24,11 +23,12 @@ const RavenConfig = {
   },
 
   bindRavenErrors() {
-    $(document).on('ajaxError.raven', this.handleRavenErrors);
+    window.$(document).on('ajaxError.raven', this.handleRavenErrors);
   },
 
   handleRavenErrors(event, req, config, err) {
     const error = err || req.statusText;
+    const responseText = req.responseText || 'Unknown response text';
 
     Raven.captureMessage(error, {
       extra: {
@@ -36,7 +36,7 @@ const RavenConfig = {
         url: config.url,
         data: config.data,
         status: req.status,
-        response: req.responseText.substring(0, 100),
+        response: responseText.substring(0, 100),
         error,
         event,
       },
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 79c1d66a24214..2153c8bbaf281 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -15,7 +15,6 @@ var DEV_SERVER_HOST = process.env.DEV_SERVER_HOST || 'localhost';
 var DEV_SERVER_PORT = parseInt(process.env.DEV_SERVER_PORT, 10) || 3808;
 var DEV_SERVER_LIVERELOAD = process.env.DEV_SERVER_LIVERELOAD !== 'false';
 var WEBPACK_REPORT = process.env.WEBPACK_REPORT;
-var rewirePlugin = IS_PRODUCTION ? '?plugins=rewire' : '';
 
 var config = {
   context: path.join(ROOT_PATH, 'app/assets/javascripts'),
@@ -69,7 +68,7 @@ var config = {
       {
         test: /\.js$/,
         exclude: /(node_modules|vendor\/assets)/,
-        loader: `babel-loader${rewirePlugin}`,
+        loader: 'babel-loader',
       },
       {
         test: /\.vue$/,
diff --git a/package.json b/package.json
index cf66d7ec58e0a..4ac1526bba184 100644
--- a/package.json
+++ b/package.json
@@ -55,7 +55,6 @@
   },
   "devDependencies": {
     "babel-plugin-istanbul": "^4.0.0",
-    "babel-plugin-rewire": "^1.1.0",
     "eslint": "^3.10.1",
     "eslint-config-airbnb-base": "^10.0.1",
     "eslint-import-resolver-webpack": "^0.8.1",
diff --git a/spec/javascripts/.eslintrc b/spec/javascripts/.eslintrc
index a3a8bb050c95e..3d922021978aa 100644
--- a/spec/javascripts/.eslintrc
+++ b/spec/javascripts/.eslintrc
@@ -27,19 +27,6 @@
     "jasmine/no-suite-dupes": [1, "branch"],
     "jasmine/no-spec-dupes": [1, "branch"],
     "no-console": 0,
-    "prefer-arrow-callback": 0,
-    "no-underscore-dangle": [
-      2,
-      {
-        "allow": [
-          "__GetDependency__",
-          "__Rewire__",
-          "__ResetDependency__",
-          "__get__",
-          "__set__",
-          "__RewireAPI__"
-        ]
-      }
-    ]
+    "prefer-arrow-callback": 0
   }
 }
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index b8bb558d22ea2..55f3949c74008 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -1,5 +1,5 @@
 import Raven from 'raven-js';
-import RavenConfig, { __RewireAPI__ as RavenConfigRewire } from '~/raven/raven_config';
+import RavenConfig from '~/raven/raven_config';
 
 describe('RavenConfig', () => {
   describe('init', () => {
@@ -122,13 +122,13 @@ describe('RavenConfig', () => {
       $document = jasmine.createSpyObj('$document', ['on']);
       $ = jasmine.createSpy('$').and.returnValue($document);
 
-      RavenConfigRewire.__set__('$', $);
+      window.$ = $;
 
       RavenConfig.bindRavenErrors();
     });
 
     it('should query for document using jquery', () => {
-      expect($).toHaveBeenCalledWith(document);
+      expect(window.$).toHaveBeenCalledWith(document);
     });
 
     it('should call .on', function () {
@@ -188,5 +188,29 @@ describe('RavenConfig', () => {
         });
       });
     });
+
+    describe('if no req.responseText is provided', () => {
+      beforeEach(() => {
+        req.responseText = undefined;
+
+        Raven.captureMessage.calls.reset();
+
+        RavenConfig.handleRavenErrors(event, req, config, err);
+      });
+
+      it('should use `Unknown response text` as the response', () => {
+        expect(Raven.captureMessage).toHaveBeenCalledWith(err, {
+          extra: {
+            type: config.type,
+            url: config.url,
+            data: config.data,
+            status: req.status,
+            response: 'Unknown response text',
+            error: err,
+            event,
+          },
+        });
+      });
+    });
   });
 });
diff --git a/yarn.lock b/yarn.lock
index e41f737c88c92..dd169cf28ba81 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -425,10 +425,6 @@ babel-plugin-istanbul@^4.0.0:
     istanbul-lib-instrument "^1.4.2"
     test-exclude "^4.0.0"
 
-babel-plugin-rewire@^1.1.0:
-  version "1.1.0"
-  resolved "https://registry.yarnpkg.com/babel-plugin-rewire/-/babel-plugin-rewire-1.1.0.tgz#a6b966d9d8c06c03d95dcda2eec4e2521519549b"
-
 babel-plugin-syntax-async-functions@^6.8.0:
   version "6.13.0"
   resolved "https://registry.yarnpkg.com/babel-plugin-syntax-async-functions/-/babel-plugin-syntax-async-functions-6.13.0.tgz#cad9cad1191b5ad634bf30ae0872391e0647be95"
-- 
GitLab


From 90ba69d224eb7ef3d91332f1b7944c68ad16affd Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Sat, 15 Apr 2017 16:07:31 +0100
Subject: [PATCH 049/363] Started internationalising cycyle analytics

---
 .eslintignore                           |  1 +
 Gemfile                                 |  1 +
 Gemfile.lock                            |  8 ++++++++
 app/assets/javascripts/locale/de/app.js |  1 +
 app/assets/javascripts/locale/en/app.js |  1 +
 app/assets/javascripts/locale/es/app.js |  1 +
 app/assets/javascripts/locale/index.js  | 15 +++++++++++++++
 app/views/layouts/_head.html.haml       |  1 +
 app/views/layouts/application.html.haml |  2 +-
 config/webpack.config.js                |  5 +++++
 package.json                            |  2 ++
 yarn.lock                               | 19 +++++++++++++++----
 12 files changed, 52 insertions(+), 5 deletions(-)
 create mode 100644 app/assets/javascripts/locale/de/app.js
 create mode 100644 app/assets/javascripts/locale/en/app.js
 create mode 100644 app/assets/javascripts/locale/es/app.js
 create mode 100644 app/assets/javascripts/locale/index.js

diff --git a/.eslintignore b/.eslintignore
index c742b08c00540..1605e483e9e63 100644
--- a/.eslintignore
+++ b/.eslintignore
@@ -7,3 +7,4 @@
 /vendor/
 karma.config.js
 webpack.config.js
+/app/assets/javascripts/locale/**/*.js
diff --git a/Gemfile b/Gemfile
index 6c1f634607154..736e3e3b57849 100644
--- a/Gemfile
+++ b/Gemfile
@@ -256,6 +256,7 @@ gem 'premailer-rails', '~> 1.9.0'
 
 # I18n
 gem 'gettext_i18n_rails', '~> 1.8.0'
+gem 'gettext_i18n_rails_js', '~> 1.2.0'
 gem 'gettext', '~> 3.2.2', require: false, group: :development
 
 # Metrics
diff --git a/Gemfile.lock b/Gemfile.lock
index ca4084e18a232..94727810ac9e2 100644
--- a/Gemfile.lock
+++ b/Gemfile.lock
@@ -259,6 +259,11 @@ GEM
       text (>= 1.3.0)
     gettext_i18n_rails (1.8.0)
       fast_gettext (>= 0.9.0)
+    gettext_i18n_rails_js (1.2.0)
+      gettext (>= 3.0.2)
+      gettext_i18n_rails (>= 0.7.1)
+      po_to_json (>= 1.0.0)
+      rails (>= 3.2.0)
     gherkin-ruby (0.3.2)
     gitaly (0.5.0)
       google-protobuf (~> 3.1)
@@ -534,6 +539,8 @@ GEM
       ast (~> 2.2)
     path_expander (1.0.1)
     pg (0.18.4)
+    po_to_json (1.0.1)
+      json (>= 1.6.0)
     poltergeist (1.9.0)
       capybara (~> 2.1)
       cliver (~> 0.3.1)
@@ -914,6 +921,7 @@ DEPENDENCIES
   gemojione (~> 3.0)
   gettext (~> 3.2.2)
   gettext_i18n_rails (~> 1.8.0)
+  gettext_i18n_rails_js (~> 1.2.0)
   gitaly (~> 0.5.0)
   github-linguist (~> 4.7.0)
   gitlab-flowdock-git-hook (~> 1.0.1)
diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
new file mode 100644
index 0000000000000..643e82a90a073
--- /dev/null
+++ b/app/assets/javascripts/locale/de/app.js
@@ -0,0 +1 @@
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Deutsch":[""],"English":[""],"Spanish":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
new file mode 100644
index 0000000000000..9070b519ff330
--- /dev/null
+++ b/app/assets/javascripts/locale/en/app.js
@@ -0,0 +1 @@
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Deutsch":[""],"English":[""],"Spanish":[""]}}};
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
new file mode 100644
index 0000000000000..41f6ddef5b868
--- /dev/null
+++ b/app/assets/javascripts/locale/es/app.js
@@ -0,0 +1 @@
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-13 00:07-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Deutsch":["Alemán"],"English":["Inglés"],"Spanish":["Español"]}}};
diff --git a/app/assets/javascripts/locale/index.js b/app/assets/javascripts/locale/index.js
new file mode 100644
index 0000000000000..0feb9eb5d43a9
--- /dev/null
+++ b/app/assets/javascripts/locale/index.js
@@ -0,0 +1,15 @@
+import Jed from 'jed';
+import de from './de/app';
+import es from './es/app';
+import en from './en/app';
+
+const locales = {
+  de,
+  es,
+  en,
+};
+
+const lang = document.querySelector('html').getAttribute('lang');
+
+export { lang };
+export default new Jed(locales[lang]);
diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index 19473b6ab276d..7e1986794893f 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -31,6 +31,7 @@
   = webpack_bundle_tag "runtime"
   = webpack_bundle_tag "common"
   = webpack_bundle_tag "main"
+  = webpack_bundle_tag "locale"
 
   - if content_for?(:page_specific_javascripts)
     = yield :page_specific_javascripts
diff --git a/app/views/layouts/application.html.haml b/app/views/layouts/application.html.haml
index 36543edc040f9..dc926a615c76b 100644
--- a/app/views/layouts/application.html.haml
+++ b/app/views/layouts/application.html.haml
@@ -1,5 +1,5 @@
 !!! 5
-%html{ lang: "en", class: "#{page_class}" }
+%html{ lang: I18n.locale, class: "#{page_class}" }
   = render "layouts/head"
   %body{ class: @body_class, data: { page: body_data_page, project: "#{@project.path if @project}", group: "#{@group.path if @group}" } }
     = Gon::Base.render_data
diff --git a/config/webpack.config.js b/config/webpack.config.js
index ffb161900933d..df7b43cf053e9 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -52,6 +52,7 @@ var config = {
     vue_pipelines:        './vue_pipelines_index/index.js',
     issue_show:           './issue_show/index.js',
     group:                './group.js',
+    locale:               './locale/index.js',
   },
 
   output: {
@@ -82,6 +83,10 @@ var config = {
         exclude: /node_modules/,
         loader: 'file-loader',
       },
+      {
+        test: /locale\/[a-z]+\/(.*)\.js$/,
+        loader: 'exports-loader?locales',
+      },
     ]
   },
 
diff --git a/package.json b/package.json
index a17399ddb8f61..5cb07ddc3b75c 100644
--- a/package.json
+++ b/package.json
@@ -26,7 +26,9 @@
     "dropzone": "^4.2.0",
     "emoji-unicode-version": "^0.2.1",
     "eslint-plugin-html": "^2.0.1",
+    "exports-loader": "^0.6.4",
     "file-loader": "^0.11.1",
+    "jed": "^1.1.1",
     "jquery": "^2.2.1",
     "jquery-ujs": "^1.2.1",
     "js-cookie": "^2.1.3",
diff --git a/yarn.lock b/yarn.lock
index e16cd9c367301..b95df2d23838f 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -2143,6 +2143,13 @@ expand-range@^1.8.1:
   dependencies:
     fill-range "^2.1.0"
 
+exports-loader@^0.6.4:
+  version "0.6.4"
+  resolved "https://registry.yarnpkg.com/exports-loader/-/exports-loader-0.6.4.tgz#d70fc6121975b35fc12830cf52754be2740fc886"
+  dependencies:
+    loader-utils "^1.0.2"
+    source-map "0.5.x"
+
 express@^4.13.3, express@^4.14.1:
   version "4.14.1"
   resolved "https://registry.yarnpkg.com/express/-/express-4.14.1.tgz#646c237f766f148c2120aff073817b9e4d7e0d33"
@@ -3080,6 +3087,10 @@ jasmine-jquery@^2.1.1:
   version "2.1.1"
   resolved "https://registry.yarnpkg.com/jasmine-jquery/-/jasmine-jquery-2.1.1.tgz#d4095e646944a26763235769ab018d9f30f0d47b"
 
+jed@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/jed/-/jed-1.1.1.tgz#7a549bbd9ffe1585b0cd0a191e203055bee574b4"
+
 jodid25519@^1.0.0:
   version "1.0.2"
   resolved "https://registry.yarnpkg.com/jodid25519/-/jodid25519-1.0.2.tgz#06d4912255093419477d425633606e0e90782967"
@@ -5071,6 +5082,10 @@ source-map-support@^0.4.2:
   dependencies:
     source-map "^0.5.3"
 
+source-map@0.5.x, source-map@^0.5.0, source-map@^0.5.3, source-map@^0.5.6, source-map@~0.5.1, source-map@~0.5.3:
+  version "0.5.6"
+  resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.6.tgz#75ce38f52bf0733c5a7f0c118d81334a2bb5f412"
+
 source-map@^0.1.41:
   version "0.1.43"
   resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.1.43.tgz#c24bc146ca517c1471f5dacbe2571b2b7f9e3346"
@@ -5083,10 +5098,6 @@ source-map@^0.4.4:
   dependencies:
     amdefine ">=0.0.4"
 
-source-map@^0.5.0, source-map@^0.5.3, source-map@^0.5.6, source-map@~0.5.1, source-map@~0.5.3:
-  version "0.5.6"
-  resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.6.tgz#75ce38f52bf0733c5a7f0c118d81334a2bb5f412"
-
 source-map@~0.2.0:
   version "0.2.0"
   resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.2.0.tgz#dab73fbcfc2ba819b4de03bd6f6eaa48164b3f9d"
-- 
GitLab


From 7963c2c25114e871eb42c0859b6d24fc37437a8a Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Tue, 18 Apr 2017 14:55:09 +0100
Subject: [PATCH 050/363] Added Vue filters & directives for translating

---
 .../components/limit_warning_component.js     |  2 +-
 .../components/stage_code_component.js        |  4 ++--
 .../components/stage_issue_component.js       |  4 ++--
 .../components/stage_plan_component.js        |  4 ++--
 .../components/stage_production_component.js  |  4 ++--
 .../components/stage_review_component.js      |  4 ++--
 .../components/stage_staging_component.js     |  2 +-
 .../components/total_time_component.js        |  6 ++---
 .../cycle_analytics/cycle_analytics_bundle.js |  3 +++
 .../cycle_analytics/cycle_analytics_store.js  | 16 ++++++++------
 .../javascripts/vue_shared/translate.js       | 14 ++++++++++++
 .../cycle_analytics/_empty_stage.html.haml    |  2 +-
 .../cycle_analytics/_no_access.html.haml      |  4 ++--
 .../projects/cycle_analytics/show.html.haml   | 22 +++++++++----------
 config/webpack.config.js                      |  8 +++++++
 15 files changed, 63 insertions(+), 36 deletions(-)
 create mode 100644 app/assets/javascripts/vue_shared/translate.js

diff --git a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
index abe48572347b0..a7b187a0a36ba 100644
--- a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
@@ -11,7 +11,7 @@ export default {
           aria-hidden="true"
           title="Limited to showing 50 events at most"
           data-placement="top"></i>
-      Showing 50 events
+      {{ 'Showing 50 events' | translate }}
     </span>
   `,
 };
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
index 80bd2df6f42ea..f72882872cdf1 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageCodeComponent = Vue.extend({
             <a :href="mergeRequest.url" class="issue-link">!{{ mergeRequest.iid }}</a>
             &middot;
             <span>
-              Opened
+              {{ 'Opened' | translate }}
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              by
+              {{ 'by' | translate }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
           </div>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
index 20a43798fbedc..bb265c8316feb 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageIssueComponent = Vue.extend({
             <a :href="issue.url" class="issue-link">#{{ issue.iid }}</a>
             &middot;
             <span>
-              Opened
+              {{ 'Opened' | translate }}
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-              by
+              {{ 'by' | translate }}
               <a :href="issue.author.webUrl" class="issue-author-link">
                 {{ issue.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
index f33cac3da8248..32b685faece6a 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
@@ -31,10 +31,10 @@ global.cycleAnalytics.StagePlanComponent = Vue.extend({
               </a>
             </h5>
             <span>
-              First
+              {{ 'First' | translate }}
               <span class="commit-icon">${iconCommit}</span>
               <a :href="commit.commitUrl" class="commit-hash-link monospace">{{ commit.shortSha }}</a>
-              pushed by
+              {{ 'pushed by' | translate }}
               <a :href="commit.author.webUrl" class="commit-author-link">
                 {{ commit.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
index 657f538537433..5c9186a2e497c 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageProductionComponent = Vue.extend({
             <a :href="issue.url" class="issue-link">#{{ issue.iid }}</a>
             &middot;
             <span>
-              Opened
+              {{ 'Opened' | translate }}
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-            by
+            {{ 'by' | translate }}
             <a :href="issue.author.webUrl" class="issue-author-link">
               {{ issue.author.name }}
             </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
index 8a801300647aa..a047573548df0 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageReviewComponent = Vue.extend({
             <a :href="mergeRequest.url" class="issue-link">!{{ mergeRequest.iid }}</a>
             &middot;
             <span>
-              Opened
+              {{ 'Opened' | translate }}
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              by
+              {{ 'by' | translate }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
             <template v-if="mergeRequest.state === 'closed'">
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
index 4a28637958848..e8dfaf4294e19 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageStagingComponent = Vue.extend({
             </h5>
             <span>
               <a :href="build.url" class="build-date">{{ build.date }}</a>
-              by
+              {{ 'by' | translate }}
               <a :href="build.author.webUrl" class="issue-author-link">
                 {{ build.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/total_time_component.js b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
index 77edcb7627323..e07963b1b45c6 100644
--- a/app/assets/javascripts/cycle_analytics/components/total_time_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
@@ -12,9 +12,9 @@ global.cycleAnalytics.TotalTimeComponent = Vue.extend({
   template: `
     <span class="total-time">
       <template v-if="Object.keys(time).length">
-        <template v-if="time.days">{{ time.days }} <span>{{ time.days === 1 ? 'day' : 'days' }}</span></template>
-        <template v-if="time.hours">{{ time.hours }} <span>hr</span></template>
-        <template v-if="time.mins && !time.days">{{ time.mins }} <span>mins</span></template>
+        <template v-if="time.days">{{ time.days }} <span>{{ time.days === 1 ? 'day' : 'days' | translate }}</span></template>
+        <template v-if="time.hours">{{ time.hours }} <span v-translate>hr</span></template>
+        <template v-if="time.mins && !time.days">{{ time.mins }} <span v-translate>mins</span></template>
         <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>s</span></template>
       </template>
       <template v-else>
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_bundle.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_bundle.js
index b099b39e58f20..95257e58e6b0c 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_bundle.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_bundle.js
@@ -2,6 +2,7 @@
 
 import Vue from 'vue';
 import Cookies from 'js-cookie';
+import Translate from '../vue_shared/translate';
 import LimitWarningComponent from './components/limit_warning_component';
 
 require('./components/stage_code_component');
@@ -16,6 +17,8 @@ require('./cycle_analytics_service');
 require('./cycle_analytics_store');
 require('./default_event_objects');
 
+Vue.use(Translate);
+
 $(() => {
   const OVERVIEW_DIALOG_COOKIE = 'cycle_analytics_help_dismissed';
   const cycleAnalyticsEl = document.querySelector('#cycle-analytics');
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
index 6536a8fd7fa28..25d5092a1fd8e 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
@@ -1,5 +1,7 @@
 /* eslint-disable no-param-reassign */
 
+import locale from '~/locale';
+
 require('../lib/utils/text_utility');
 const DEFAULT_EVENT_OBJECTS = require('./default_event_objects');
 
@@ -7,13 +9,13 @@ const global = window.gl || (window.gl = {});
 global.cycleAnalytics = global.cycleAnalytics || {};
 
 const EMPTY_STAGE_TEXTS = {
-  issue: 'The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.',
-  plan: 'The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.',
-  code: 'The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.',
-  test: 'The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.',
-  review: 'The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.',
-  staging: 'The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.',
-  production: 'The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.',
+  issue: locale.gettext('The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.'),
+  plan: locale.gettext('The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.'),
+  code: locale.gettext('The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.'),
+  test: locale.gettext('The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.'),
+  review: locale.gettext('The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.'),
+  staging: locale.gettext('The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.'),
+  production: locale.gettext('The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.'),
 };
 
 global.cycleAnalytics.CycleAnalyticsStore = {
diff --git a/app/assets/javascripts/vue_shared/translate.js b/app/assets/javascripts/vue_shared/translate.js
new file mode 100644
index 0000000000000..88b7c0bb95436
--- /dev/null
+++ b/app/assets/javascripts/vue_shared/translate.js
@@ -0,0 +1,14 @@
+import locale from '../locale';
+
+export default (Vue) => {
+  Vue.filter('translate', text => locale.gettext(text));
+
+  Vue.directive('translate', {
+    bind(el) {
+      const $el = el;
+      const text = $el.textContent.trim();
+
+      $el.textContent = locale.gettext(text);
+    },
+  });
+};
diff --git a/app/views/projects/cycle_analytics/_empty_stage.html.haml b/app/views/projects/cycle_analytics/_empty_stage.html.haml
index c3f95860e9237..27190785fff2d 100644
--- a/app/views/projects/cycle_analytics/_empty_stage.html.haml
+++ b/app/views/projects/cycle_analytics/_empty_stage.html.haml
@@ -2,6 +2,6 @@
   .empty-stage
     .icon-no-data
       = custom_icon ('icon_no_data')
-    %h4 We don't have enough data to show this stage.
+    %h4 {{ 'We don\'t have enough data to show this stage.' | translate }}
     %p
       {{currentStage.emptyStageText}}
diff --git a/app/views/projects/cycle_analytics/_no_access.html.haml b/app/views/projects/cycle_analytics/_no_access.html.haml
index 0ffc79b318122..474d0f410a704 100644
--- a/app/views/projects/cycle_analytics/_no_access.html.haml
+++ b/app/views/projects/cycle_analytics/_no_access.html.haml
@@ -2,6 +2,6 @@
   .no-access-stage
     .icon-lock
       = custom_icon ('icon_lock')
-    %h4 You need permission.
+    %h4 {{ 'You need permission.' | translate }}
     %p
-      Want to see the data? Please ask administrator for access.
+      {{ 'Want to see the data? Please ask administrator for access.' | translate }}
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index dd3fa814716ea..47fde758ffcff 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -15,9 +15,9 @@
           = custom_icon('icon_cycle_analytics_splash')
         .col-sm-8.col-xs-12.inner-content
           %h4
-            Introducing Cycle Analytics
+            {{ 'Introducing Cycle Analytics' | translate }}
           %p
-            Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.
+            {{ 'Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.' | translate }}
 
           = link_to "Read more",  help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn'
   = icon("spinner spin", "v-show" => "isLoading")
@@ -34,15 +34,15 @@
             .col-sm-3.col-xs-12.column
               .dropdown.inline.js-ca-dropdown
                 %button.dropdown-menu-toggle{ "data-toggle" => "dropdown", :type => "button" }
-                  %span.dropdown-label Last 30 days
+                  %span.dropdown-label {{ 'Last 30 days' | translate }}
                   %i.fa.fa-chevron-down
                 %ul.dropdown-menu.dropdown-menu-align-right
                   %li
                     %a{ "href" => "#", "data-value" => "30" }
-                      Last 30 days
+                      {{ 'Last 30 days' | translate }}
                   %li
                     %a{ "href" => "#", "data-value" => "90" }
-                      Last 90 days
+                      {{ 'Last 30 days' | translate }}
     .stage-panel-container
       .panel.panel-default.stage-panel
         .panel-heading
@@ -50,19 +50,19 @@
             %ul
               %li.stage-header
                 %span.stage-name
-                  Stage
+                  {{ 'Stage' | translate }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The phase of the development lifecycle.", "aria-hidden" => "true" }
               %li.median-header
                 %span.stage-name
-                  Median
+                  {{ 'Median' | translate }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.", "aria-hidden" => "true" }
               %li.event-header
                 %span.stage-name
-                  {{ currentStage ? currentStage.legend : 'Related Issues' }}
+                  {{ currentStage ? currentStage.legend : 'Related Issues' | translate }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The collection of events added to the data gathered for that stage.", "aria-hidden" => "true" }
               %li.total-time-header
                 %span.stage-name
-                  Total Time
+                  {{ 'Total Time' | translate }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The time taken by each data entry gathered by that stage.", "aria-hidden" => "true" }
         .stage-panel-body
           %nav.stage-nav
@@ -75,10 +75,10 @@
                     %span{ "v-if" => "stage.value" }
                       {{ stage.value }}
                     %span.stage-empty{ "v-else" => true }
-                      Not enough data
+                      {{ 'Not enough data' | translate }}
                   %template{ "v-else" => true }
                     %span.not-available
-                      Not available
+                      {{ 'Not available' | translate }}
           .section.stage-events
             %template{ "v-if" => "isLoadingStage" }
               = icon("spinner spin")
diff --git a/config/webpack.config.js b/config/webpack.config.js
index df7b43cf053e9..92ecba45d6e86 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -150,6 +150,14 @@ var config = {
     new webpack.optimize.CommonsChunkPlugin({
       names: ['main', 'common', 'runtime'],
     }),
+
+    // locale common library
+    new webpack.optimize.CommonsChunkPlugin({
+      name: 'locale',
+      chunks: [
+        'cycle_analytics',
+      ],
+    }),
   ],
 
   resolve: {
-- 
GitLab


From b0df1ed4884a16c5c45800031667e0dd3251c978 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Tue, 18 Apr 2017 15:11:32 +0100
Subject: [PATCH 051/363] Make the locale file page specific

---
 app/views/layouts/_head.html.haml                 | 1 -
 app/views/projects/cycle_analytics/show.html.haml | 4 +++-
 2 files changed, 3 insertions(+), 2 deletions(-)

diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index 7e1986794893f..19473b6ab276d 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -31,7 +31,6 @@
   = webpack_bundle_tag "runtime"
   = webpack_bundle_tag "common"
   = webpack_bundle_tag "main"
-  = webpack_bundle_tag "locale"
 
   - if content_for?(:page_specific_javascripts)
     = yield :page_specific_javascripts
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index 47fde758ffcff..a0a629eaa4108 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -2,6 +2,7 @@
 - page_title "Cycle Analytics"
 - content_for :page_specific_javascripts do
   = page_specific_javascript_bundle_tag('common_vue')
+  = page_specific_javascript_bundle_tag('locale')
   = page_specific_javascript_bundle_tag('cycle_analytics')
 
 = render "projects/head"
@@ -19,7 +20,8 @@
           %p
             {{ 'Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.' | translate }}
 
-          = link_to "Read more",  help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn'
+          = link_to help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn' do
+            {{ 'Read more' | translate }}
   = icon("spinner spin", "v-show" => "isLoading")
   .wrapper{ "v-show" => "!isLoading && !hasError" }
     .panel.panel-default
-- 
GitLab


From 372841975feace348ffdcc14819b9c9a9da8b181 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Tue, 18 Apr 2017 16:29:22 +0100
Subject: [PATCH 052/363] Default the language to en - more useful for specs

---
 app/assets/javascripts/locale/index.js                         | 2 +-
 .../cycle_analytics/limit_warning_component_spec.js            | 3 +++
 2 files changed, 4 insertions(+), 1 deletion(-)

diff --git a/app/assets/javascripts/locale/index.js b/app/assets/javascripts/locale/index.js
index 0feb9eb5d43a9..56791968e5375 100644
--- a/app/assets/javascripts/locale/index.js
+++ b/app/assets/javascripts/locale/index.js
@@ -9,7 +9,7 @@ const locales = {
   en,
 };
 
-const lang = document.querySelector('html').getAttribute('lang');
+const lang = document.querySelector('html').getAttribute('lang') || 'en';
 
 export { lang };
 export default new Jed(locales[lang]);
diff --git a/spec/javascripts/cycle_analytics/limit_warning_component_spec.js b/spec/javascripts/cycle_analytics/limit_warning_component_spec.js
index 50000c5a5f560..2fb9eb0ca8561 100644
--- a/spec/javascripts/cycle_analytics/limit_warning_component_spec.js
+++ b/spec/javascripts/cycle_analytics/limit_warning_component_spec.js
@@ -1,6 +1,9 @@
 import Vue from 'vue';
+import Translate from '~/vue_shared/translate';
 import limitWarningComp from '~/cycle_analytics/components/limit_warning_component';
 
+Vue.use(Translate);
+
 describe('Limit warning component', () => {
   let component;
   let LimitWarningComponent;
-- 
GitLab


From a3506a228723d7e31fb37580dcd3b30245436f22 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Tue, 18 Apr 2017 17:00:44 +0100
Subject: [PATCH 053/363] Fixed missing text Fixed duplicated time in dropdown

---
 app/views/projects/cycle_analytics/show.html.haml | 4 ++--
 config/webpack.config.js                          | 2 +-
 2 files changed, 3 insertions(+), 3 deletions(-)

diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index a0a629eaa4108..a7783b6958803 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -26,7 +26,7 @@
   .wrapper{ "v-show" => "!isLoading && !hasError" }
     .panel.panel-default
       .panel-heading
-        Pipeline Health
+        {{ 'Pipeline Health' | translate }}
       .content-block
         .container-fluid
           .row
@@ -44,7 +44,7 @@
                       {{ 'Last 30 days' | translate }}
                   %li
                     %a{ "href" => "#", "data-value" => "90" }
-                      {{ 'Last 30 days' | translate }}
+                      {{ 'Last 90 days' | translate }}
     .stage-panel-container
       .panel.panel-default.stage-panel
         .panel-heading
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 92ecba45d6e86..f93f4d6cd5e4c 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -34,6 +34,7 @@ var config = {
     graphs:               './graphs/graphs_bundle.js',
     groups_list:          './groups_list.js',
     issuable:             './issuable/issuable_bundle.js',
+    locale:               './locale/index.js',
     merge_conflicts:      './merge_conflicts/merge_conflicts_bundle.js',
     merge_request_widget: './merge_request_widget/ci_bundle.js',
     monitoring:           './monitoring/monitoring_bundle.js',
@@ -52,7 +53,6 @@ var config = {
     vue_pipelines:        './vue_pipelines_index/index.js',
     issue_show:           './issue_show/index.js',
     group:                './group.js',
-    locale:               './locale/index.js',
   },
 
   output: {
-- 
GitLab


From af6e3e57e4c0ee861e26310c65be2e5e8547ca68 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 18 Apr 2017 18:58:03 +0100
Subject: [PATCH 054/363] Added sampling function and blacklisted common urls
 and error messages

---
 app/assets/javascripts/raven/raven_config.js | 53 +++++++++++++++++
 spec/javascripts/raven/raven_config_spec.js  | 62 ++++++++++++++++++++
 2 files changed, 115 insertions(+)

diff --git a/app/assets/javascripts/raven/raven_config.js b/app/assets/javascripts/raven/raven_config.js
index 9c756a4130b74..6c20cd9aabdb7 100644
--- a/app/assets/javascripts/raven/raven_config.js
+++ b/app/assets/javascripts/raven/raven_config.js
@@ -1,6 +1,52 @@
 import Raven from 'raven-js';
 
+const IGNORE_ERRORS = [
+  // Random plugins/extensions
+  'top.GLOBALS',
+  // See: http://blog.errorception.com/2012/03/tale-of-unfindable-js-error. html
+  'originalCreateNotification',
+  'canvas.contentDocument',
+  'MyApp_RemoveAllHighlights',
+  'http://tt.epicplay.com',
+  'Can\'t find variable: ZiteReader',
+  'jigsaw is not defined',
+  'ComboSearch is not defined',
+  'http://loading.retry.widdit.com/',
+  'atomicFindClose',
+  // Facebook borked
+  'fb_xd_fragment',
+  // ISP "optimizing" proxy - `Cache-Control: no-transform` seems to
+  // reduce this. (thanks @acdha)
+  // See http://stackoverflow.com/questions/4113268
+  'bmi_SafeAddOnload',
+  'EBCallBackMessageReceived',
+  // See http://toolbar.conduit.com/Developer/HtmlAndGadget/Methods/JSInjection.aspx
+  'conduitPage',
+];
+
+const IGNORE_URLS = [
+  // Facebook flakiness
+  /graph\.facebook\.com/i,
+  // Facebook blocked
+  /connect\.facebook\.net\/en_US\/all\.js/i,
+  // Woopra flakiness
+  /eatdifferent\.com\.woopra-ns\.com/i,
+  /static\.woopra\.com\/js\/woopra\.js/i,
+  // Chrome extensions
+  /extensions\//i,
+  /^chrome:\/\//i,
+  // Other plugins
+  /127\.0\.0\.1:4001\/isrunning/i,  // Cacaoweb
+  /webappstoolbarba\.texthelp\.com\//i,
+  /metrics\.itunes\.apple\.com\.edgesuite\.net\//i,
+];
+
+const SAMPLE_RATE = 95;
+
 const RavenConfig = {
+  IGNORE_ERRORS,
+  IGNORE_URLS,
+  SAMPLE_RATE,
   init(options = {}) {
     this.options = options;
 
@@ -13,6 +59,9 @@ const RavenConfig = {
     Raven.config(this.options.sentryDsn, {
       whitelistUrls: this.options.whitelistUrls,
       environment: this.options.isProduction ? 'production' : 'development',
+      ignoreErrors: this.IGNORE_ERRORS,
+      ignoreUrls: this.IGNORE_URLS,
+      shouldSendCallback: this.shouldSendSample,
     }).install();
   },
 
@@ -42,6 +91,10 @@ const RavenConfig = {
       },
     });
   },
+
+  shouldSendSample() {
+    return Math.random() * 100 <= this.SAMPLE_RATE;
+  },
 };
 
 export default RavenConfig;
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index 55f3949c74008..78251b4f61b6c 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -2,6 +2,28 @@ import Raven from 'raven-js';
 import RavenConfig from '~/raven/raven_config';
 
 describe('RavenConfig', () => {
+  describe('IGNORE_ERRORS', () => {
+    it('should be an array of strings', () => {
+      const areStrings = RavenConfig.IGNORE_ERRORS.every(error => typeof error === 'string');
+
+      expect(areStrings).toBe(true);
+    });
+  });
+
+  describe('IGNORE_URLS', () => {
+    it('should be an array of regexps', () => {
+      const areRegExps = RavenConfig.IGNORE_URLS.every(url => url instanceof RegExp);
+
+      expect(areRegExps).toBe(true);
+    });
+  });
+
+  describe('SAMPLE_RATE', () => {
+    it('should be a finite number', () => {
+      expect(typeof RavenConfig.SAMPLE_RATE).toEqual('number');
+    });
+  });
+
   describe('init', () => {
     let options;
 
@@ -76,6 +98,9 @@ describe('RavenConfig', () => {
       expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
         whitelistUrls: options.whitelistUrls,
         environment: 'production',
+        ignoreErrors: Raven.IGNORE_ERRORS,
+        ignoreUrls: Raven.IGNORE_URLS,
+        shouldSendCallback: Raven.shouldSendSample,
       });
     });
 
@@ -93,6 +118,9 @@ describe('RavenConfig', () => {
       expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
         whitelistUrls: options.whitelistUrls,
         environment: 'development',
+        ignoreErrors: Raven.IGNORE_ERRORS,
+        ignoreUrls: Raven.IGNORE_URLS,
+        shouldSendCallback: Raven.shouldSendSample,
       });
     });
   });
@@ -213,4 +241,38 @@ describe('RavenConfig', () => {
       });
     });
   });
+
+  describe('shouldSendSample', () => {
+    let randomNumber;
+
+    beforeEach(() => {
+      RavenConfig.SAMPLE_RATE = 50;
+
+      spyOn(Math, 'random').and.callFake(() => randomNumber);
+    });
+
+    it('should call Math.random', () => {
+      RavenConfig.shouldSendSample();
+
+      expect(Math.random).toHaveBeenCalled();
+    });
+
+    it('should return true if the sample rate is greater than the random number * 100', () => {
+      randomNumber = 0.1;
+
+      expect(RavenConfig.shouldSendSample()).toBe(true);
+    });
+
+    it('should return false if the sample rate is less than the random number * 100', () => {
+      randomNumber = 0.9;
+
+      expect(RavenConfig.shouldSendSample()).toBe(false);
+    });
+
+    it('should return true if the sample rate is equal to the random number * 100', () => {
+      randomNumber = 0.5;
+
+      expect(RavenConfig.shouldSendSample()).toBe(true);
+    });
+  });
 });
-- 
GitLab


From 7d16537cac7eb808d8d18cd0b89475db4e4eeaaa Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 19 Apr 2017 16:53:06 +0100
Subject: [PATCH 055/363] Created a plural filter Added tests for the filter

[ci skip]
---
 .../components/limit_warning_component.js     |  2 +-
 .../components/total_time_component.js        |  4 +-
 .../javascripts/vue_shared/translate.js       |  3 +
 spec/javascripts/vue_shared/translate_spec.js | 90 +++++++++++++++++++
 4 files changed, 96 insertions(+), 3 deletions(-)
 create mode 100644 spec/javascripts/vue_shared/translate_spec.js

diff --git a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
index a7b187a0a36ba..63e20478e9494 100644
--- a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
@@ -11,7 +11,7 @@ export default {
           aria-hidden="true"
           title="Limited to showing 50 events at most"
           data-placement="top"></i>
-      {{ 'Showing 50 events' | translate }}
+      {{ 'Showing %d event' | translate-plural('Showing %d events', 50) }}
     </span>
   `,
 };
diff --git a/app/assets/javascripts/cycle_analytics/components/total_time_component.js b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
index e07963b1b45c6..a0d735f159ce0 100644
--- a/app/assets/javascripts/cycle_analytics/components/total_time_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
@@ -12,9 +12,9 @@ global.cycleAnalytics.TotalTimeComponent = Vue.extend({
   template: `
     <span class="total-time">
       <template v-if="Object.keys(time).length">
-        <template v-if="time.days">{{ time.days }} <span>{{ time.days === 1 ? 'day' : 'days' | translate }}</span></template>
+        <template v-if="time.days">{{ time.days }} <span>{{ 'day' | translate-plural('days', time.days) }}</span></template>
         <template v-if="time.hours">{{ time.hours }} <span v-translate>hr</span></template>
-        <template v-if="time.mins && !time.days">{{ time.mins }} <span v-translate>mins</span></template>
+        <template v-if="time.mins && !time.days">{{ time.mins }} <span>{{ 'min' | translate-plural('mins', time.mins) }}</span></template>
         <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>s</span></template>
       </template>
       <template v-else>
diff --git a/app/assets/javascripts/vue_shared/translate.js b/app/assets/javascripts/vue_shared/translate.js
index 88b7c0bb95436..072828b310ece 100644
--- a/app/assets/javascripts/vue_shared/translate.js
+++ b/app/assets/javascripts/vue_shared/translate.js
@@ -3,6 +3,9 @@ import locale from '../locale';
 export default (Vue) => {
   Vue.filter('translate', text => locale.gettext(text));
 
+  Vue.filter('translate-plural', (text, pluralText, count) =>
+    locale.ngettext(text, pluralText, count).replace(/%d/g, count));
+
   Vue.directive('translate', {
     bind(el) {
       const $el = el;
diff --git a/spec/javascripts/vue_shared/translate_spec.js b/spec/javascripts/vue_shared/translate_spec.js
new file mode 100644
index 0000000000000..74bd4ff86b1bb
--- /dev/null
+++ b/spec/javascripts/vue_shared/translate_spec.js
@@ -0,0 +1,90 @@
+import Vue from 'vue';
+import Translate from '~/vue_shared/translate';
+
+Vue.use(Translate);
+
+describe('Vue translate filter', () => {
+  let el;
+
+  beforeEach(() => {
+    el = document.createElement('div');
+
+    document.body.appendChild(el);
+  });
+
+  it('translate single text', (done) => {
+    const comp = new Vue({
+      el,
+      template: `
+        <span>
+          {{ 'testing' | translate }}
+        </span>
+      `,
+    }).$mount();
+
+    Vue.nextTick(() => {
+      expect(
+        comp.$el.textContent.trim(),
+      ).toBe('testing');
+
+      done();
+    });
+  });
+
+  it('translate plural text with single count', (done) => {
+    const comp = new Vue({
+      el,
+      template: `
+        <span>
+          {{ '%d day' | translate-plural('%d days', 1) }}
+        </span>
+      `,
+    }).$mount();
+
+    Vue.nextTick(() => {
+      expect(
+        comp.$el.textContent.trim(),
+      ).toBe('1 day');
+
+      done();
+    });
+  });
+
+  it('translate plural text with multiple count', (done) => {
+    const comp = new Vue({
+      el,
+      template: `
+        <span>
+          {{ '%d day' | translate-plural('%d days', 2) }}
+        </span>
+      `,
+    }).$mount();
+
+    Vue.nextTick(() => {
+      expect(
+        comp.$el.textContent.trim(),
+      ).toBe('2 days');
+
+      done();
+    });
+  });
+
+  it('translate plural without replacing any text', (done) => {
+    const comp = new Vue({
+      el,
+      template: `
+        <span>
+          {{ 'day' | translate-plural('days', 2) }}
+        </span>
+      `,
+    }).$mount();
+
+    Vue.nextTick(() => {
+      expect(
+        comp.$el.textContent.trim(),
+      ).toBe('days');
+
+      done();
+    });
+  });
+});
-- 
GitLab


From 1de135bc0408a871e3bfa8b0ba5aa81a7936bd01 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 19 Apr 2017 23:19:24 -0500
Subject: [PATCH 056/363] Fix Rubocop complains plus some small refactor

---
 app/controllers/application_controller.rb              |  1 +
 app/views/profiles/show.html.haml                      |  2 +-
 config/initializers/fast_gettext.rb                    |  2 +-
 .../20170413035209_add_preferred_language_to_users.rb  |  6 +++++-
 lib/gitlab/i18n.rb                                     | 10 +++++-----
 5 files changed, 13 insertions(+), 8 deletions(-)

diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index 5a3bd4040cc03..0af05992cceb5 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -269,6 +269,7 @@ def u2f_app_id
   def set_locale
     requested_locale = current_user&.preferred_language || request.env['HTTP_ACCEPT_LANGUAGE'] || I18n.default_locale
     locale = FastGettext.set_locale(requested_locale)
+
     I18n.locale = locale
   end
 end
diff --git a/app/views/profiles/show.html.haml b/app/views/profiles/show.html.haml
index d8ef64ceb7288..9846f01603fd2 100644
--- a/app/views/profiles/show.html.haml
+++ b/app/views/profiles/show.html.haml
@@ -74,7 +74,7 @@
         %span.help-block This email will be displayed on your public profile.
       .form-group
         = f.label :preferred_language, class: "label-light"
-        = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES.map { |lang| [_(lang[0]), lang[1]] },
+        = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES.map { |value, label| [_(label), value] },
           {}, class: "select2"
       .form-group
         = f.label :skype, class: "label-light"
diff --git a/config/initializers/fast_gettext.rb b/config/initializers/fast_gettext.rb
index cfc3427b5ae69..54b0049033b7e 100644
--- a/config/initializers/fast_gettext.rb
+++ b/config/initializers/fast_gettext.rb
@@ -1,3 +1,3 @@
 FastGettext.add_text_domain 'gitlab', path: 'locale', type: :po
-FastGettext.default_available_locales = ['en', 'es','de']
+FastGettext.default_available_locales = Gitlab::I18n::AVAILABLE_LANGUAGES.keys
 FastGettext.default_text_domain = 'gitlab'
diff --git a/db/migrate/20170413035209_add_preferred_language_to_users.rb b/db/migrate/20170413035209_add_preferred_language_to_users.rb
index 5dc128dbaea0a..6fe91656eeba5 100644
--- a/db/migrate/20170413035209_add_preferred_language_to_users.rb
+++ b/db/migrate/20170413035209_add_preferred_language_to_users.rb
@@ -8,7 +8,11 @@ class AddPreferredLanguageToUsers < ActiveRecord::Migration
 
   disable_ddl_transaction!
 
-  def change
+  def up
     add_column_with_default :users, :preferred_language, :string, default: 'en'
   end
+
+  def down
+    remove_column :users, :preferred_language
+  end
 end
diff --git a/lib/gitlab/i18n.rb b/lib/gitlab/i18n.rb
index 85e527afd7148..0459def651727 100644
--- a/lib/gitlab/i18n.rb
+++ b/lib/gitlab/i18n.rb
@@ -1,9 +1,9 @@
 module Gitlab
   module I18n
-    AVAILABLE_LANGUAGES = [
-      ['English', 'en'],
-      ['Spanish', 'es'],
-      ['Deutsch', 'de']
-    ]
+    AVAILABLE_LANGUAGES = {
+      'en' => 'English',
+      'es' => 'Spanish',
+      'de' => 'Deutsch'
+    }.freeze
   end
 end
-- 
GitLab


From aa29bd43451f18e79eacaf92af6859a08da8b5a4 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 20 Apr 2017 07:38:28 +0000
Subject: [PATCH 057/363] Include droplab-item-ignore in droplab docs

---
 doc/development/fe_guide/droplab/droplab.md | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/doc/development/fe_guide/droplab/droplab.md b/doc/development/fe_guide/droplab/droplab.md
index 8f0b6b21953a0..112ff3419d975 100644
--- a/doc/development/fe_guide/droplab/droplab.md
+++ b/doc/development/fe_guide/droplab/droplab.md
@@ -183,6 +183,8 @@ For example,
 either by a mouse click or by enter key selection.
 * The `droplab-item-active` css class is added to items that have been selected
 using arrow key navigation.
+* You can add the `droplab-item-ignore` css class to any item that you do not want to be selectable. For example,
+an `<li class="divider"></li>` list divider element that should not be interactive.
 
 ## Internal events
 
-- 
GitLab


From 4471d7b84fc783a08909473ef0243e6e07d2342a Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 20 Apr 2017 18:09:13 -0500
Subject: [PATCH 058/363] Some small fixes for the current I18n implementation

---
 app/controllers/projects/cycle_analytics_controller.rb | 1 +
 config/initializers/fast_gettext.rb                    | 2 +-
 lib/gitlab/i18n.rb                                     | 6 +++---
 3 files changed, 5 insertions(+), 4 deletions(-)

diff --git a/app/controllers/projects/cycle_analytics_controller.rb b/app/controllers/projects/cycle_analytics_controller.rb
index 88ac3ad046b57..7ef8872a90b5b 100644
--- a/app/controllers/projects/cycle_analytics_controller.rb
+++ b/app/controllers/projects/cycle_analytics_controller.rb
@@ -4,6 +4,7 @@ class Projects::CycleAnalyticsController < Projects::ApplicationController
   include CycleAnalyticsParams
 
   before_action :authorize_read_cycle_analytics!
+  before_action :set_locale, only: :show
 
   def show
     @cycle_analytics = ::CycleAnalytics.new(@project, options(cycle_analytics_params))
diff --git a/config/initializers/fast_gettext.rb b/config/initializers/fast_gettext.rb
index 54b0049033b7e..10a3ee02b8510 100644
--- a/config/initializers/fast_gettext.rb
+++ b/config/initializers/fast_gettext.rb
@@ -1,3 +1,3 @@
 FastGettext.add_text_domain 'gitlab', path: 'locale', type: :po
-FastGettext.default_available_locales = Gitlab::I18n::AVAILABLE_LANGUAGES.keys
 FastGettext.default_text_domain = 'gitlab'
+FastGettext.default_available_locales = Gitlab::I18n::AVAILABLE_LANGUAGES.keys
diff --git a/lib/gitlab/i18n.rb b/lib/gitlab/i18n.rb
index 0459def651727..64a86b55c7f5f 100644
--- a/lib/gitlab/i18n.rb
+++ b/lib/gitlab/i18n.rb
@@ -1,9 +1,9 @@
 module Gitlab
   module I18n
     AVAILABLE_LANGUAGES = {
-      'en' => 'English',
-      'es' => 'Spanish',
-      'de' => 'Deutsch'
+      'en' => N_('English'),
+      'es' => N_('Spanish'),
+      'de' => N_('Deutsch')
     }.freeze
   end
 end
-- 
GitLab


From 6fbc6befa1768cc68471d17091ef08b2e73fec82 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 20 Apr 2017 22:47:31 -0500
Subject: [PATCH 059/363] Monkey patch gettext_i18n_rails so it can parse
 content in Mustache format

---
 config/initializers/gettext_rails_i18n_patch.rb | 17 +++++++++++++++++
 1 file changed, 17 insertions(+)
 create mode 100644 config/initializers/gettext_rails_i18n_patch.rb

diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
new file mode 100644
index 0000000000000..0413330c6087a
--- /dev/null
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -0,0 +1,17 @@
+module GettextI18nRails
+  class HamlParser
+    singleton_class.send(:alias_method, :old_convert_to_code, :convert_to_code)
+
+    # We need to convert text in Mustache format
+    # to a format that can be parsed by Gettext scripts.
+    # If we found a content like "{{ 'Stage' | translate }}"
+    # in a HAML file we convert it to "= _('Stage')", that way
+    # it can be processed by the "rake gettext:find" script.
+    def self.convert_to_code(text)
+      text.gsub!(/{{ (.*)( \| translate) }}/, "= _(\\1)")
+
+      old_convert_to_code(text)
+    end
+  end
+end
+
-- 
GitLab


From 4f31d5963f44b1803fa54d0732cae86b0ec45e06 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 21 Apr 2017 09:16:50 +0100
Subject: [PATCH 060/363] Revert line being commented out

---
 .../initializers/gettext_rails_i18n_patch.rb  | 24 +++++++++----------
 1 file changed, 12 insertions(+), 12 deletions(-)

diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
index 744e983a094a7..c6b3795d0c430 100644
--- a/config/initializers/gettext_rails_i18n_patch.rb
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -1,16 +1,16 @@
 module GettextI18nRails
   class HamlParser
-    # singleton_class.send(:alias_method, :old_convert_to_code, :convert_to_code)
-    #
-    # # We need to convert text in Mustache format
-    # # to a format that can be parsed by Gettext scripts.
-    # # If we found a content like "{{ 'Stage' | translate }}"
-    # # in a HAML file we convert it to "= _('Stage')", that way
-    # # it can be processed by the "rake gettext:find" script.
-    # def self.convert_to_code(text)
-    #   text.gsub!(/{{ (.*)( \| translate) }}/, "= _(\\1)")
-    #
-    #   old_convert_to_code(text)
-    # end
+    singleton_class.send(:alias_method, :old_convert_to_code, :convert_to_code)
+
+    # We need to convert text in Mustache format
+    # to a format that can be parsed by Gettext scripts.
+    # If we found a content like "{{ 'Stage' | translate }}"
+    # in a HAML file we convert it to "= _('Stage')", that way
+    # it can be processed by the "rake gettext:find" script.
+    def self.convert_to_code(text)
+      text.gsub!(/{{ (.*)( \| translate) }}/, "= _(\\1)")
+
+      old_convert_to_code(text)
+    end
   end
 end
-- 
GitLab


From 76cdb8ee127942a3e5b88eb7e8a9c199bdf3363e Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Fri, 21 Apr 2017 14:34:11 -0600
Subject: [PATCH 061/363] render description - tasks work - gfm is juicy

---
 app/assets/javascripts/issue_show/index.js    |   6 +-
 .../javascripts/issue_show/issue_title.vue    |  80 -----------
 .../issue_show/issue_title_description.vue    | 128 ++++++++++++++++++
 app/assets/stylesheets/pages/issues.scss      |   9 ++
 app/controllers/projects/issues_controller.rb |   6 +-
 app/views/projects/issues/show.html.haml      |  12 +-
 ...pec.js => issue_title_description_spec.js} |   2 +-
 7 files changed, 149 insertions(+), 94 deletions(-)
 delete mode 100644 app/assets/javascripts/issue_show/issue_title.vue
 create mode 100644 app/assets/javascripts/issue_show/issue_title_description.vue
 rename spec/javascripts/issue_show/{issue_title_spec.js => issue_title_description_spec.js} (88%)

diff --git a/app/assets/javascripts/issue_show/index.js b/app/assets/javascripts/issue_show/index.js
index 4d491e70d8379..db1cdb6d498a6 100644
--- a/app/assets/javascripts/issue_show/index.js
+++ b/app/assets/javascripts/issue_show/index.js
@@ -1,16 +1,16 @@
 import Vue from 'vue';
-import IssueTitle from './issue_title.vue';
+import IssueTitle from './issue_title_description.vue';
 import '../vue_shared/vue_resource_interceptor';
 
 (() => {
   const issueTitleData = document.querySelector('.issue-title-data').dataset;
-  const { initialTitle, endpoint } = issueTitleData;
+  const { candescription, endpoint } = issueTitleData;
 
   const vm = new Vue({
     el: '.issue-title-entrypoint',
     render: createElement => createElement(IssueTitle, {
       props: {
-        initialTitle,
+        candescription,
         endpoint,
       },
     }),
diff --git a/app/assets/javascripts/issue_show/issue_title.vue b/app/assets/javascripts/issue_show/issue_title.vue
deleted file mode 100644
index 00b0e56030ae0..0000000000000
--- a/app/assets/javascripts/issue_show/issue_title.vue
+++ /dev/null
@@ -1,80 +0,0 @@
-<script>
-import Visibility from 'visibilityjs';
-import Poll from './../lib/utils/poll';
-import Service from './services/index';
-
-export default {
-  props: {
-    initialTitle: { required: true, type: String },
-    endpoint: { required: true, type: String },
-  },
-  data() {
-    const resource = new Service(this.$http, this.endpoint);
-
-    const poll = new Poll({
-      resource,
-      method: 'getTitle',
-      successCallback: (res) => {
-        this.renderResponse(res);
-      },
-      errorCallback: (err) => {
-        if (process.env.NODE_ENV !== 'production') {
-          // eslint-disable-next-line no-console
-          console.error('ISSUE SHOW TITLE REALTIME ERROR', err);
-        } else {
-          throw new Error(err);
-        }
-      },
-    });
-
-    return {
-      poll,
-      timeoutId: null,
-      title: this.initialTitle,
-    };
-  },
-  methods: {
-    renderResponse(res) {
-      const body = JSON.parse(res.body);
-      this.triggerAnimation(body);
-    },
-    triggerAnimation(body) {
-      const { title } = body;
-
-      /**
-      * since opacity is changed, even if there is no diff for Vue to update
-      * we must check the title even on a 304 to ensure no visual change
-      */
-      if (this.title === title) return;
-
-      this.$el.style.opacity = 0;
-
-      this.timeoutId = setTimeout(() => {
-        this.title = title;
-
-        this.$el.style.transition = 'opacity 0.2s ease';
-        this.$el.style.opacity = 1;
-
-        clearTimeout(this.timeoutId);
-      }, 100);
-    },
-  },
-  created() {
-    if (!Visibility.hidden()) {
-      this.poll.makeRequest();
-    }
-
-    Visibility.change(() => {
-      if (!Visibility.hidden()) {
-        this.poll.restart();
-      } else {
-        this.poll.stop();
-      }
-    });
-  },
-};
-</script>
-
-<template>
-  <h2 class="title" v-html="title"></h2>
-</template>
diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
new file mode 100644
index 0000000000000..4605fdadf8d75
--- /dev/null
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -0,0 +1,128 @@
+<script>
+import Visibility from 'visibilityjs';
+import Poll from './../lib/utils/poll';
+import Service from './services/index';
+
+export default {
+  props: {
+    endpoint: { required: true, type: String },
+    candescription: { required: true, type: String },
+  },
+  data() {
+    const resource = new Service(this.$http, this.endpoint);
+
+    const poll = new Poll({
+      resource,
+      method: 'getTitle',
+      successCallback: (res) => {
+        this.renderResponse(res);
+      },
+      errorCallback: (err) => {
+        if (process.env.NODE_ENV !== 'production') {
+          // eslint-disable-next-line no-console
+          console.error('ISSUE SHOW TITLE REALTIME ERROR', err);
+        } else {
+          throw new Error(err);
+        }
+      },
+    });
+
+    return {
+      poll,
+      timeoutId: null,
+      title: null,
+      description: null,
+    };
+  },
+  methods: {
+    renderResponse(res) {
+      const body = JSON.parse(res.body);
+      this.triggerAnimation(body);
+    },
+    triggerAnimation(body) {
+      const { title, description } = body;
+
+      this.descriptionText = body.description_text;
+
+      /**
+      * since opacity is changed, even if there is no diff for Vue to update
+      * we must check the title even on a 304 to ensure no visual change
+      */
+      const noTitleChange = this.title === title;
+      const noDescriptionChange = this.description === description;
+
+      if (noTitleChange && noDescriptionChange) return;
+
+      const elementsToVisualize = [];
+
+      if (!noTitleChange) {
+        elementsToVisualize.push(this.$el.querySelector('.title'));
+      } else if (!noDescriptionChange) {
+        elementsToVisualize.push(this.$el.querySelector('.wiki'));
+      }
+
+      elementsToVisualize.forEach((element) => {
+        element.classList.remove('issue-realtime-trigger-pulse');
+        element.classList.add('issue-realtime-pre-pulse');
+      });
+
+      this.timeoutId = setTimeout(() => {
+        this.title = title;
+        this.description = description;
+
+        elementsToVisualize.forEach((element) => {
+          element.classList.remove('issue-realtime-pre-pulse');
+          element.classList.add('issue-realtime-trigger-pulse');
+        });
+
+        clearTimeout(this.timeoutId);
+      }, 100);
+    },
+  },
+  computed: {
+    descriptionClass() {
+      return `description ${this.candescription} is-task-list-enabled`;
+    },
+  },
+  created() {
+    if (!Visibility.hidden()) {
+      this.poll.makeRequest();
+    }
+
+    Visibility.change(() => {
+      if (!Visibility.hidden()) {
+        this.poll.restart();
+      } else {
+        this.poll.stop();
+      }
+    });
+  },
+  updated() {
+    new gl.TaskList({
+      dataType: 'issue',
+      fieldName: 'description',
+      selector: '.detail-page-description',
+    }).init();
+
+    $(this.$refs['issue-content-container-gfm-entry']).renderGFM();
+  },
+};
+</script>
+
+<template>
+  <div>
+    <h2 class="title issue-realtime-trigger-pulse" v-html="title"></h2>
+    <div
+      :class="descriptionClass"
+      v-if="description"
+    >
+      <div
+        class="wiki issue-realtime-trigger-pulse"
+        v-html="description"
+        ref="issue-content-container-gfm-entry"
+      >
+      </div>
+      <textarea class="hidden js-task-list-field" v-if="descriptionText">{{descriptionText}}</textarea>
+    </div>
+  </div>
+</template>
diff --git a/app/assets/stylesheets/pages/issues.scss b/app/assets/stylesheets/pages/issues.scss
index b2f45625a2a46..1311291661547 100644
--- a/app/assets/stylesheets/pages/issues.scss
+++ b/app/assets/stylesheets/pages/issues.scss
@@ -18,6 +18,15 @@
   }
 }
 
+.issue-realtime-pre-pulse {
+  opacity: 0;
+}
+
+.issue-realtime-trigger-pulse {
+  transition: opacity 0.2s ease;
+  opacity: 1;
+}
+
 .check-all-holder {
   line-height: 36px;
   float: left;
diff --git a/app/controllers/projects/issues_controller.rb b/app/controllers/projects/issues_controller.rb
index cbf6713726178..b1df50ba84902 100644
--- a/app/controllers/projects/issues_controller.rb
+++ b/app/controllers/projects/issues_controller.rb
@@ -198,7 +198,11 @@ def can_create_branch
 
   def rendered_title
     Gitlab::PollingInterval.set_header(response, interval: 3_000)
-    render json: { title: view_context.markdown_field(@issue, :title) }
+    render json: {
+      title: view_context.markdown_field(@issue, :title),
+      description: view_context.markdown_field(@issue, :description),
+      description_text: @issue.description,
+    }
   end
 
   protected
diff --git a/app/views/projects/issues/show.html.haml b/app/views/projects/issues/show.html.haml
index fcbd88295958e..bf98efbdfdfee 100644
--- a/app/views/projects/issues/show.html.haml
+++ b/app/views/projects/issues/show.html.haml
@@ -51,17 +51,11 @@
 
 .issue-details.issuable-details
   .detail-page-description.content-block{ class: ('hide-bottom-border' unless @issue.description.present? ) }
-    .issue-title-data.hidden{ "data" => { "initial-title" => markdown_field(@issue, :title),
-      "endpoint" => rendered_title_namespace_project_issue_path(@project.namespace, @project, @issue),
+    .issue-title-data.hidden{ "data" => { "endpoint" => rendered_title_namespace_project_issue_path(@project.namespace, @project, @issue),
+      "canDescription" => can?(current_user, :update_issue, @issue) ? 'js-task-list-container' : '',
     } }
     .issue-title-entrypoint
-    - if @issue.description.present?
-      .description{ class: can?(current_user, :update_issue, @issue) ? 'js-task-list-container' : '' }
-        .wiki
-          = preserve do
-            = markdown_field(@issue, :description)
-        %textarea.hidden.js-task-list-field
-          = @issue.description
+
     = edited_time_ago_with_tooltip(@issue, placement: 'bottom', html_class: 'issue_edited_ago')
 
     #merge-requests{ data: { url: referenced_merge_requests_namespace_project_issue_url(@project.namespace, @project, @issue) } }
diff --git a/spec/javascripts/issue_show/issue_title_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
similarity index 88%
rename from spec/javascripts/issue_show/issue_title_spec.js
rename to spec/javascripts/issue_show/issue_title_description_spec.js
index 03edbf9f94742..f351ca7d8e657 100644
--- a/spec/javascripts/issue_show/issue_title_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -1,5 +1,5 @@
 import Vue from 'vue';
-import issueTitle from '~/issue_show/issue_title.vue';
+import issueTitle from '~/issue_show/issue_title_description.vue';
 
 describe('Issue Title', () => {
   let IssueTitleComponent;
-- 
GitLab


From 52bfc0efa95f991f17618aa049f799f4e44e13ac Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Sat, 22 Apr 2017 21:02:31 +0200
Subject: [PATCH 062/363] Fix typo in CI/CD build partial view

---
 app/views/projects/ci/builds/_build.html.haml | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/views/projects/ci/builds/_build.html.haml b/app/views/projects/ci/builds/_build.html.haml
index 3e1c8f25deac3..2227d36eed22d 100644
--- a/app/views/projects/ci/builds/_build.html.haml
+++ b/app/views/projects/ci/builds/_build.html.haml
@@ -102,7 +102,7 @@
           = link_to cancel_namespace_project_build_path(job.project.namespace, job.project, job, return_to: request.original_url), method: :post, title: 'Cancel', class: 'btn btn-build' do
             = icon('remove', class: 'cred')
         - elsif allow_retry
-          - if job.playable? && !admin && can?(current_user, :play_build, jop)
+          - if job.playable? && !admin && can?(current_user, :play_build, job)
             = link_to play_namespace_project_build_path(job.project.namespace, job.project, job, return_to: request.original_url), method: :post, title: 'Play', class: 'btn btn-build' do
               = custom_icon('icon_play')
           - elsif job.retryable?
-- 
GitLab


From b34534b60a593f2416fd7854f5fa08a84af9cedf Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Mon, 24 Apr 2017 13:37:14 +0100
Subject: [PATCH 063/363] Fixed dodgy merge

---
 README.md                                     |  2 +-
 app/assets/stylesheets/framework/blocks.scss  |  1 +
 app/assets/stylesheets/framework/nav.scss     |  2 +-
 .../users/migrate_to_ghost_user_service.rb    | 34 +++-----
 ...01-add-slash-slack-commands-to-api-doc.yml |  5 --
 ...tion-while-moving-issues-to-ghost-user.yml |  4 -
 changelogs/unreleased/fix_link_in_readme.yml  |  4 -
 doc/administration/integration/plantuml.md    |  2 +-
 doc/api/services.md                           | 79 +++----------------
 doc/development/fe_guide/testing.md           |  6 +-
 .../migrate_to_ghost_user_service_spec.rb     | 18 -----
 ...e_to_ghost_user_service_shared_examples.rb | 52 ------------
 12 files changed, 28 insertions(+), 181 deletions(-)
 delete mode 100644 changelogs/unreleased/29801-add-slash-slack-commands-to-api-doc.yml
 delete mode 100644 changelogs/unreleased/30306-transaction-while-moving-issues-to-ghost-user.yml
 delete mode 100644 changelogs/unreleased/fix_link_in_readme.yml

diff --git a/README.md b/README.md
index 10d69efdc6bf3..f0e3b52ef6f06 100644
--- a/README.md
+++ b/README.md
@@ -73,7 +73,7 @@ One small thing you also have to do when installing it yourself is to copy the e
 
     cp config/unicorn.rb.example.development config/unicorn.rb
 
-Instructions on how to start GitLab and how to run the tests can be found in the [getting started section of the GitLab Development Kit](https://gitlab.com/gitlab-org/gitlab-development-kit#getting-started).
+Instructions on how to start GitLab and how to run the tests can be found in the [development section of the GitLab Development Kit](https://gitlab.com/gitlab-org/gitlab-development-kit#development).
 
 ## Software stack
 
diff --git a/app/assets/stylesheets/framework/blocks.scss b/app/assets/stylesheets/framework/blocks.scss
index f3e2a5db0a64d..524252629254e 100644
--- a/app/assets/stylesheets/framework/blocks.scss
+++ b/app/assets/stylesheets/framework/blocks.scss
@@ -230,6 +230,7 @@
     float: right;
     margin-top: 8px;
     padding-bottom: 8px;
+    border-bottom: 1px solid $border-color;
   }
 }
 
diff --git a/app/assets/stylesheets/framework/nav.scss b/app/assets/stylesheets/framework/nav.scss
index b6cf5101d6094..e6d808717f37f 100644
--- a/app/assets/stylesheets/framework/nav.scss
+++ b/app/assets/stylesheets/framework/nav.scss
@@ -110,7 +110,7 @@
 
 .top-area {
   @include clearfix;
-  border-bottom: 1px solid $border-color;
+  border-bottom: 1px solid $white-normal;
 
   .nav-text {
     padding-top: 16px;
diff --git a/app/services/users/migrate_to_ghost_user_service.rb b/app/services/users/migrate_to_ghost_user_service.rb
index 4628c4c6f6e7d..1e1ed1791ec7e 100644
--- a/app/services/users/migrate_to_ghost_user_service.rb
+++ b/app/services/users/migrate_to_ghost_user_service.rb
@@ -15,39 +15,27 @@ def initialize(user)
     end
 
     def execute
-      transition = user.block_transition
+      # Block the user before moving records to prevent a data race.
+      # For example, if the user creates an issue after `migrate_issues`
+      # runs and before the user is destroyed, the destroy will fail with
+      # an exception.
+      user.block
 
       user.transaction do
-        # Block the user before moving records to prevent a data race.
-        # For example, if the user creates an issue after `migrate_issues`
-        # runs and before the user is destroyed, the destroy will fail with
-        # an exception.
-        user.block
-
-        # Reverse the user block if record migration fails
-        if !migrate_records && transition
-          transition.rollback
-          user.save!
-        end
-      end
-
-      user.reload
-    end
-
-    private
-
-    def migrate_records
-      user.transaction(requires_new: true) do
         @ghost_user = User.ghost
 
         migrate_issues
         migrate_merge_requests
         migrate_notes
         migrate_abuse_reports
-        migrate_award_emojis
+        migrate_award_emoji
       end
+
+      user.reload
     end
 
+    private
+
     def migrate_issues
       user.issues.update_all(author_id: ghost_user.id)
     end
@@ -64,7 +52,7 @@ def migrate_abuse_reports
       user.reported_abuse_reports.update_all(reporter_id: ghost_user.id)
     end
 
-    def migrate_award_emojis
+    def migrate_award_emoji
       user.award_emoji.update_all(user_id: ghost_user.id)
     end
   end
diff --git a/changelogs/unreleased/29801-add-slash-slack-commands-to-api-doc.yml b/changelogs/unreleased/29801-add-slash-slack-commands-to-api-doc.yml
deleted file mode 100644
index 9c5df690085c8..0000000000000
--- a/changelogs/unreleased/29801-add-slash-slack-commands-to-api-doc.yml
+++ /dev/null
@@ -1,5 +0,0 @@
----
-title: Add Slack slash command api to services documentation and rearrange order and
-  cases
-merge_request: 10757
-author: TM Lee
diff --git a/changelogs/unreleased/30306-transaction-while-moving-issues-to-ghost-user.yml b/changelogs/unreleased/30306-transaction-while-moving-issues-to-ghost-user.yml
deleted file mode 100644
index 5fc57e44be617..0000000000000
--- a/changelogs/unreleased/30306-transaction-while-moving-issues-to-ghost-user.yml
+++ /dev/null
@@ -1,4 +0,0 @@
----
-title: Add a transaction around move_issues_to_ghost_user
-merge_request: 10465
-author:
diff --git a/changelogs/unreleased/fix_link_in_readme.yml b/changelogs/unreleased/fix_link_in_readme.yml
deleted file mode 100644
index be5ceac86563e..0000000000000
--- a/changelogs/unreleased/fix_link_in_readme.yml
+++ /dev/null
@@ -1,4 +0,0 @@
----
-title: Fix dead link to GDK on the README page
-merge_request:
-author: Dino Maric
diff --git a/doc/administration/integration/plantuml.md b/doc/administration/integration/plantuml.md
index b21817c1fd31a..5c85683503913 100644
--- a/doc/administration/integration/plantuml.md
+++ b/doc/administration/integration/plantuml.md
@@ -28,7 +28,7 @@ using Tomcat:
 sudo apt-get install tomcat7
 sudo cp target/plantuml.war /var/lib/tomcat7/webapps/plantuml.war
 sudo chown tomcat7:tomcat7 /var/lib/tomcat7/webapps/plantuml.war
-sudo service tomcat7 restart
+sudo service restart tomcat7
 ```
 
 Once the Tomcat service restarts the PlantUML service will be ready and
diff --git a/doc/api/services.md b/doc/api/services.md
index 0f42c2560992f..7d4779f113763 100644
--- a/doc/api/services.md
+++ b/doc/api/services.md
@@ -490,98 +490,41 @@ Remove all previously JIRA settings from a project.
 DELETE /projects/:id/services/jira
 ```
 
-## Slack slash commands
+## Mattermost Slash Commands
 
-Ability to receive slash commands from a Slack chat instance.
-
-### Get Slack slash command service settings
-
-Get Slack slash command service settings for a project.
-
-```
-GET /projects/:id/services/slack-slash-commands
-```
-
-Example response:
-
-```json
-{
-  "id": 4,
-  "title": "Slack slash commands",
-  "created_at": "2017-06-27T05:51:39-07:00",
-  "updated_at": "2017-06-27T05:51:39-07:00",
-  "active": true,
-  "push_events": true,
-  "issues_events": true,
-  "merge_requests_events": true,
-  "tag_push_events": true,
-  "note_events": true,
-  "build_events": true,
-  "pipeline_events": true,
-  "properties": {
-    "token": "9koXpg98eAheJpvBs5tK"
-  }
-}
-```
+Ability to receive slash commands from a Mattermost chat instance.
 
-### Create/Edit Slack slash command service
+### Create/Edit Mattermost Slash Command service
 
-Set Slack slash command for a project.
+Set Mattermost Slash Command for a project.
 
 ```
-PUT /projects/:id/services/slack-slash-commands
+PUT /projects/:id/services/mattermost-slash-commands
 ```
 
 Parameters:
 
 | Attribute | Type | Required | Description |
 | --------- | ---- | -------- | ----------- |
-| `token` | string | yes | The Slack token |
+| `token` | string | yes | The Mattermost token |
 
 
-### Delete Slack slash command service
+### Delete Mattermost Slash Command service
 
-Delete Slack slash command service for a project.
+Delete Mattermost Slash Command service for a project.
 
 ```
-DELETE /projects/:id/services/slack-slash-commands
+DELETE /projects/:id/services/mattermost-slash-commands
 ```
 
-## Mattermost slash commands
-
-Ability to receive slash commands from a Mattermost chat instance.
-
-### Get Mattermost slash command service settings
+### Get Mattermost Slash Command service settings
 
-Get Mattermost slash command service settings for a project.
+Get Mattermost Slash Command service settings for a project.
 
 ```
 GET /projects/:id/services/mattermost-slash-commands
 ```
 
-### Create/Edit Mattermost slash command service
-
-Set Mattermost slash command for a project.
-
-```
-PUT /projects/:id/services/mattermost-slash-commands
-```
-
-Parameters:
-
-| Attribute | Type | Required | Description |
-| --------- | ---- | -------- | ----------- |
-| `token` | string | yes | The Mattermost token |
-
-
-### Delete Mattermost slash command service
-
-Delete Mattermost slash command service for a project.
-
-```
-DELETE /projects/:id/services/mattermost-slash-commands
-```
-
 ## Pipeline-Emails
 
 Get emails for GitLab CI pipelines.
diff --git a/doc/development/fe_guide/testing.md b/doc/development/fe_guide/testing.md
index 157c13352ca07..66afbf4db4d64 100644
--- a/doc/development/fe_guide/testing.md
+++ b/doc/development/fe_guide/testing.md
@@ -14,10 +14,8 @@ for more information on general testing practices at GitLab.
 
 GitLab uses the [Karma][karma] test runner with [Jasmine][jasmine] as its test
 framework for our JavaScript unit tests. For tests that rely on DOM
-manipulation, we generate HTML files using RSpec suites (see `spec/javascripts/fixtures/*.rb` for examples).
-Some fixtures are still HAML templates that are translated to HTML files using the same mechanism (see `static_fixtures.rb`).
-Those will be migrated over time.
-Fixtures are served during testing by the [jasmine-jquery][jasmine-jquery] plugin.
+manipulation we use fixtures which are pre-compiled from HAML source files and
+served during testing by the [jasmine-jquery][jasmine-jquery] plugin.
 
 JavaScript tests live in `spec/javascripts/`, matching the folder structure
 of `app/assets/javascripts/`: `app/assets/javascripts/behaviors/autosize.js`
diff --git a/spec/services/users/migrate_to_ghost_user_service_spec.rb b/spec/services/users/migrate_to_ghost_user_service_spec.rb
index 9e1edf1ac30d8..8c5b7e41c1550 100644
--- a/spec/services/users/migrate_to_ghost_user_service_spec.rb
+++ b/spec/services/users/migrate_to_ghost_user_service_spec.rb
@@ -60,23 +60,5 @@
         end
       end
     end
-
-    context "when record migration fails with a rollback exception" do
-      before do
-        expect_any_instance_of(MergeRequest::ActiveRecord_Associations_CollectionProxy)
-          .to receive(:update_all).and_raise(ActiveRecord::Rollback)
-      end
-
-      context "for records that were already migrated" do
-        let!(:issue) { create(:issue, project: project, author: user) }
-        let!(:merge_request) { create(:merge_request, source_project: project, author: user, target_branch: "first") }
-
-        it "reverses the migration" do
-          service.execute
-
-          expect(issue.reload.author).to eq(user)
-        end
-      end
-    end
   end
 end
diff --git a/spec/support/services/migrate_to_ghost_user_service_shared_examples.rb b/spec/support/services/migrate_to_ghost_user_service_shared_examples.rb
index dcc562c684b0f..0eac587e9730a 100644
--- a/spec/support/services/migrate_to_ghost_user_service_shared_examples.rb
+++ b/spec/support/services/migrate_to_ghost_user_service_shared_examples.rb
@@ -35,57 +35,5 @@
 
       expect(user).to be_blocked
     end
-
-    context "race conditions" do
-      context "when #{record_class_name} migration fails and is rolled back" do
-        before do
-          expect_any_instance_of(record_class::ActiveRecord_Associations_CollectionProxy)
-            .to receive(:update_all).and_raise(ActiveRecord::Rollback)
-        end
-
-        it 'rolls back the user block' do
-          service.execute
-
-          expect(user.reload).not_to be_blocked
-        end
-
-        it "doesn't unblock an previously-blocked user" do
-          user.block
-
-          service.execute
-
-          expect(user.reload).to be_blocked
-        end
-      end
-
-      context "when #{record_class_name} migration fails with a non-rollback exception" do
-        before do
-          expect_any_instance_of(record_class::ActiveRecord_Associations_CollectionProxy)
-            .to receive(:update_all).and_raise(ArgumentError)
-        end
-
-        it 'rolls back the user block' do
-          service.execute rescue nil
-
-          expect(user.reload).not_to be_blocked
-        end
-
-        it "doesn't unblock an previously-blocked user" do
-          user.block
-
-          service.execute rescue nil
-
-          expect(user.reload).to be_blocked
-        end
-      end
-
-      it "blocks the user before #{record_class_name} migration begins" do
-        expect(service).to receive("migrate_#{record_class_name.parameterize('_')}s".to_sym) do
-          expect(user.reload).to be_blocked
-        end
-
-        service.execute
-      end
-    end
   end
 end
-- 
GitLab


From 3417e97c18bcf28fbe9f7061e9d9036c29931969 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Mon, 24 Apr 2017 15:03:58 -0600
Subject: [PATCH 064/363] fix a few tests (not all) - update title in tab with
 new issue

---
 .../issue_show/issue_title_description.vue    |  8 +++++--
 spec/features/task_lists_spec.rb              | 24 ++++++++++++++++++-
 2 files changed, 29 insertions(+), 3 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 4605fdadf8d75..77562dac74c87 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -68,6 +68,8 @@ export default {
 
       this.timeoutId = setTimeout(() => {
         this.title = title;
+        document.querySelector('title').innerText = title;
+
         this.description = description;
 
         elementsToVisualize.forEach((element) => {
@@ -98,13 +100,15 @@ export default {
     });
   },
   updated() {
-    new gl.TaskList({
+    const tl = new gl.TaskList({
       dataType: 'issue',
       fieldName: 'description',
       selector: '.detail-page-description',
-    }).init();
+    });
 
     $(this.$refs['issue-content-container-gfm-entry']).renderGFM();
+
+    return tl;
   },
 };
 </script>
diff --git a/spec/features/task_lists_spec.rb b/spec/features/task_lists_spec.rb
index a5d14aa19f160..b8b4fb75dd9e9 100644
--- a/spec/features/task_lists_spec.rb
+++ b/spec/features/task_lists_spec.rb
@@ -63,12 +63,16 @@ def visit_issue(project, issue)
   end
 
   describe 'for Issues' do
-    describe 'multiple tasks' do
+    include WaitForVueResource
+
+    describe 'multiple tasks', js: true do
       let!(:issue) { create(:issue, description: markdown, author: user, project: project) }
 
       it 'renders' do
         visit_issue(project, issue)
 
+        wait_for_vue_resource
+
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 6)
         expect(page).to have_selector('ul input[checked]', count: 2)
@@ -79,6 +83,8 @@ def visit_issue(project, issue)
 
         container = '.detail-page-description .description.js-task-list-container'
 
+        wait_for_vue_resource
+
         expect(page).to have_selector(container)
         expect(page).to have_selector("#{container} .wiki .task-list .task-list-item .task-list-item-checkbox")
         expect(page).to have_selector("#{container} .js-task-list-field")
@@ -88,12 +94,17 @@ def visit_issue(project, issue)
 
       it 'is only editable by author' do
         visit_issue(project, issue)
+
+        wait_for_vue_resource
+
         expect(page).to have_selector('.js-task-list-container')
 
         logout(:user)
 
         login_as(user2)
         visit current_path
+
+        wait_for_vue_resource
         expect(page).not_to have_selector('.js-task-list-container')
       end
 
@@ -109,6 +120,8 @@ def visit_issue(project, issue)
       it 'renders' do
         visit_issue(project, issue)
 
+        wait_for_vue_resource
+
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 1)
         expect(page).to have_selector('ul input[checked]', count: 0)
@@ -126,6 +139,8 @@ def visit_issue(project, issue)
       it 'renders' do
         visit_issue(project, issue)
 
+        wait_for_vue_resource
+
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 1)
         expect(page).to have_selector('ul input[checked]', count: 1)
@@ -143,6 +158,9 @@ def visit_issue(project, issue)
       before { visit_issue(project, issue) }
 
       it 'renders' do
+
+        wait_for_vue_resource
+
         expect(page).to have_selector('ul.task-list',      count: 2)
         expect(page).to have_selector('li.task-list-item', count: 7)
         expect(page).to have_selector('ul input[checked]', count: 1)
@@ -152,6 +170,8 @@ def visit_issue(project, issue)
       it 'solves tasks' do
         expect(page).to have_content("2 of 7 tasks completed")
 
+        wait_for_vue_resource
+
         page.find('li.task-list-item', text: 'Task b').find('input').click
         page.find('li.task-list-item ul li.task-list-item', text: 'Task a.2').find('input').click
         page.find('li.task-list-item ol li.task-list-item', text: 'Task 1.1').find('input').click
@@ -160,6 +180,8 @@ def visit_issue(project, issue)
 
         visit_issue(project, issue) # reload to see new system notes
 
+        wait_for_vue_resource
+
         expect(page).to have_content('marked the task Task b as complete')
         expect(page).to have_content('marked the task Task a.2 as complete')
         expect(page).to have_content('marked the task Task 1.1 as complete')
-- 
GitLab


From 77d9e3f9b49821af3ec9b4fbac574c75b02d0351 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Mon, 24 Apr 2017 22:36:14 -0500
Subject: [PATCH 065/363] Monkey patch gettext_i18n_rails so it can parse
 content in Mustache format

---
 .../initializers/gettext_rails_i18n_patch.rb  | 19 +++++++++++++++++++
 1 file changed, 19 insertions(+)
 create mode 100644 config/initializers/gettext_rails_i18n_patch.rb

diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
new file mode 100644
index 0000000000000..c6b841aacd61f
--- /dev/null
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -0,0 +1,19 @@
+require 'gettext_i18n_rails/haml_parser'
+
+module GettextI18nRails
+  class HamlParser
+    singleton_class.send(:alias_method, :old_convert_to_code, :convert_to_code)
+
+    # We need to convert text in Mustache format
+    # to a format that can be parsed by Gettext scripts.
+    # If we found a content like "{{ 'Stage' | translate }}"
+    # in a HAML file we convert it to "= _('Stage')", that way
+    # it can be processed by the "rake gettext:find" script.
+    def self.convert_to_code(text)
+      text.gsub!(/{{ (.*)( \| translate) }}/, "= _(\\1)")
+
+      old_convert_to_code(text)
+    end
+  end
+end
+
-- 
GitLab


From 80d7f4166fec5476c07ca7afe78e3e8a50071f2b Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Mon, 24 Apr 2017 23:03:38 -0500
Subject: [PATCH 066/363] Parse the translate-plural filter in HAML views

---
 config/initializers/gettext_rails_i18n_patch.rb | 4 ++++
 1 file changed, 4 insertions(+)

diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
index a50be2ea27914..3b50501c3daea 100644
--- a/config/initializers/gettext_rails_i18n_patch.rb
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -10,8 +10,12 @@ class HamlParser
     # in a HAML file we convert it to "= _('Stage')", that way
     # it can be processed by the "rake gettext:find" script.
     def self.convert_to_code(text)
+      # {{ 'Stage' | translate }} => = _('Stage')
       text.gsub!(/{{ (.*)( \| translate) }}/, "= _(\\1)")
 
+      # {{ 'user' | translate-plural('users', users.size) }} => = n_('user', 'users', users.size)
+      text.gsub!(/{{ (.*)( \| translate-plural\((.*), (.*)\)) }}/, "= n_(\\1, \\3, \\4)")
+
       old_convert_to_code(text)
     end
   end
-- 
GitLab


From 422c48baa7eeaca025a423c4eb48256c786514ed Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Tue, 25 Apr 2017 00:01:59 -0500
Subject: [PATCH 067/363] Parse translate filters from JS files.

---
 .../initializers/gettext_rails_i18n_patch.rb  | 31 +++++++++++++++++--
 1 file changed, 29 insertions(+), 2 deletions(-)

diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
index 3b50501c3daea..5570b7e31e9e4 100644
--- a/config/initializers/gettext_rails_i18n_patch.rb
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -1,4 +1,8 @@
 require 'gettext_i18n_rails/haml_parser'
+require 'gettext_i18n_rails_js/parser/javascript'
+
+VUE_TRANSLATE_REGEX = /{{ ([^{]*)( \| translate) }}/
+VUE_TRANSLATE_PLURAL_REGEX = /{{ ([^{]*)( \| translate-plural\((.*), (.*)\)) }}/
 
 module GettextI18nRails
   class HamlParser
@@ -9,14 +13,37 @@ class HamlParser
     # If we found a content like "{{ 'Stage' | translate }}"
     # in a HAML file we convert it to "= _('Stage')", that way
     # it can be processed by the "rake gettext:find" script.
+    # Overwrites: https://github.com/grosser/gettext_i18n_rails/blob/8396387a431e0f8ead72fc1cd425cad2fa4992f2/lib/gettext_i18n_rails/haml_parser.rb#L9
     def self.convert_to_code(text)
       # {{ 'Stage' | translate }} => = _('Stage')
-      text.gsub!(/{{ (.*)( \| translate) }}/, "= _(\\1)")
+      text.gsub!(VUE_TRANSLATE_REGEX, "= _(\\1)")
 
       # {{ 'user' | translate-plural('users', users.size) }} => = n_('user', 'users', users.size)
-      text.gsub!(/{{ (.*)( \| translate-plural\((.*), (.*)\)) }}/, "= n_(\\1, \\3, \\4)")
+      text.gsub!(VUE_TRANSLATE_PLURAL_REGEX, "= n_(\\1, \\3, \\4)")
 
       old_convert_to_code(text)
     end
   end
 end
+
+module GettextI18nRailsJs
+  module Parser
+    module Javascript
+      protected
+
+      # Overwrites: https://github.com/webhippie/gettext_i18n_rails_js/blob/46c58db6d2053a4f5f36a0eb024ea706ff5707cb/lib/gettext_i18n_rails_js/parser/javascript.rb#L46
+      def collect_for(value)
+        ::File.open(value) do |f|
+          f.each_line.each_with_index.collect do |line, idx|
+            line.gsub!(VUE_TRANSLATE_REGEX, "__(\\1)")
+            line.gsub!(VUE_TRANSLATE_PLURAL_REGEX, "n__(\\1, \\3, \\4)")
+
+            line.scan(invoke_regex).collect do |function, arguments|
+              yield(function, arguments, idx + 1)
+            end
+          end.inject([], :+).compact
+        end
+      end
+    end
+  end
+end
-- 
GitLab


From b4e2f9f1a711b5c4103d56752f7d968d009b51a5 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 25 Apr 2017 10:31:15 -0600
Subject: [PATCH 068/363] test async nature of issue show

---
 .../issue_show/issue_title_description.vue    |  2 +-
 spec/features/task_lists_spec.rb              |  9 +---
 .../issue_title_description_spec.js           | 41 +++++++++++++++----
 spec/javascripts/issue_show/mock_data.js      |  5 +++
 4 files changed, 41 insertions(+), 16 deletions(-)
 create mode 100644 spec/javascripts/issue_show/mock_data.js

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 77562dac74c87..5ef0037e48837 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -20,7 +20,7 @@ export default {
       errorCallback: (err) => {
         if (process.env.NODE_ENV !== 'production') {
           // eslint-disable-next-line no-console
-          console.error('ISSUE SHOW TITLE REALTIME ERROR', err);
+          console.error('ISSUE SHOW REALTIME ERROR', err);
         } else {
           throw new Error(err);
         }
diff --git a/spec/features/task_lists_spec.rb b/spec/features/task_lists_spec.rb
index b8b4fb75dd9e9..3c341fef25399 100644
--- a/spec/features/task_lists_spec.rb
+++ b/spec/features/task_lists_spec.rb
@@ -70,7 +70,6 @@ def visit_issue(project, issue)
 
       it 'renders' do
         visit_issue(project, issue)
-
         wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
@@ -80,11 +79,10 @@ def visit_issue(project, issue)
 
       it 'contains the required selectors' do
         visit_issue(project, issue)
+        wait_for_vue_resource
 
         container = '.detail-page-description .description.js-task-list-container'
 
-        wait_for_vue_resource
-
         expect(page).to have_selector(container)
         expect(page).to have_selector("#{container} .wiki .task-list .task-list-item .task-list-item-checkbox")
         expect(page).to have_selector("#{container} .js-task-list-field")
@@ -94,16 +92,13 @@ def visit_issue(project, issue)
 
       it 'is only editable by author' do
         visit_issue(project, issue)
-
         wait_for_vue_resource
 
         expect(page).to have_selector('.js-task-list-container')
 
         logout(:user)
-
         login_as(user2)
         visit current_path
-
         wait_for_vue_resource
         expect(page).not_to have_selector('.js-task-list-container')
       end
@@ -119,7 +114,6 @@ def visit_issue(project, issue)
 
       it 'renders' do
         visit_issue(project, issue)
-
         wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
@@ -138,7 +132,6 @@ def visit_issue(project, issue)
 
       it 'renders' do
         visit_issue(project, issue)
-
         wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index f351ca7d8e657..2f81660aeb11a 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -1,22 +1,49 @@
 import Vue from 'vue';
+import $ from 'jquery';
+import '~/render_math';
+import '~/render_gfm';
 import issueTitle from '~/issue_show/issue_title_description.vue';
+import issueShowData from './mock_data';
+
+window.$ = $;
+
+const issueShowInterceptor = (request, next) => {
+  console.log(issueShowData);
+  next(request.respondWith(JSON.stringify(issueShowData), {
+    status: 200,
+  }));
+};
 
 describe('Issue Title', () => {
-  let IssueTitleComponent;
+  const comps = {
+    IssueTitleComponent: {},
+  };
 
   beforeEach(() => {
-    IssueTitleComponent = Vue.extend(issueTitle);
+    comps.IssueTitleComponent = Vue.extend(issueTitle);
+    Vue.http.interceptors.push(issueShowInterceptor);
+  });
+
+  afterEach(() => {
+    Vue.http.interceptors = _.without(
+      Vue.http.interceptors, issueShowInterceptor,
+    );
   });
 
-  it('should render a title', () => {
-    const component = new IssueTitleComponent({
+  it('should render a title', (done) => {
+    const issueShowComponent = new comps.IssueTitleComponent({
       propsData: {
-        initialTitle: 'wow',
+        candescription: '.css-stuff',
         endpoint: '/gitlab-org/gitlab-shell/issues/9/rendered_title',
       },
     }).$mount();
 
-    expect(component.$el.classList).toContain('title');
-    expect(component.$el.innerHTML).toContain('wow');
+    // need setTimeout because actual setTimeout in code :P
+    setTimeout(() => {
+      expect(issueShowComponent.$el.querySelector('.title').innerHTML)
+        .toContain('this is a title');
+      done();
+    }, 300);
+    // 300 is just three times the Vue comps setTimeout to ensure pass
   });
 });
diff --git a/spec/javascripts/issue_show/mock_data.js b/spec/javascripts/issue_show/mock_data.js
new file mode 100644
index 0000000000000..37e86cfe33d5b
--- /dev/null
+++ b/spec/javascripts/issue_show/mock_data.js
@@ -0,0 +1,5 @@
+export default {
+  title: '<p>this is a title</p>',
+  description: '<p>this is a description!</p>',
+  description_text: 'this is a description',
+};
-- 
GitLab


From 6176ca37b8f6cd44c17c2f3e015d18e0b892d4be Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 25 Apr 2017 11:06:43 -0600
Subject: [PATCH 069/363] change comp timeout to 0ms and test timeout to 10ms

---
 .../javascripts/issue_show/issue_title_description.vue       | 2 +-
 spec/javascripts/issue_show/issue_title_description_spec.js  | 5 ++---
 2 files changed, 3 insertions(+), 4 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 5ef0037e48837..c31b980dfdec7 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -78,7 +78,7 @@ export default {
         });
 
         clearTimeout(this.timeoutId);
-      }, 100);
+      }, 0);
     },
   },
   computed: {
diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index 2f81660aeb11a..a06865fa39a06 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -8,7 +8,6 @@ import issueShowData from './mock_data';
 window.$ = $;
 
 const issueShowInterceptor = (request, next) => {
-  console.log(issueShowData);
   next(request.respondWith(JSON.stringify(issueShowData), {
     status: 200,
   }));
@@ -43,7 +42,7 @@ describe('Issue Title', () => {
       expect(issueShowComponent.$el.querySelector('.title').innerHTML)
         .toContain('this is a title');
       done();
-    }, 300);
-    // 300 is just three times the Vue comps setTimeout to ensure pass
+    }, 10);
+    // 10ms is just long enough for the update hook to fire
   });
 });
-- 
GitLab


From a39a5cafecb476460c23b62029064590731940f9 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 25 Apr 2017 11:10:24 -0600
Subject: [PATCH 070/363] update comment explaining visual change check

---
 app/assets/javascripts/issue_show/issue_title_description.vue | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index c31b980dfdec7..af2fe61af2bb1 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -46,7 +46,7 @@ export default {
 
       /**
       * since opacity is changed, even if there is no diff for Vue to update
-      * we must check the title even on a 304 to ensure no visual change
+      * we must check the title/description even on a 304 to ensure no visual change
       */
       const noTitleChange = this.title === title;
       const noDescriptionChange = this.description === description;
-- 
GitLab


From 9b09ff098cc1efd4f90bbe54fa3cbcd5d4f8cef9 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 25 Apr 2017 11:45:11 -0600
Subject: [PATCH 071/363] add more assertions

---
 .../issue_show/issue_title_description_spec.js         | 10 ++++++++--
 1 file changed, 8 insertions(+), 2 deletions(-)

diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index a06865fa39a06..b81e5d74f7ec3 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -13,7 +13,7 @@ const issueShowInterceptor = (request, next) => {
   }));
 };
 
-describe('Issue Title', () => {
+fdescribe('Issue Title', () => {
   const comps = {
     IssueTitleComponent: {},
   };
@@ -39,8 +39,14 @@ describe('Issue Title', () => {
 
     // need setTimeout because actual setTimeout in code :P
     setTimeout(() => {
-      expect(issueShowComponent.$el.querySelector('.title').innerHTML)
+      expect(document.querySelector('title').innerText)
         .toContain('this is a title');
+      expect(issueShowComponent.$el.querySelector('.title').innerHTML)
+        .toContain('<p>this is a title</p>');
+      expect(issueShowComponent.$el.querySelector('.wiki').innerHTML)
+        .toContain('<p>this is a description!</p>');
+      expect(issueShowComponent.$el.querySelector('.js-task-list-field').innerText)
+        .toContain('this is a description');
       done();
     }, 10);
     // 10ms is just long enough for the update hook to fire
-- 
GitLab


From b5e58c3e082c0b611b1527e72f9ae7a64f7a865a Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 25 Apr 2017 11:51:27 -0600
Subject: [PATCH 072/363] remove fdescribe :(

---
 spec/javascripts/issue_show/issue_title_description_spec.js | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index b81e5d74f7ec3..9148f16714b64 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -13,7 +13,7 @@ const issueShowInterceptor = (request, next) => {
   }));
 };
 
-fdescribe('Issue Title', () => {
+describe('Issue Title', () => {
   const comps = {
     IssueTitleComponent: {},
   };
-- 
GitLab


From ab7e59122437cbf64d731b6a27304d0f96dd2b2c Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 25 Apr 2017 12:48:00 -0600
Subject: [PATCH 073/363] formatting in test

---
 .../issue_show/issue_title_description_spec.js           | 9 ++++++++-
 1 file changed, 8 insertions(+), 1 deletion(-)

diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index 9148f16714b64..756eda54f798c 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -41,12 +41,19 @@ describe('Issue Title', () => {
     setTimeout(() => {
       expect(document.querySelector('title').innerText)
         .toContain('this is a title');
+
       expect(issueShowComponent.$el.querySelector('.title').innerHTML)
         .toContain('<p>this is a title</p>');
+
       expect(issueShowComponent.$el.querySelector('.wiki').innerHTML)
         .toContain('<p>this is a description!</p>');
-      expect(issueShowComponent.$el.querySelector('.js-task-list-field').innerText)
+
+      const hiddenText = issueShowComponent.$el
+        .querySelector('.js-task-list-field').innerText;
+
+      expect(hiddenText)
         .toContain('this is a description');
+
       done();
     }, 10);
     // 10ms is just long enough for the update hook to fire
-- 
GitLab


From d8f767445933f41a7629a7637ba962bdbc461c9a Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Tue, 25 Apr 2017 21:47:49 -0500
Subject: [PATCH 074/363] Add ability to parse *.vue files through the `ruby
 gettext:find` script

---
 .../initializers/gettext_rails_i18n_patch.rb  | 15 +++++++++
 lib/tasks/gettext.rake                        | 33 +++++++++++++++++++
 2 files changed, 48 insertions(+)
 create mode 100644 lib/tasks/gettext.rake

diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
index 5570b7e31e9e4..d74c1f597fefc 100644
--- a/config/initializers/gettext_rails_i18n_patch.rb
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -13,6 +13,7 @@ class HamlParser
     # If we found a content like "{{ 'Stage' | translate }}"
     # in a HAML file we convert it to "= _('Stage')", that way
     # it can be processed by the "rake gettext:find" script.
+    #
     # Overwrites: https://github.com/grosser/gettext_i18n_rails/blob/8396387a431e0f8ead72fc1cd425cad2fa4992f2/lib/gettext_i18n_rails/haml_parser.rb#L9
     def self.convert_to_code(text)
       # {{ 'Stage' | translate }} => = _('Stage')
@@ -29,6 +30,20 @@ def self.convert_to_code(text)
 module GettextI18nRailsJs
   module Parser
     module Javascript
+
+      # This is required to tell the `rake gettext:find` script to use the Javascript
+      # parser for *.vue files.
+      #
+      # Overwrites: https://github.com/webhippie/gettext_i18n_rails_js/blob/46c58db6d2053a4f5f36a0eb024ea706ff5707cb/lib/gettext_i18n_rails_js/parser/javascript.rb#L36
+      def target?(file)
+        [
+          ".js",
+          ".jsx",
+          ".coffee",
+          ".vue"
+        ].include? ::File.extname(file)
+      end
+
       protected
 
       # Overwrites: https://github.com/webhippie/gettext_i18n_rails_js/blob/46c58db6d2053a4f5f36a0eb024ea706ff5707cb/lib/gettext_i18n_rails_js/parser/javascript.rb#L46
diff --git a/lib/tasks/gettext.rake b/lib/tasks/gettext.rake
new file mode 100644
index 0000000000000..cc18616d2a785
--- /dev/null
+++ b/lib/tasks/gettext.rake
@@ -0,0 +1,33 @@
+require "gettext_i18n_rails/tasks"
+
+namespace :gettext do
+  # Customize list of translatable files
+  # See: https://github.com/grosser/gettext_i18n_rails#customizing-list-of-translatable-files
+  def files_to_translate
+    folders = [
+      "app",
+      "lib",
+      "config",
+      locale_path
+    ].join(",")
+
+    exts = [
+      "rb",
+      "erb",
+      "haml",
+      "slim",
+      "rhtml",
+      "js",
+      "jsx",
+      "vue",
+      "coffee",
+      "handlebars",
+      "hbs",
+      "mustache"
+    ].join(",")
+
+    Dir.glob(
+      "{#{folders}}/**/*.{#{exts}}"
+    )
+  end
+end
-- 
GitLab


From 597c5c762ff5c86f8fc8c2c3c3aed2af270c2a71 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 26 Apr 2017 11:17:53 +0100
Subject: [PATCH 075/363] Fixed being behind

---
 README.md                                     |  2 +-
 app/assets/stylesheets/framework/blocks.scss  |  1 -
 app/assets/stylesheets/framework/nav.scss     |  2 +-
 .../users/migrate_to_ghost_user_service.rb    | 34 +++++---
 ...01-add-slash-slack-commands-to-api-doc.yml |  5 ++
 ...tion-while-moving-issues-to-ghost-user.yml |  4 +
 changelogs/unreleased/fix_link_in_readme.yml  |  4 +
 doc/administration/integration/plantuml.md    |  2 +-
 doc/api/services.md                           | 79 ++++++++++++++++---
 doc/development/fe_guide/testing.md           |  6 +-
 .../migrate_to_ghost_user_service_spec.rb     | 18 +++++
 ...e_to_ghost_user_service_shared_examples.rb | 52 ++++++++++++
 12 files changed, 181 insertions(+), 28 deletions(-)
 create mode 100644 changelogs/unreleased/29801-add-slash-slack-commands-to-api-doc.yml
 create mode 100644 changelogs/unreleased/30306-transaction-while-moving-issues-to-ghost-user.yml
 create mode 100644 changelogs/unreleased/fix_link_in_readme.yml

diff --git a/README.md b/README.md
index f0e3b52ef6f06..10d69efdc6bf3 100644
--- a/README.md
+++ b/README.md
@@ -73,7 +73,7 @@ One small thing you also have to do when installing it yourself is to copy the e
 
     cp config/unicorn.rb.example.development config/unicorn.rb
 
-Instructions on how to start GitLab and how to run the tests can be found in the [development section of the GitLab Development Kit](https://gitlab.com/gitlab-org/gitlab-development-kit#development).
+Instructions on how to start GitLab and how to run the tests can be found in the [getting started section of the GitLab Development Kit](https://gitlab.com/gitlab-org/gitlab-development-kit#getting-started).
 
 ## Software stack
 
diff --git a/app/assets/stylesheets/framework/blocks.scss b/app/assets/stylesheets/framework/blocks.scss
index 524252629254e..f3e2a5db0a64d 100644
--- a/app/assets/stylesheets/framework/blocks.scss
+++ b/app/assets/stylesheets/framework/blocks.scss
@@ -230,7 +230,6 @@
     float: right;
     margin-top: 8px;
     padding-bottom: 8px;
-    border-bottom: 1px solid $border-color;
   }
 }
 
diff --git a/app/assets/stylesheets/framework/nav.scss b/app/assets/stylesheets/framework/nav.scss
index e6d808717f37f..b6cf5101d6094 100644
--- a/app/assets/stylesheets/framework/nav.scss
+++ b/app/assets/stylesheets/framework/nav.scss
@@ -110,7 +110,7 @@
 
 .top-area {
   @include clearfix;
-  border-bottom: 1px solid $white-normal;
+  border-bottom: 1px solid $border-color;
 
   .nav-text {
     padding-top: 16px;
diff --git a/app/services/users/migrate_to_ghost_user_service.rb b/app/services/users/migrate_to_ghost_user_service.rb
index 1e1ed1791ec7e..4628c4c6f6e7d 100644
--- a/app/services/users/migrate_to_ghost_user_service.rb
+++ b/app/services/users/migrate_to_ghost_user_service.rb
@@ -15,27 +15,39 @@ def initialize(user)
     end
 
     def execute
-      # Block the user before moving records to prevent a data race.
-      # For example, if the user creates an issue after `migrate_issues`
-      # runs and before the user is destroyed, the destroy will fail with
-      # an exception.
-      user.block
+      transition = user.block_transition
 
       user.transaction do
+        # Block the user before moving records to prevent a data race.
+        # For example, if the user creates an issue after `migrate_issues`
+        # runs and before the user is destroyed, the destroy will fail with
+        # an exception.
+        user.block
+
+        # Reverse the user block if record migration fails
+        if !migrate_records && transition
+          transition.rollback
+          user.save!
+        end
+      end
+
+      user.reload
+    end
+
+    private
+
+    def migrate_records
+      user.transaction(requires_new: true) do
         @ghost_user = User.ghost
 
         migrate_issues
         migrate_merge_requests
         migrate_notes
         migrate_abuse_reports
-        migrate_award_emoji
+        migrate_award_emojis
       end
-
-      user.reload
     end
 
-    private
-
     def migrate_issues
       user.issues.update_all(author_id: ghost_user.id)
     end
@@ -52,7 +64,7 @@ def migrate_abuse_reports
       user.reported_abuse_reports.update_all(reporter_id: ghost_user.id)
     end
 
-    def migrate_award_emoji
+    def migrate_award_emojis
       user.award_emoji.update_all(user_id: ghost_user.id)
     end
   end
diff --git a/changelogs/unreleased/29801-add-slash-slack-commands-to-api-doc.yml b/changelogs/unreleased/29801-add-slash-slack-commands-to-api-doc.yml
new file mode 100644
index 0000000000000..9c5df690085c8
--- /dev/null
+++ b/changelogs/unreleased/29801-add-slash-slack-commands-to-api-doc.yml
@@ -0,0 +1,5 @@
+---
+title: Add Slack slash command api to services documentation and rearrange order and
+  cases
+merge_request: 10757
+author: TM Lee
diff --git a/changelogs/unreleased/30306-transaction-while-moving-issues-to-ghost-user.yml b/changelogs/unreleased/30306-transaction-while-moving-issues-to-ghost-user.yml
new file mode 100644
index 0000000000000..5fc57e44be617
--- /dev/null
+++ b/changelogs/unreleased/30306-transaction-while-moving-issues-to-ghost-user.yml
@@ -0,0 +1,4 @@
+---
+title: Add a transaction around move_issues_to_ghost_user
+merge_request: 10465
+author:
diff --git a/changelogs/unreleased/fix_link_in_readme.yml b/changelogs/unreleased/fix_link_in_readme.yml
new file mode 100644
index 0000000000000..be5ceac86563e
--- /dev/null
+++ b/changelogs/unreleased/fix_link_in_readme.yml
@@ -0,0 +1,4 @@
+---
+title: Fix dead link to GDK on the README page
+merge_request:
+author: Dino Maric
diff --git a/doc/administration/integration/plantuml.md b/doc/administration/integration/plantuml.md
index 5c85683503913..b21817c1fd31a 100644
--- a/doc/administration/integration/plantuml.md
+++ b/doc/administration/integration/plantuml.md
@@ -28,7 +28,7 @@ using Tomcat:
 sudo apt-get install tomcat7
 sudo cp target/plantuml.war /var/lib/tomcat7/webapps/plantuml.war
 sudo chown tomcat7:tomcat7 /var/lib/tomcat7/webapps/plantuml.war
-sudo service restart tomcat7
+sudo service tomcat7 restart
 ```
 
 Once the Tomcat service restarts the PlantUML service will be ready and
diff --git a/doc/api/services.md b/doc/api/services.md
index 7d4779f113763..0f42c2560992f 100644
--- a/doc/api/services.md
+++ b/doc/api/services.md
@@ -490,41 +490,98 @@ Remove all previously JIRA settings from a project.
 DELETE /projects/:id/services/jira
 ```
 
-## Mattermost Slash Commands
+## Slack slash commands
 
-Ability to receive slash commands from a Mattermost chat instance.
+Ability to receive slash commands from a Slack chat instance.
 
-### Create/Edit Mattermost Slash Command service
+### Get Slack slash command service settings
 
-Set Mattermost Slash Command for a project.
+Get Slack slash command service settings for a project.
 
 ```
-PUT /projects/:id/services/mattermost-slash-commands
+GET /projects/:id/services/slack-slash-commands
+```
+
+Example response:
+
+```json
+{
+  "id": 4,
+  "title": "Slack slash commands",
+  "created_at": "2017-06-27T05:51:39-07:00",
+  "updated_at": "2017-06-27T05:51:39-07:00",
+  "active": true,
+  "push_events": true,
+  "issues_events": true,
+  "merge_requests_events": true,
+  "tag_push_events": true,
+  "note_events": true,
+  "build_events": true,
+  "pipeline_events": true,
+  "properties": {
+    "token": "9koXpg98eAheJpvBs5tK"
+  }
+}
+```
+
+### Create/Edit Slack slash command service
+
+Set Slack slash command for a project.
+
+```
+PUT /projects/:id/services/slack-slash-commands
 ```
 
 Parameters:
 
 | Attribute | Type | Required | Description |
 | --------- | ---- | -------- | ----------- |
-| `token` | string | yes | The Mattermost token |
+| `token` | string | yes | The Slack token |
 
 
-### Delete Mattermost Slash Command service
+### Delete Slack slash command service
 
-Delete Mattermost Slash Command service for a project.
+Delete Slack slash command service for a project.
 
 ```
-DELETE /projects/:id/services/mattermost-slash-commands
+DELETE /projects/:id/services/slack-slash-commands
 ```
 
-### Get Mattermost Slash Command service settings
+## Mattermost slash commands
+
+Ability to receive slash commands from a Mattermost chat instance.
+
+### Get Mattermost slash command service settings
 
-Get Mattermost Slash Command service settings for a project.
+Get Mattermost slash command service settings for a project.
 
 ```
 GET /projects/:id/services/mattermost-slash-commands
 ```
 
+### Create/Edit Mattermost slash command service
+
+Set Mattermost slash command for a project.
+
+```
+PUT /projects/:id/services/mattermost-slash-commands
+```
+
+Parameters:
+
+| Attribute | Type | Required | Description |
+| --------- | ---- | -------- | ----------- |
+| `token` | string | yes | The Mattermost token |
+
+
+### Delete Mattermost slash command service
+
+Delete Mattermost slash command service for a project.
+
+```
+DELETE /projects/:id/services/mattermost-slash-commands
+```
+
 ## Pipeline-Emails
 
 Get emails for GitLab CI pipelines.
diff --git a/doc/development/fe_guide/testing.md b/doc/development/fe_guide/testing.md
index 66afbf4db4d64..157c13352ca07 100644
--- a/doc/development/fe_guide/testing.md
+++ b/doc/development/fe_guide/testing.md
@@ -14,8 +14,10 @@ for more information on general testing practices at GitLab.
 
 GitLab uses the [Karma][karma] test runner with [Jasmine][jasmine] as its test
 framework for our JavaScript unit tests. For tests that rely on DOM
-manipulation we use fixtures which are pre-compiled from HAML source files and
-served during testing by the [jasmine-jquery][jasmine-jquery] plugin.
+manipulation, we generate HTML files using RSpec suites (see `spec/javascripts/fixtures/*.rb` for examples).
+Some fixtures are still HAML templates that are translated to HTML files using the same mechanism (see `static_fixtures.rb`).
+Those will be migrated over time.
+Fixtures are served during testing by the [jasmine-jquery][jasmine-jquery] plugin.
 
 JavaScript tests live in `spec/javascripts/`, matching the folder structure
 of `app/assets/javascripts/`: `app/assets/javascripts/behaviors/autosize.js`
diff --git a/spec/services/users/migrate_to_ghost_user_service_spec.rb b/spec/services/users/migrate_to_ghost_user_service_spec.rb
index 8c5b7e41c1550..9e1edf1ac30d8 100644
--- a/spec/services/users/migrate_to_ghost_user_service_spec.rb
+++ b/spec/services/users/migrate_to_ghost_user_service_spec.rb
@@ -60,5 +60,23 @@
         end
       end
     end
+
+    context "when record migration fails with a rollback exception" do
+      before do
+        expect_any_instance_of(MergeRequest::ActiveRecord_Associations_CollectionProxy)
+          .to receive(:update_all).and_raise(ActiveRecord::Rollback)
+      end
+
+      context "for records that were already migrated" do
+        let!(:issue) { create(:issue, project: project, author: user) }
+        let!(:merge_request) { create(:merge_request, source_project: project, author: user, target_branch: "first") }
+
+        it "reverses the migration" do
+          service.execute
+
+          expect(issue.reload.author).to eq(user)
+        end
+      end
+    end
   end
 end
diff --git a/spec/support/services/migrate_to_ghost_user_service_shared_examples.rb b/spec/support/services/migrate_to_ghost_user_service_shared_examples.rb
index 0eac587e9730a..dcc562c684b0f 100644
--- a/spec/support/services/migrate_to_ghost_user_service_shared_examples.rb
+++ b/spec/support/services/migrate_to_ghost_user_service_shared_examples.rb
@@ -35,5 +35,57 @@
 
       expect(user).to be_blocked
     end
+
+    context "race conditions" do
+      context "when #{record_class_name} migration fails and is rolled back" do
+        before do
+          expect_any_instance_of(record_class::ActiveRecord_Associations_CollectionProxy)
+            .to receive(:update_all).and_raise(ActiveRecord::Rollback)
+        end
+
+        it 'rolls back the user block' do
+          service.execute
+
+          expect(user.reload).not_to be_blocked
+        end
+
+        it "doesn't unblock an previously-blocked user" do
+          user.block
+
+          service.execute
+
+          expect(user.reload).to be_blocked
+        end
+      end
+
+      context "when #{record_class_name} migration fails with a non-rollback exception" do
+        before do
+          expect_any_instance_of(record_class::ActiveRecord_Associations_CollectionProxy)
+            .to receive(:update_all).and_raise(ArgumentError)
+        end
+
+        it 'rolls back the user block' do
+          service.execute rescue nil
+
+          expect(user.reload).not_to be_blocked
+        end
+
+        it "doesn't unblock an previously-blocked user" do
+          user.block
+
+          service.execute rescue nil
+
+          expect(user.reload).to be_blocked
+        end
+      end
+
+      it "blocks the user before #{record_class_name} migration begins" do
+        expect(service).to receive("migrate_#{record_class_name.parameterize('_')}s".to_sym) do
+          expect(user.reload).to be_blocked
+        end
+
+        service.execute
+      end
+    end
   end
 end
-- 
GitLab


From cc248f3e7be6ae3cfd0d9f24eae92500e14b9e98 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 26 Apr 2017 14:05:13 +0100
Subject: [PATCH 076/363] Swapped flex out for grid system

---
 app/assets/stylesheets/framework/files.scss | 28 +++++++++++++++++----
 1 file changed, 23 insertions(+), 5 deletions(-)

diff --git a/app/assets/stylesheets/framework/files.scss b/app/assets/stylesheets/framework/files.scss
index 099187a519383..3737ae3e653f6 100644
--- a/app/assets/stylesheets/framework/files.scss
+++ b/app/assets/stylesheets/framework/files.scss
@@ -162,13 +162,31 @@
     }
 
     .list-inline.previews {
-      display: flex;
-      flex-wrap: wrap;
-      justify-content: space-between;
-      padding: $gl-padding;
+      display: inline-block;
 
       .preview {
-        flex-shrink: 0;
+        display: inline-block;
+        height: 280px;
+        min-width: 260px;
+        margin: 0;
+        padding: 16px;
+        @include make-xs-column(3);
+      }
+
+      .panel {
+        margin: 0;
+        height: 100%;
+
+        .panel-body {
+          padding: $gl-padding 0 0;
+          text-align: center;
+        }
+      }
+
+      .img-thumbnail {
+        max-height: 195px;
+        max-width: 195px;
+        padding: 0;
       }
     }
   }
-- 
GitLab


From 0242c4dfc92da0c59583b0ac9f1d3f2e6dcd11b8 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 26 Apr 2017 17:03:06 +0100
Subject: [PATCH 077/363] Changed JS translation functions Added context
 function

---
 .../components/limit_warning_component.js     |  2 +-
 .../components/stage_code_component.js        |  4 +-
 .../components/stage_issue_component.js       |  4 +-
 .../components/stage_plan_component.js        |  4 +-
 .../components/stage_production_component.js  |  4 +-
 .../components/stage_review_component.js      |  4 +-
 .../components/stage_staging_component.js     |  2 +-
 .../components/total_time_component.js        |  6 +--
 .../cycle_analytics/cycle_analytics_store.js  | 17 +++---
 app/assets/javascripts/locale/index.js        |  9 +++-
 .../javascripts/vue_shared/translate.js       | 19 +++++--
 .../cycle_analytics/_empty_stage.html.haml    |  2 +-
 .../cycle_analytics/_no_access.html.haml      |  4 +-
 .../projects/cycle_analytics/show.html.haml   | 29 ++++++-----
 locale/de/gitlab.po                           | 48 +++++++++++------
 locale/en/gitlab.po                           | 48 +++++++++++------
 locale/es/gitlab.po                           | 50 ++++++++++++------
 locale/gitlab.pot                             | 52 +++++++++++++------
 spec/javascripts/vue_shared/translate_spec.js |  8 +--
 19 files changed, 202 insertions(+), 114 deletions(-)

diff --git a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
index 63e20478e9494..72ada943fea0f 100644
--- a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
@@ -11,7 +11,7 @@ export default {
           aria-hidden="true"
           title="Limited to showing 50 events at most"
           data-placement="top"></i>
-      {{ 'Showing %d event' | translate-plural('Showing %d events', 50) }}
+      {{ n__('Showing %d event', 'Showing %d events', 50) }}
     </span>
   `,
 };
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
index f72882872cdf1..0bd15ee773d34 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageCodeComponent = Vue.extend({
             <a :href="mergeRequest.url" class="issue-link">!{{ mergeRequest.iid }}</a>
             &middot;
             <span>
-              {{ 'Opened' | translate }}
+              {{ __('Opened') }}
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              {{ 'by' | translate }}
+              {{ __('by') }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
           </div>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
index bb265c8316feb..b6bf9a2572e8f 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageIssueComponent = Vue.extend({
             <a :href="issue.url" class="issue-link">#{{ issue.iid }}</a>
             &middot;
             <span>
-              {{ 'Opened' | translate }}
+              {{ __('Opened') }}
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-              {{ 'by' | translate }}
+              {{ __('by') }}
               <a :href="issue.author.webUrl" class="issue-author-link">
                 {{ issue.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
index 32b685faece6a..116c118b82413 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
@@ -31,10 +31,10 @@ global.cycleAnalytics.StagePlanComponent = Vue.extend({
               </a>
             </h5>
             <span>
-              {{ 'First' | translate }}
+              {{ __('First') }}
               <span class="commit-icon">${iconCommit}</span>
               <a :href="commit.commitUrl" class="commit-hash-link monospace">{{ commit.shortSha }}</a>
-              {{ 'pushed by' | translate }}
+              {{ __('pushed by') }}
               <a :href="commit.author.webUrl" class="commit-author-link">
                 {{ commit.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
index 5c9186a2e497c..18fb09d6d2f95 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageProductionComponent = Vue.extend({
             <a :href="issue.url" class="issue-link">#{{ issue.iid }}</a>
             &middot;
             <span>
-              {{ 'Opened' | translate }}
+              {{ __('Opened') }}
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-            {{ 'by' | translate }}
+            {{ __('by') }}
             <a :href="issue.author.webUrl" class="issue-author-link">
               {{ issue.author.name }}
             </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
index a047573548df0..af5f1b42ea2b7 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageReviewComponent = Vue.extend({
             <a :href="mergeRequest.url" class="issue-link">!{{ mergeRequest.iid }}</a>
             &middot;
             <span>
-              {{ 'Opened' | translate }}
+              {{ __('Opened') }}
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              {{ 'by' | translate }}
+              {{ __('by') }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
             <template v-if="mergeRequest.state === 'closed'">
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
index e8dfaf4294e19..a0bd1bab96e55 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageStagingComponent = Vue.extend({
             </h5>
             <span>
               <a :href="build.url" class="build-date">{{ build.date }}</a>
-              {{ 'by' | translate }}
+              {{ __('by') }}
               <a :href="build.author.webUrl" class="issue-author-link">
                 {{ build.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/total_time_component.js b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
index 47d82bdda90d5..48cc0a5c58dbb 100644
--- a/app/assets/javascripts/cycle_analytics/components/total_time_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
@@ -12,9 +12,9 @@ global.cycleAnalytics.TotalTimeComponent = Vue.extend({
   template: `
     <span class="total-time">
       <template v-if="Object.keys(time).length">
-        <template v-if="time.days">{{ time.days }} <span>{{ 'day' | translate-plural('days', time.days) }}</span></template>
-        <template v-if="time.hours">{{ time.hours }} <span>{{ 'hr' | translate }}</span></template>
-        <template v-if="time.mins && !time.days">{{ time.mins }} <span>{{ 'min' | translate-plural('mins', time.mins) }}</span></template>
+        <template v-if="time.days">{{ time.days }} <span>{{ n__('day', 'days', time.days) }}</span></template>
+        <template v-if="time.hours">{{ time.hours }} <span>{{ __('hr') }}</span></template>
+        <template v-if="time.mins && !time.days">{{ time.mins }} <span>{{ n__('min', 'mins', time.mins) }}</span></template>
         <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>s</span></template>
       </template>
       <template v-else>
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
index 25d5092a1fd8e..1c57495121141 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
@@ -1,6 +1,5 @@
 /* eslint-disable no-param-reassign */
-
-import locale from '~/locale';
+import { __ } from '../locale';
 
 require('../lib/utils/text_utility');
 const DEFAULT_EVENT_OBJECTS = require('./default_event_objects');
@@ -9,13 +8,13 @@ const global = window.gl || (window.gl = {});
 global.cycleAnalytics = global.cycleAnalytics || {};
 
 const EMPTY_STAGE_TEXTS = {
-  issue: locale.gettext('The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.'),
-  plan: locale.gettext('The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.'),
-  code: locale.gettext('The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.'),
-  test: locale.gettext('The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.'),
-  review: locale.gettext('The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.'),
-  staging: locale.gettext('The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.'),
-  production: locale.gettext('The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.'),
+  issue: __('The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.'),
+  plan: __('The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.'),
+  code: __('The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.'),
+  test: __('The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.'),
+  review: __('The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.'),
+  staging: __('The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.'),
+  production: __('The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.'),
 };
 
 global.cycleAnalytics.CycleAnalyticsStore = {
diff --git a/app/assets/javascripts/locale/index.js b/app/assets/javascripts/locale/index.js
index 9f81024ff68c2..3907b0e2aba34 100644
--- a/app/assets/javascripts/locale/index.js
+++ b/app/assets/javascripts/locale/index.js
@@ -10,6 +10,13 @@ const locales = {
 };
 
 const lang = document.querySelector('html').getAttribute('lang') || 'en';
+const locale = new Jed(locales[lang]);
+const gettext = locale.gettext.bind(locale);
+const ngettext = locale.ngettext.bind(locale);
+const pgettext = locale.pgettext.bind(locale);
 
 export { lang };
-export default new Jed(locales[lang]);
+export { gettext as __ };
+export { ngettext as n__ };
+export { pgettext as s__ };
+export default locale;
diff --git a/app/assets/javascripts/vue_shared/translate.js b/app/assets/javascripts/vue_shared/translate.js
index 7fc6f1ce38af5..c2c20ea0853d3 100644
--- a/app/assets/javascripts/vue_shared/translate.js
+++ b/app/assets/javascripts/vue_shared/translate.js
@@ -1,8 +1,17 @@
-import locale from '../locale';
+import {
+  __,
+  n__,
+  s__,
+} from '../locale';
 
 export default (Vue) => {
-  Vue.filter('translate', text => locale.gettext(text));
-
-  Vue.filter('translate-plural', (text, pluralText, count) =>
-    locale.ngettext(text, pluralText, count).replace(/%d/g, count));
+  Vue.mixin({
+    methods: {
+      __(text) { return __(text); },
+      n__(text, pluralText, count) {
+        return n__(text, pluralText, count).replace(/%d/g, count);
+      },
+      s__(context, key) { return s__(context, key); },
+    },
+  });
 };
diff --git a/app/views/projects/cycle_analytics/_empty_stage.html.haml b/app/views/projects/cycle_analytics/_empty_stage.html.haml
index 27190785fff2d..57e95c4c14f13 100644
--- a/app/views/projects/cycle_analytics/_empty_stage.html.haml
+++ b/app/views/projects/cycle_analytics/_empty_stage.html.haml
@@ -2,6 +2,6 @@
   .empty-stage
     .icon-no-data
       = custom_icon ('icon_no_data')
-    %h4 {{ 'We don\'t have enough data to show this stage.' | translate }}
+    %h4 {{ 'We don\'t have enough data to show this stage.' | t }}
     %p
       {{currentStage.emptyStageText}}
diff --git a/app/views/projects/cycle_analytics/_no_access.html.haml b/app/views/projects/cycle_analytics/_no_access.html.haml
index 474d0f410a704..25109bd61cd34 100644
--- a/app/views/projects/cycle_analytics/_no_access.html.haml
+++ b/app/views/projects/cycle_analytics/_no_access.html.haml
@@ -2,6 +2,6 @@
   .no-access-stage
     .icon-lock
       = custom_icon ('icon_lock')
-    %h4 {{ 'You need permission.' | translate }}
+    %h4 {{ 'You need permission.' | t }}
     %p
-      {{ 'Want to see the data? Please ask administrator for access.' | translate }}
+      {{ 'Want to see the data? Please ask administrator for access.' | t }}
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index a7783b6958803..2e0e4753e6888 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -8,7 +8,8 @@
 = render "projects/head"
 
 #cycle-analytics{ class: container_class, "v-cloak" => "true", data: { request_path: project_cycle_analytics_path(@project) } }
-  - if @cycle_analytics_no_data
+  - if @cycle_analytics_no_data || true
+    {{ s__('context', 'key') }}
     .bordered-box.landing.content-block{ "v-if" => "!isOverviewDialogDismissed" }
       = icon("times", class: "dismiss-icon", "@click" => "dismissOverviewDialog()")
       .row
@@ -16,17 +17,17 @@
           = custom_icon('icon_cycle_analytics_splash')
         .col-sm-8.col-xs-12.inner-content
           %h4
-            {{ 'Introducing Cycle Analytics' | translate }}
+            {{ __('Introducing Cycle Analytics') }}
           %p
-            {{ 'Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.' | translate }}
+            {{ __('Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.') }}
 
           = link_to help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn' do
-            {{ 'Read more' | translate }}
+            {{ __('Read more') }}
   = icon("spinner spin", "v-show" => "isLoading")
   .wrapper{ "v-show" => "!isLoading && !hasError" }
     .panel.panel-default
       .panel-heading
-        {{ 'Pipeline Health' | translate }}
+        {{ __('Pipeline Health') }}
       .content-block
         .container-fluid
           .row
@@ -36,15 +37,15 @@
             .col-sm-3.col-xs-12.column
               .dropdown.inline.js-ca-dropdown
                 %button.dropdown-menu-toggle{ "data-toggle" => "dropdown", :type => "button" }
-                  %span.dropdown-label {{ 'Last 30 days' | translate }}
+                  %span.dropdown-label {{ __('Last 30 days') }}
                   %i.fa.fa-chevron-down
                 %ul.dropdown-menu.dropdown-menu-align-right
                   %li
                     %a{ "href" => "#", "data-value" => "30" }
-                      {{ 'Last 30 days' | translate }}
+                      {{ __('Last 30 days') }}
                   %li
                     %a{ "href" => "#", "data-value" => "90" }
-                      {{ 'Last 90 days' | translate }}
+                      {{ __('Last 90 days') }}
     .stage-panel-container
       .panel.panel-default.stage-panel
         .panel-heading
@@ -52,19 +53,19 @@
             %ul
               %li.stage-header
                 %span.stage-name
-                  {{ 'Stage' | translate }}
+                  {{ __('Stage') }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The phase of the development lifecycle.", "aria-hidden" => "true" }
               %li.median-header
                 %span.stage-name
-                  {{ 'Median' | translate }}
+                  {{ __('Median') }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.", "aria-hidden" => "true" }
               %li.event-header
                 %span.stage-name
-                  {{ currentStage ? currentStage.legend : 'Related Issues' | translate }}
+                  {{ __(currentStage ? currentStage.legend : 'Related Issues') }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The collection of events added to the data gathered for that stage.", "aria-hidden" => "true" }
               %li.total-time-header
                 %span.stage-name
-                  {{ 'Total Time' | translate }}
+                  {{ __('Total Time') }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The time taken by each data entry gathered by that stage.", "aria-hidden" => "true" }
         .stage-panel-body
           %nav.stage-nav
@@ -77,10 +78,10 @@
                     %span{ "v-if" => "stage.value" }
                       {{ stage.value }}
                     %span.stage-empty{ "v-else" => true }
-                      {{ 'Not enough data' | translate }}
+                      {{ __('Not enough data') }}
                   %template{ "v-else" => true }
                     %span.not-available
-                      {{ 'Not available' | translate }}
+                      {{ __('Not available') }}
           .section.stage-events
             %template{ "v-if" => "isLoadingStage" }
               = icon("spinner spin")
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index f3b6cb6e90590..c291b6adab698 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -17,47 +17,65 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
-msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
-msgstr ""
-
 msgid "Deutsch"
 msgstr ""
 
 msgid "English"
 msgstr ""
 
-msgid "Introducing Cycle Analytics"
+msgid "First"
 msgstr ""
 
-msgid "Last 30 days"
+msgid "Opened"
 msgstr ""
 
-msgid "Last 90 days"
-msgstr ""
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
 
-msgid "Median"
+msgid "Spanish"
 msgstr ""
 
-msgid "Not available"
+msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
-msgid "Not enough data"
+msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr ""
 
-msgid "Pipeline Health"
+msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr ""
 
-msgid "Read more"
+msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
 msgstr ""
 
-msgid "Spanish"
+msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
 msgstr ""
 
-msgid "Stage"
+msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
 msgstr ""
 
-msgid "Total Time"
+msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
+
+msgid "by"
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "hr"
+msgstr ""
+
+msgid "min"
+msgid_plural "mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "pushed by"
+msgstr ""
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 9302986848695..7e13ddb1c1878 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -17,47 +17,65 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
-msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
-msgstr ""
-
 msgid "Deutsch"
 msgstr ""
 
 msgid "English"
 msgstr ""
 
-msgid "Introducing Cycle Analytics"
+msgid "First"
 msgstr ""
 
-msgid "Last 30 days"
+msgid "Opened"
 msgstr ""
 
-msgid "Last 90 days"
-msgstr ""
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
 
-msgid "Median"
+msgid "Spanish"
 msgstr ""
 
-msgid "Not available"
+msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
-msgid "Not enough data"
+msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr ""
 
-msgid "Pipeline Health"
+msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr ""
 
-msgid "Read more"
+msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
 msgstr ""
 
-msgid "Spanish"
+msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
 msgstr ""
 
-msgid "Stage"
+msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
 msgstr ""
 
-msgid "Total Time"
+msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
+
+msgid "by"
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "hr"
+msgstr ""
+
+msgid "min"
+msgid_plural "mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "pushed by"
+msgstr ""
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 23863cbdb92e7..f5bf45e2cd31f 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -17,47 +17,65 @@ msgstr ""
 "Last-Translator: \n"
 "X-Generator: Poedit 2.0.1\n"
 
-msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
-msgstr ""
-
 msgid "Deutsch"
 msgstr "Alemán"
 
 msgid "English"
 msgstr "Inglés"
 
-msgid "Introducing Cycle Analytics"
+msgid "First"
 msgstr ""
 
-msgid "Last 30 days"
+msgid "Opened"
 msgstr ""
 
-msgid "Last 90 days"
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Spanish"
+msgstr "Español"
+
+msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
-msgid "Median"
+msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr ""
 
-msgid "Not available"
+msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr ""
 
-msgid "Not enough data"
+msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
 msgstr ""
 
-msgid "Pipeline Health"
+msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
 msgstr ""
 
-msgid "Read more"
+msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
 msgstr ""
 
-msgid "Spanish"
-msgstr "Español"
+msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
+msgstr ""
 
-msgid "Stage"
+msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
 
-msgid "Total Time"
+msgid "by"
 msgstr ""
 
-msgid "Want to see the data? Please ask administrator for access."
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "hr"
+msgstr ""
+
+msgid "min"
+msgid_plural "mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "pushed by"
 msgstr ""
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index ada103157d987..372ab3cd3ccda 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-21 09:01+0100\n"
-"PO-Revision-Date: 2017-04-21 09:01+0100\n"
+"POT-Creation-Date: 2017-04-26 16:26+0100\n"
+"PO-Revision-Date: 2017-04-26 16:26+0100\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -18,47 +18,65 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
 
-msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
-msgstr ""
-
 msgid "Deutsch"
 msgstr ""
 
 msgid "English"
 msgstr ""
 
-msgid "Introducing Cycle Analytics"
+msgid "First"
 msgstr ""
 
-msgid "Last 30 days"
+msgid "Opened"
 msgstr ""
 
-msgid "Last 90 days"
-msgstr ""
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
 
-msgid "Median"
+msgid "Spanish"
 msgstr ""
 
-msgid "Not available"
+msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
-msgid "Not enough data"
+msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr ""
 
-msgid "Pipeline Health"
+msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr ""
 
-msgid "Read more"
+msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
 msgstr ""
 
-msgid "Spanish"
+msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
 msgstr ""
 
-msgid "Stage"
+msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
 msgstr ""
 
-msgid "Total Time"
+msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
+
+msgid "by"
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "hr"
+msgstr ""
+
+msgid "min"
+msgid_plural "mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "pushed by"
+msgstr ""
diff --git a/spec/javascripts/vue_shared/translate_spec.js b/spec/javascripts/vue_shared/translate_spec.js
index 74bd4ff86b1bb..5dc0be5011647 100644
--- a/spec/javascripts/vue_shared/translate_spec.js
+++ b/spec/javascripts/vue_shared/translate_spec.js
@@ -17,7 +17,7 @@ describe('Vue translate filter', () => {
       el,
       template: `
         <span>
-          {{ 'testing' | translate }}
+          {{ 'testing' | t }}
         </span>
       `,
     }).$mount();
@@ -36,7 +36,7 @@ describe('Vue translate filter', () => {
       el,
       template: `
         <span>
-          {{ '%d day' | translate-plural('%d days', 1) }}
+          {{ '%d day' | nt('%d days', 1) }}
         </span>
       `,
     }).$mount();
@@ -55,7 +55,7 @@ describe('Vue translate filter', () => {
       el,
       template: `
         <span>
-          {{ '%d day' | translate-plural('%d days', 2) }}
+          {{ '%d day' | nt('%d days', 2) }}
         </span>
       `,
     }).$mount();
@@ -74,7 +74,7 @@ describe('Vue translate filter', () => {
       el,
       template: `
         <span>
-          {{ 'day' | translate-plural('days', 2) }}
+          {{ 'day' | nt('days', 2) }}
         </span>
       `,
     }).$mount();
-- 
GitLab


From cb06f09e982248ff6dc8218ffa1f431141fd1ad8 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 26 Apr 2017 17:20:13 +0100
Subject: [PATCH 078/363] Removed t filter from other places

---
 app/assets/javascripts/locale/de/app.js       |  2 +-
 app/assets/javascripts/locale/en/app.js       |  2 +-
 app/assets/javascripts/locale/es/app.js       |  2 +-
 .../cycle_analytics/_empty_stage.html.haml    |  2 +-
 .../cycle_analytics/_no_access.html.haml      |  4 +-
 locale/de/gitlab.po                           | 51 +++++++++--------
 locale/en/gitlab.po                           | 51 +++++++++--------
 locale/es/gitlab.po                           | 51 +++++++++--------
 locale/gitlab.pot                             | 55 ++++++++++---------
 spec/javascripts/vue_shared/translate_spec.js |  8 +--
 10 files changed, 120 insertions(+), 108 deletions(-)

diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index 036572c5568e1..76f01f1affec2 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":[""],"Stage":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index dfab3c8f0ded3..9a2d27c9f24b2 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":[""],"Stage":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index a12129bef1356..1518d8852334a 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-13 00:07-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":["Alemán"],"English":["Inglés"],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":["Español"],"Stage":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-13 00:07-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":["Alemán"],"English":["Inglés"],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":["Español"],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""]}}};
\ No newline at end of file
diff --git a/app/views/projects/cycle_analytics/_empty_stage.html.haml b/app/views/projects/cycle_analytics/_empty_stage.html.haml
index 57e95c4c14f13..cdad0bc723188 100644
--- a/app/views/projects/cycle_analytics/_empty_stage.html.haml
+++ b/app/views/projects/cycle_analytics/_empty_stage.html.haml
@@ -2,6 +2,6 @@
   .empty-stage
     .icon-no-data
       = custom_icon ('icon_no_data')
-    %h4 {{ 'We don\'t have enough data to show this stage.' | t }}
+    %h4 {{ __('We don\'t have enough data to show this stage.') }}
     %p
       {{currentStage.emptyStageText}}
diff --git a/app/views/projects/cycle_analytics/_no_access.html.haml b/app/views/projects/cycle_analytics/_no_access.html.haml
index 25109bd61cd34..dcfd1b7afc09d 100644
--- a/app/views/projects/cycle_analytics/_no_access.html.haml
+++ b/app/views/projects/cycle_analytics/_no_access.html.haml
@@ -2,6 +2,6 @@
   .no-access-stage
     .icon-lock
       = custom_icon ('icon_lock')
-    %h4 {{ 'You need permission.' | t }}
+    %h4 {{ __('You need permission.') }}
     %p
-      {{ 'Want to see the data? Please ask administrator for access.' | t }}
+      {{ __('Want to see the data? Please ask administrator for access.') }}
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index c291b6adab698..aeee577c1444e 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -17,26 +17,45 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgstr ""
+
 msgid "Deutsch"
 msgstr ""
 
 msgid "English"
 msgstr ""
 
-msgid "First"
+msgid "Introducing Cycle Analytics"
+msgstr ""
+
+msgid "Last 30 days"
+msgstr ""
+
+msgid "Last 90 days"
 msgstr ""
 
-msgid "Opened"
+msgid "Median"
 msgstr ""
 
-msgid "Showing %d event"
-msgid_plural "Showing %d events"
-msgstr[0] ""
-msgstr[1] ""
+msgid "Not available"
+msgstr ""
+
+msgid "Not enough data"
+msgstr ""
+
+msgid "Pipeline Health"
+msgstr ""
+
+msgid "Read more"
+msgstr ""
 
 msgid "Spanish"
 msgstr ""
 
+msgid "Stage"
+msgstr ""
+
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
@@ -58,24 +77,8 @@ msgstr ""
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
-msgid "Want to see the data? Please ask administrator for access."
-msgstr ""
-
-msgid "by"
+msgid "Total Time"
 msgstr ""
 
-msgid "day"
-msgid_plural "days"
-msgstr[0] ""
-msgstr[1] ""
-
-msgid "hr"
-msgstr ""
-
-msgid "min"
-msgid_plural "mins"
-msgstr[0] ""
-msgstr[1] ""
-
-msgid "pushed by"
+msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 7e13ddb1c1878..9d32f6c9f8f25 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -17,26 +17,45 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgstr ""
+
 msgid "Deutsch"
 msgstr ""
 
 msgid "English"
 msgstr ""
 
-msgid "First"
+msgid "Introducing Cycle Analytics"
+msgstr ""
+
+msgid "Last 30 days"
+msgstr ""
+
+msgid "Last 90 days"
 msgstr ""
 
-msgid "Opened"
+msgid "Median"
 msgstr ""
 
-msgid "Showing %d event"
-msgid_plural "Showing %d events"
-msgstr[0] ""
-msgstr[1] ""
+msgid "Not available"
+msgstr ""
+
+msgid "Not enough data"
+msgstr ""
+
+msgid "Pipeline Health"
+msgstr ""
+
+msgid "Read more"
+msgstr ""
 
 msgid "Spanish"
 msgstr ""
 
+msgid "Stage"
+msgstr ""
+
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
@@ -58,24 +77,8 @@ msgstr ""
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
-msgid "Want to see the data? Please ask administrator for access."
-msgstr ""
-
-msgid "by"
+msgid "Total Time"
 msgstr ""
 
-msgid "day"
-msgid_plural "days"
-msgstr[0] ""
-msgstr[1] ""
-
-msgid "hr"
-msgstr ""
-
-msgid "min"
-msgid_plural "mins"
-msgstr[0] ""
-msgstr[1] ""
-
-msgid "pushed by"
+msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index f5bf45e2cd31f..af7e127bc3a7d 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -17,26 +17,45 @@ msgstr ""
 "Last-Translator: \n"
 "X-Generator: Poedit 2.0.1\n"
 
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgstr ""
+
 msgid "Deutsch"
 msgstr "Alemán"
 
 msgid "English"
 msgstr "Inglés"
 
-msgid "First"
+msgid "Introducing Cycle Analytics"
+msgstr ""
+
+msgid "Last 30 days"
+msgstr ""
+
+msgid "Last 90 days"
 msgstr ""
 
-msgid "Opened"
+msgid "Median"
 msgstr ""
 
-msgid "Showing %d event"
-msgid_plural "Showing %d events"
-msgstr[0] ""
-msgstr[1] ""
+msgid "Not available"
+msgstr ""
+
+msgid "Not enough data"
+msgstr ""
+
+msgid "Pipeline Health"
+msgstr ""
+
+msgid "Read more"
+msgstr ""
 
 msgid "Spanish"
 msgstr "Español"
 
+msgid "Stage"
+msgstr ""
+
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
@@ -58,24 +77,8 @@ msgstr ""
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
-msgid "Want to see the data? Please ask administrator for access."
-msgstr ""
-
-msgid "by"
+msgid "Total Time"
 msgstr ""
 
-msgid "day"
-msgid_plural "days"
-msgstr[0] ""
-msgstr[1] ""
-
-msgid "hr"
-msgstr ""
-
-msgid "min"
-msgid_plural "mins"
-msgstr[0] ""
-msgstr[1] ""
-
-msgid "pushed by"
+msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index 372ab3cd3ccda..b99506788c72f 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-26 16:26+0100\n"
-"PO-Revision-Date: 2017-04-26 16:26+0100\n"
+"POT-Creation-Date: 2017-04-26 17:18+0100\n"
+"PO-Revision-Date: 2017-04-26 17:18+0100\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -18,26 +18,45 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
 
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgstr ""
+
 msgid "Deutsch"
 msgstr ""
 
 msgid "English"
 msgstr ""
 
-msgid "First"
+msgid "Introducing Cycle Analytics"
+msgstr ""
+
+msgid "Last 30 days"
+msgstr ""
+
+msgid "Last 90 days"
 msgstr ""
 
-msgid "Opened"
+msgid "Median"
 msgstr ""
 
-msgid "Showing %d event"
-msgid_plural "Showing %d events"
-msgstr[0] ""
-msgstr[1] ""
+msgid "Not available"
+msgstr ""
+
+msgid "Not enough data"
+msgstr ""
+
+msgid "Pipeline Health"
+msgstr ""
+
+msgid "Read more"
+msgstr ""
 
 msgid "Spanish"
 msgstr ""
 
+msgid "Stage"
+msgstr ""
+
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
@@ -59,24 +78,8 @@ msgstr ""
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
-msgid "Want to see the data? Please ask administrator for access."
-msgstr ""
-
-msgid "by"
+msgid "Total Time"
 msgstr ""
 
-msgid "day"
-msgid_plural "days"
-msgstr[0] ""
-msgstr[1] ""
-
-msgid "hr"
-msgstr ""
-
-msgid "min"
-msgid_plural "mins"
-msgstr[0] ""
-msgstr[1] ""
-
-msgid "pushed by"
+msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
diff --git a/spec/javascripts/vue_shared/translate_spec.js b/spec/javascripts/vue_shared/translate_spec.js
index 5dc0be5011647..cbb3cbdff4675 100644
--- a/spec/javascripts/vue_shared/translate_spec.js
+++ b/spec/javascripts/vue_shared/translate_spec.js
@@ -17,7 +17,7 @@ describe('Vue translate filter', () => {
       el,
       template: `
         <span>
-          {{ 'testing' | t }}
+          {{ __('testing') }}
         </span>
       `,
     }).$mount();
@@ -36,7 +36,7 @@ describe('Vue translate filter', () => {
       el,
       template: `
         <span>
-          {{ '%d day' | nt('%d days', 1) }}
+          {{ n__('%d day', '%d days', 1) }}
         </span>
       `,
     }).$mount();
@@ -55,7 +55,7 @@ describe('Vue translate filter', () => {
       el,
       template: `
         <span>
-          {{ '%d day' | nt('%d days', 2) }}
+          {{ n__('%d day', '%d days', 2) }}
         </span>
       `,
     }).$mount();
@@ -74,7 +74,7 @@ describe('Vue translate filter', () => {
       el,
       template: `
         <span>
-          {{ 'day' | nt('days', 2) }}
+          {{ n__('day', 'days', 2) }}
         </span>
       `,
     }).$mount();
-- 
GitLab


From c159727894d399152bc04914ef7b936417d6e1bb Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 26 Apr 2017 17:26:07 +0100
Subject: [PATCH 079/363] Removed random test code

---
 app/views/projects/cycle_analytics/show.html.haml | 3 +--
 1 file changed, 1 insertion(+), 2 deletions(-)

diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index 2e0e4753e6888..65afe41fe9e53 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -8,8 +8,7 @@
 = render "projects/head"
 
 #cycle-analytics{ class: container_class, "v-cloak" => "true", data: { request_path: project_cycle_analytics_path(@project) } }
-  - if @cycle_analytics_no_data || true
-    {{ s__('context', 'key') }}
+  - if @cycle_analytics_no_data
     .bordered-box.landing.content-block{ "v-if" => "!isOverviewDialogDismissed" }
       = icon("times", class: "dismiss-icon", "@click" => "dismissOverviewDialog()")
       .row
-- 
GitLab


From 82e967bc3ebc6b884651ddcadff2142332c365a5 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 26 Apr 2017 13:02:07 -0500
Subject: [PATCH 080/363] Adjust the patch for the gettext_i18n_rails and
 gettext_i18n_rails_js gems.

This is more simple now given we're using the recommended code to
annotate content to be translated.
---
 .../initializers/gettext_rails_i18n_patch.rb  | 27 +++----------------
 1 file changed, 3 insertions(+), 24 deletions(-)

diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
index d74c1f597fefc..6785f361eda77 100644
--- a/config/initializers/gettext_rails_i18n_patch.rb
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -1,8 +1,7 @@
 require 'gettext_i18n_rails/haml_parser'
 require 'gettext_i18n_rails_js/parser/javascript'
 
-VUE_TRANSLATE_REGEX = /{{ ([^{]*)( \| translate) }}/
-VUE_TRANSLATE_PLURAL_REGEX = /{{ ([^{]*)( \| translate-plural\((.*), (.*)\)) }}/
+VUE_TRANSLATE_REGEX = /{{ (N|n|s)?__\((.*)\) }}/
 
 module GettextI18nRails
   class HamlParser
@@ -10,17 +9,13 @@ class HamlParser
 
     # We need to convert text in Mustache format
     # to a format that can be parsed by Gettext scripts.
-    # If we found a content like "{{ 'Stage' | translate }}"
+    # If we found a content like "{{ __('Stage') }}"
     # in a HAML file we convert it to "= _('Stage')", that way
     # it can be processed by the "rake gettext:find" script.
     #
     # Overwrites: https://github.com/grosser/gettext_i18n_rails/blob/8396387a431e0f8ead72fc1cd425cad2fa4992f2/lib/gettext_i18n_rails/haml_parser.rb#L9
     def self.convert_to_code(text)
-      # {{ 'Stage' | translate }} => = _('Stage')
-      text.gsub!(VUE_TRANSLATE_REGEX, "= _(\\1)")
-
-      # {{ 'user' | translate-plural('users', users.size) }} => = n_('user', 'users', users.size)
-      text.gsub!(VUE_TRANSLATE_PLURAL_REGEX, "= n_(\\1, \\3, \\4)")
+      text.gsub!(VUE_TRANSLATE_REGEX, "= \\1_(\\2)")
 
       old_convert_to_code(text)
     end
@@ -43,22 +38,6 @@ def target?(file)
           ".vue"
         ].include? ::File.extname(file)
       end
-
-      protected
-
-      # Overwrites: https://github.com/webhippie/gettext_i18n_rails_js/blob/46c58db6d2053a4f5f36a0eb024ea706ff5707cb/lib/gettext_i18n_rails_js/parser/javascript.rb#L46
-      def collect_for(value)
-        ::File.open(value) do |f|
-          f.each_line.each_with_index.collect do |line, idx|
-            line.gsub!(VUE_TRANSLATE_REGEX, "__(\\1)")
-            line.gsub!(VUE_TRANSLATE_PLURAL_REGEX, "n__(\\1, \\3, \\4)")
-
-            line.scan(invoke_regex).collect do |function, arguments|
-              yield(function, arguments, idx + 1)
-            end
-          end.inject([], :+).compact
-        end
-      end
     end
   end
 end
-- 
GitLab


From 1f8226e1837eaaaffa8bb2efc498d3058049fa9c Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Wed, 26 Apr 2017 15:17:07 -0600
Subject: [PATCH 081/363] check if description changed prior to binding
 TaskLists and calling renderGFM

---
 .../issue_show/issue_title_description.vue    | 33 +++++++++++++------
 1 file changed, 23 insertions(+), 10 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index af2fe61af2bb1..ef7ac83f336a0 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -30,8 +30,9 @@ export default {
     return {
       poll,
       timeoutId: null,
-      title: null,
-      description: null,
+      title: '<span></span>',
+      description: '<span></span>',
+      descriptionChange: false,
     };
   },
   methods: {
@@ -40,6 +41,9 @@ export default {
       this.triggerAnimation(body);
     },
     triggerAnimation(body) {
+      // always reset to false before checking the change
+      this.descriptionChange = false;
+
       const { title, description } = body;
 
       this.descriptionText = body.description_text;
@@ -57,7 +61,12 @@ export default {
 
       if (!noTitleChange) {
         elementsToVisualize.push(this.$el.querySelector('.title'));
-      } else if (!noDescriptionChange) {
+      }
+
+      if (!noDescriptionChange) {
+        // only change to true when we need to bind TaskLists the html of description
+        this.descriptionChange = true;
+
         elementsToVisualize.push(this.$el.querySelector('.wiki'));
       }
 
@@ -100,15 +109,19 @@ export default {
     });
   },
   updated() {
-    const tl = new gl.TaskList({
-      dataType: 'issue',
-      fieldName: 'description',
-      selector: '.detail-page-description',
-    });
+    // if new html is injected (description changed) - bind TaskList and call renderGFM
+    if (this.descriptionChange) {
+      const tl = new gl.TaskList({
+        dataType: 'issue',
+        fieldName: 'description',
+        selector: '.detail-page-description',
+      });
 
-    $(this.$refs['issue-content-container-gfm-entry']).renderGFM();
+      $(this.$refs['issue-content-container-gfm-entry']).renderGFM();
 
-    return tl;
+      return tl;
+    }
+    return null;
   },
 };
 </script>
-- 
GitLab


From a4267d2d0f9b414aeeb1e42c6e0eca8a53af72e8 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 26 Apr 2017 16:19:06 -0500
Subject: [PATCH 082/363] Add some translations for Spanish.

---
 app/assets/javascripts/locale/de/app.js |  2 +-
 app/assets/javascripts/locale/en/app.js |  2 +-
 app/assets/javascripts/locale/es/app.js |  2 +-
 locale/de/gitlab.po                     | 30 +++++++++++
 locale/en/gitlab.po                     | 30 +++++++++++
 locale/es/gitlab.po                     | 69 ++++++++++++++++++-------
 locale/gitlab.pot                       | 34 +++++++++++-
 7 files changed, 145 insertions(+), 24 deletions(-)

diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index 76f01f1affec2..d74d36711d360 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index 9a2d27c9f24b2..05ddc42e33386 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index 1518d8852334a..bb5976c572a6c 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-13 00:07-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":["Alemán"],"English":["Inglés"],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Pipeline Health":[""],"Read more":[""],"Spanish":["Español"],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-26 16:07-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Median":["Mediana"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde la primera confirmación hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primera confirmación."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirá automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirá automáticamente una vez que se implementa en producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirá automáticamente luego de que el primer pipeline termine de ejecutarse."],"Total Time":["Tiempo total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
\ No newline at end of file
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index aeee577c1444e..747d446ebc41f 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -26,6 +26,9 @@ msgstr ""
 msgid "English"
 msgstr ""
 
+msgid "First"
+msgstr ""
+
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
@@ -44,12 +47,20 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
+msgid "Opened"
+msgstr ""
+
 msgid "Pipeline Health"
 msgstr ""
 
 msgid "Read more"
 msgstr ""
 
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
+
 msgid "Spanish"
 msgstr ""
 
@@ -82,3 +93,22 @@ msgstr ""
 
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
+
+msgid "by"
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "hr"
+msgstr ""
+
+msgid "min"
+msgid_plural "mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "pushed by"
+msgstr ""
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 9d32f6c9f8f25..d2ecc6cd73aff 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -26,6 +26,9 @@ msgstr ""
 msgid "English"
 msgstr ""
 
+msgid "First"
+msgstr ""
+
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
@@ -44,12 +47,20 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
+msgid "Opened"
+msgstr ""
+
 msgid "Pipeline Health"
 msgstr ""
 
 msgid "Read more"
 msgstr ""
 
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
+
 msgid "Spanish"
 msgstr ""
 
@@ -82,3 +93,22 @@ msgstr ""
 
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
+
+msgid "by"
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "hr"
+msgstr ""
+
+msgid "min"
+msgid_plural "mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "pushed by"
+msgstr ""
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index af7e127bc3a7d..404ab89c7cd94 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-04-13 00:07-0500\n"
+"PO-Revision-Date: 2017-04-26 16:07-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -18,7 +18,7 @@ msgstr ""
 "X-Generator: Poedit 2.0.1\n"
 
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
-msgstr ""
+msgstr "Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."
 
 msgid "Deutsch"
 msgstr "Alemán"
@@ -26,59 +26,90 @@ msgstr "Alemán"
 msgid "English"
 msgstr "Inglés"
 
+msgid "First"
+msgstr "Primer"
+
 msgid "Introducing Cycle Analytics"
-msgstr ""
+msgstr "Introducción a Cycle Analytics"
 
 msgid "Last 30 days"
-msgstr ""
+msgstr "Últimos 30 días"
 
 msgid "Last 90 days"
-msgstr ""
+msgstr "Últimos 90 días"
 
 msgid "Median"
-msgstr ""
+msgstr "Mediana"
 
 msgid "Not available"
-msgstr ""
+msgstr "No disponible"
 
 msgid "Not enough data"
-msgstr ""
+msgstr "No hay suficientes datos"
+
+msgid "Opened"
+msgstr "Abiertos"
 
 msgid "Pipeline Health"
-msgstr ""
+msgstr "Estado del Pipeline"
 
 msgid "Read more"
-msgstr ""
+msgstr "Leer más"
+
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] "Mostrando %d evento"
+msgstr[1] "Mostrando %d eventos"
 
 msgid "Spanish"
 msgstr "Español"
 
 msgid "Stage"
-msgstr ""
+msgstr "Etapa"
 
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
-msgstr ""
+msgstr "La etapa de codificación muestra el tiempo desde la primera confirmación hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."
 
 msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
-msgstr ""
+msgstr "La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."
 
 msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
-msgstr ""
+msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primera confirmación."
 
 msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
-msgstr ""
+msgstr "La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirá automáticamente una vez haya finalizado por completo el ciclo de idea a producción."
 
 msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
-msgstr ""
+msgstr "La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."
 
 msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
-msgstr ""
+msgstr "La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirá automáticamente una vez que se implementa en producción por primera vez."
 
+#, fuzzy
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
-msgstr ""
+msgstr "La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirá automáticamente luego de que el primer pipeline termine de ejecutarse."
 
 msgid "Total Time"
-msgstr ""
+msgstr "Tiempo total"
 
 msgid "Want to see the data? Please ask administrator for access."
+msgstr "¿Quieres ver los datos? Por favor pedir acceso al administrador."
+
+msgid "by"
+msgstr "por"
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] "día"
+msgstr[1] "días"
+
+msgid "hr"
 msgstr ""
+
+msgid "min"
+msgid_plural "mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "pushed by"
+msgstr "enviado por"
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index b99506788c72f..ab9ab88b24e4e 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-26 17:18+0100\n"
-"PO-Revision-Date: 2017-04-26 17:18+0100\n"
+"POT-Creation-Date: 2017-04-26 12:42-0500\n"
+"PO-Revision-Date: 2017-04-26 12:42-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -27,6 +27,9 @@ msgstr ""
 msgid "English"
 msgstr ""
 
+msgid "First"
+msgstr ""
+
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
@@ -45,12 +48,20 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
+msgid "Opened"
+msgstr ""
+
 msgid "Pipeline Health"
 msgstr ""
 
 msgid "Read more"
 msgstr ""
 
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
+
 msgid "Spanish"
 msgstr ""
 
@@ -83,3 +94,22 @@ msgstr ""
 
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
+
+msgid "by"
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "hr"
+msgstr ""
+
+msgid "min"
+msgid_plural "mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "pushed by"
+msgstr ""
-- 
GitLab


From edbf9880f95c4bcdd71bf4414f287d4d3f3cc290 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Wed, 26 Apr 2017 17:31:09 -0600
Subject: [PATCH 083/363] update task_list n/n

---
 .../issue_show/issue_title_description.vue    | 63 +++++++++++--------
 app/controllers/projects/issues_controller.rb |  2 +
 2 files changed, 40 insertions(+), 25 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index ef7ac83f336a0..d13e02b8cc0e0 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -32,7 +32,9 @@ export default {
       timeoutId: null,
       title: '<span></span>',
       description: '<span></span>',
+      descriptionText: '',
       descriptionChange: false,
+      taskStatus: '',
     };
   },
   methods: {
@@ -40,46 +42,35 @@ export default {
       const body = JSON.parse(res.body);
       this.triggerAnimation(body);
     },
-    triggerAnimation(body) {
-      // always reset to false before checking the change
-      this.descriptionChange = false;
-
-      const { title, description } = body;
-
-      this.descriptionText = body.description_text;
-
-      /**
-      * since opacity is changed, even if there is no diff for Vue to update
-      * we must check the title/description even on a 304 to ensure no visual change
-      */
-      const noTitleChange = this.title === title;
-      const noDescriptionChange = this.description === description;
-
-      if (noTitleChange && noDescriptionChange) return;
-
-      const elementsToVisualize = [];
+    updateTaskHTML(body) {
+      this.taskStatus = body.task_status;
+      document.querySelector('#task_status').innerText = this.taskStatus;
+    },
+    elementsToVisualize(noTitleChange, noDescriptionChange) {
+      const elementStack = [];
 
       if (!noTitleChange) {
-        elementsToVisualize.push(this.$el.querySelector('.title'));
+        elementStack.push(this.$el.querySelector('.title'));
       }
 
       if (!noDescriptionChange) {
         // only change to true when we need to bind TaskLists the html of description
         this.descriptionChange = true;
-
-        elementsToVisualize.push(this.$el.querySelector('.wiki'));
+        elementStack.push(this.$el.querySelector('.wiki'));
       }
 
-      elementsToVisualize.forEach((element) => {
+      elementStack.forEach((element) => {
         element.classList.remove('issue-realtime-trigger-pulse');
         element.classList.add('issue-realtime-pre-pulse');
       });
 
+      return elementStack;
+    },
+    animate(title, description, elementsToVisualize) {
       this.timeoutId = setTimeout(() => {
         this.title = title;
-        document.querySelector('title').innerText = title;
-
         this.description = description;
+        document.querySelector('title').innerText = title;
 
         elementsToVisualize.forEach((element) => {
           element.classList.remove('issue-realtime-pre-pulse');
@@ -89,6 +80,29 @@ export default {
         clearTimeout(this.timeoutId);
       }, 0);
     },
+    triggerAnimation(body) {
+      // always reset to false before checking the change
+      this.descriptionChange = false;
+
+      const { title, description } = body;
+      this.descriptionText = body.description_text;
+      this.updateTaskHTML(body);
+      /**
+      * since opacity is changed, even if there is no diff for Vue to update
+      * we must check the title/description even on a 304 to ensure no visual change
+      */
+      const noTitleChange = this.title === title;
+      const noDescriptionChange = this.description === description;
+
+      if (noTitleChange && noDescriptionChange) return;
+
+      const elementsToVisualize = this.elementsToVisualize(
+        noTitleChange,
+        noDescriptionChange,
+      );
+
+      this.animate(title, description, elementsToVisualize);
+    },
   },
   computed: {
     descriptionClass() {
@@ -118,7 +132,6 @@ export default {
       });
 
       $(this.$refs['issue-content-container-gfm-entry']).renderGFM();
-
       return tl;
     }
     return null;
diff --git a/app/controllers/projects/issues_controller.rb b/app/controllers/projects/issues_controller.rb
index b1df50ba84902..e5c1505ece65d 100644
--- a/app/controllers/projects/issues_controller.rb
+++ b/app/controllers/projects/issues_controller.rb
@@ -198,10 +198,12 @@ def can_create_branch
 
   def rendered_title
     Gitlab::PollingInterval.set_header(response, interval: 3_000)
+
     render json: {
       title: view_context.markdown_field(@issue, :title),
       description: view_context.markdown_field(@issue, :description),
       description_text: @issue.description,
+      task_status: @issue.task_status,
     }
   end
 
-- 
GitLab


From 4c9e68cb90740ce7967dd748e44c3792e21cf87c Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Wed, 26 Apr 2017 17:34:17 -0600
Subject: [PATCH 084/363] change 'body' to 'data' after 'JSON.parse'

---
 .../issue_show/issue_title_description.vue       | 16 ++++++++--------
 1 file changed, 8 insertions(+), 8 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index d13e02b8cc0e0..e21667f2ac709 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -39,11 +39,11 @@ export default {
   },
   methods: {
     renderResponse(res) {
-      const body = JSON.parse(res.body);
-      this.triggerAnimation(body);
+      const data = JSON.parse(res.body);
+      this.triggerAnimation(data);
     },
-    updateTaskHTML(body) {
-      this.taskStatus = body.task_status;
+    updateTaskHTML(data) {
+      this.taskStatus = data.task_status;
       document.querySelector('#task_status').innerText = this.taskStatus;
     },
     elementsToVisualize(noTitleChange, noDescriptionChange) {
@@ -80,13 +80,13 @@ export default {
         clearTimeout(this.timeoutId);
       }, 0);
     },
-    triggerAnimation(body) {
+    triggerAnimation(data) {
       // always reset to false before checking the change
       this.descriptionChange = false;
 
-      const { title, description } = body;
-      this.descriptionText = body.description_text;
-      this.updateTaskHTML(body);
+      const { title, description } = data;
+      this.descriptionText = data.description_text;
+      this.updateTaskHTML(data);
       /**
       * since opacity is changed, even if there is no diff for Vue to update
       * we must check the title/description even on a 304 to ensure no visual change
-- 
GitLab


From 0f1273fa44a9122bffbd5cecbaea99b1db781d7e Mon Sep 17 00:00:00 2001
From: Chris Wilson <chris@chrisjwilson.com>
Date: Fri, 14 Apr 2017 11:53:30 +1000
Subject: [PATCH 085/363] Add configurable timeout for git fetch and clone
 operations

GitLab uses the import_project method in GitLab Shell,
This method uses a timeout for the operation, hardcoded to 800 seconds.
With this MR the timeout is now configurable in the gitlab_shell
settings.
---
 .../mrchrisw-import-shell-timeout.yml         |  4 ++
 config/gitlab.yml.example                     |  3 ++
 config/initializers/1_settings.rb             |  1 +
 lib/gitlab/shell.rb                           |  4 +-
 spec/lib/gitlab/backend/shell_spec.rb         | 41 +++++++++++++++++++
 5 files changed, 51 insertions(+), 2 deletions(-)
 create mode 100644 changelogs/unreleased/mrchrisw-import-shell-timeout.yml

diff --git a/changelogs/unreleased/mrchrisw-import-shell-timeout.yml b/changelogs/unreleased/mrchrisw-import-shell-timeout.yml
new file mode 100644
index 0000000000000..e43409109d6c5
--- /dev/null
+++ b/changelogs/unreleased/mrchrisw-import-shell-timeout.yml
@@ -0,0 +1,4 @@
+---
+title: Add configurable timeout for git fetch and clone operations
+merge_request: 10697
+author: 
diff --git a/config/gitlab.yml.example b/config/gitlab.yml.example
index 06c9f734c2ade..ce477001b6fb8 100644
--- a/config/gitlab.yml.example
+++ b/config/gitlab.yml.example
@@ -502,6 +502,9 @@ production: &base
     upload_pack: true
     receive_pack: true
 
+    # Git import/fetch timeout
+    # git_timeout: 800
+
     # If you use non-standard ssh port you need to specify it
     # ssh_port: 22
 
diff --git a/config/initializers/1_settings.rb b/config/initializers/1_settings.rb
index 87bf48a3dcdd6..16bc08cf48f21 100644
--- a/config/initializers/1_settings.rb
+++ b/config/initializers/1_settings.rb
@@ -386,6 +386,7 @@ def cron_random_weekly_time
 Settings.gitlab_shell['ssh_user']     ||= Settings.gitlab.user
 Settings.gitlab_shell['owner_group']  ||= Settings.gitlab.user
 Settings.gitlab_shell['ssh_path_prefix'] ||= Settings.__send__(:build_gitlab_shell_ssh_path_prefix)
+Settings.gitlab_shell['git_timeout'] ||= 800
 
 #
 # Repositories
diff --git a/lib/gitlab/shell.rb b/lib/gitlab/shell.rb
index 36a871e5bbc00..b1d6ea665b70c 100644
--- a/lib/gitlab/shell.rb
+++ b/lib/gitlab/shell.rb
@@ -83,7 +83,7 @@ def import_repository(storage, name, url)
       # Timeout should be less than 900 ideally, to prevent the memory killer
       # to silently kill the process without knowing we are timing out here.
       output, status = Popen.popen([gitlab_shell_projects_path, 'import-project',
-                                    storage, "#{name}.git", url, '800'])
+                                    storage, "#{name}.git", url, "#{Gitlab.config.gitlab_shell.git_timeout}"])
       raise Error, output unless status.zero?
       true
     end
@@ -99,7 +99,7 @@ def import_repository(storage, name, url)
     #   fetch_remote("gitlab/gitlab-ci", "upstream")
     #
     def fetch_remote(storage, name, remote, forced: false, no_tags: false)
-      args = [gitlab_shell_projects_path, 'fetch-remote', storage, "#{name}.git", remote, '800']
+      args = [gitlab_shell_projects_path, 'fetch-remote', storage, "#{name}.git", remote, "#{Gitlab.config.gitlab_shell.git_timeout}"]
       args << '--force' if forced
       args << '--no-tags' if no_tags
 
diff --git a/spec/lib/gitlab/backend/shell_spec.rb b/spec/lib/gitlab/backend/shell_spec.rb
index 6675d26734e2e..a97a0f8452bd7 100644
--- a/spec/lib/gitlab/backend/shell_spec.rb
+++ b/spec/lib/gitlab/backend/shell_spec.rb
@@ -91,4 +91,45 @@
       end
     end
   end
+
+  describe 'projects commands' do
+    let(:projects_path) { 'tmp/tests/shell-projects-test/bin/gitlab-projects' }
+
+    before do
+      allow(Gitlab.config.gitlab_shell).to receive(:path).and_return('tmp/tests/shell-projects-test')
+      allow(Gitlab.config.gitlab_shell).to receive(:git_timeout).and_return(800)
+    end
+
+    describe '#fetch_remote' do
+      it 'returns true when the command succeeds' do
+        expect(Gitlab::Popen).to receive(:popen)
+          .with([projects_path, 'fetch-remote', 'current/storage', 'project/path.git', 'new/storage', '800']).and_return([nil, 0])
+
+        expect(gitlab_shell.fetch_remote('current/storage', 'project/path', 'new/storage')).to be true
+      end
+
+      it 'raises an exception when the command fails' do
+        expect(Gitlab::Popen).to receive(:popen)
+        .with([projects_path, 'fetch-remote', 'current/storage', 'project/path.git', 'new/storage', '800']).and_return(["error", 1])
+
+        expect { gitlab_shell.fetch_remote('current/storage', 'project/path', 'new/storage') }.to raise_error(Gitlab::Shell::Error, "error")
+      end
+    end
+
+    describe '#import_repository' do
+      it 'returns true when the command succeeds' do
+        expect(Gitlab::Popen).to receive(:popen)
+          .with([projects_path, 'import-project', 'current/storage', 'project/path.git', 'https://gitlab.com/gitlab-org/gitlab-ce.git', "800"]).and_return([nil, 0])
+
+        expect(gitlab_shell.import_repository('current/storage', 'project/path', 'https://gitlab.com/gitlab-org/gitlab-ce.git')).to be true
+      end
+
+      it 'raises an exception when the command fails' do
+        expect(Gitlab::Popen).to receive(:popen)
+        .with([projects_path, 'import-project', 'current/storage', 'project/path.git', 'https://gitlab.com/gitlab-org/gitlab-ce.git', "800"]).and_return(["error", 1])
+
+        expect { gitlab_shell.import_repository('current/storage', 'project/path', 'https://gitlab.com/gitlab-org/gitlab-ce.git') }.to raise_error(Gitlab::Shell::Error, "error")
+      end
+    end
+  end
 end
-- 
GitLab


From e4526e061cff78ce6dafc0bd5e3555c634161d3d Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 26 Apr 2017 20:04:33 -0500
Subject: [PATCH 086/363] Add more translation for Spanish.

---
 app/assets/javascripts/locale/de/app.js       |  2 +-
 app/assets/javascripts/locale/en/app.js       |  2 +-
 app/assets/javascripts/locale/es/app.js       |  2 +-
 .../projects/cycle_analytics/show.html.haml   |  4 +-
 lib/gitlab/cycle_analytics/code_stage.rb      |  2 +-
 lib/gitlab/cycle_analytics/issue_stage.rb     |  2 +-
 lib/gitlab/cycle_analytics/plan_stage.rb      |  2 +-
 .../cycle_analytics/production_stage.rb       |  2 +-
 lib/gitlab/cycle_analytics/review_stage.rb    |  2 +-
 lib/gitlab/cycle_analytics/staging_stage.rb   |  2 +-
 lib/gitlab/cycle_analytics/test_stage.rb      |  2 +-
 locale/de/gitlab.po                           | 27 ++++++++++++
 locale/en/gitlab.po                           | 27 ++++++++++++
 locale/es/gitlab.po                           | 44 +++++++++++++++----
 locale/gitlab.pot                             | 31 ++++++++++++-
 locale/unfound_translations.rb                |  3 ++
 16 files changed, 134 insertions(+), 22 deletions(-)
 create mode 100644 locale/unfound_translations.rb

diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index d74d36711d360..ba6adc7334578 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index 05ddc42e33386..d5d5b00505898 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index bb5976c572a6c..c47a743e2107a 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-26 16:07-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Median":["Mediana"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde la primera confirmación hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primera confirmación."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirá automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirá automáticamente una vez que se implementa en producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirá automáticamente luego de que el primer pipeline termine de ejecutarse."],"Total Time":["Tiempo total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-26 20:01-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Deploys":["Despliegues"],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
\ No newline at end of file
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index 65afe41fe9e53..9710c7a090597 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -32,7 +32,7 @@
           .row
             .col-sm-3.col-xs-12.column{ "v-for" => "item in state.summary" }
               %h3.header {{ item.value }}
-              %p.text {{ item.title }}
+              %p.text {{ __(item.title) }}
             .col-sm-3.col-xs-12.column
               .dropdown.inline.js-ca-dropdown
                 %button.dropdown-menu-toggle{ "data-toggle" => "dropdown", :type => "button" }
@@ -60,7 +60,7 @@
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.", "aria-hidden" => "true" }
               %li.event-header
                 %span.stage-name
-                  {{ __(currentStage ? currentStage.legend : 'Related Issues') }}
+                  {{ __(currentStage ? __(currentStage.legend) : __('Related Issues')) }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The collection of events added to the data gathered for that stage.", "aria-hidden" => "true" }
               %li.total-time-header
                 %span.stage-name
diff --git a/lib/gitlab/cycle_analytics/code_stage.rb b/lib/gitlab/cycle_analytics/code_stage.rb
index 1e52b6614a152..afb8196484775 100644
--- a/lib/gitlab/cycle_analytics/code_stage.rb
+++ b/lib/gitlab/cycle_analytics/code_stage.rb
@@ -14,7 +14,7 @@ def name
       end
 
       def legend
-        "Related Merge Requests"
+        N_("Related Merge Requests")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/issue_stage.rb b/lib/gitlab/cycle_analytics/issue_stage.rb
index 213994988a5a5..d36967d9497f8 100644
--- a/lib/gitlab/cycle_analytics/issue_stage.rb
+++ b/lib/gitlab/cycle_analytics/issue_stage.rb
@@ -15,7 +15,7 @@ def name
       end
 
       def legend
-        "Related Issues"
+        N_("Related Issues")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/plan_stage.rb b/lib/gitlab/cycle_analytics/plan_stage.rb
index 45d51d30ccc6b..7231c45378874 100644
--- a/lib/gitlab/cycle_analytics/plan_stage.rb
+++ b/lib/gitlab/cycle_analytics/plan_stage.rb
@@ -15,7 +15,7 @@ def name
       end
 
       def legend
-        "Related Commits"
+        N_("Related Commits")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/production_stage.rb b/lib/gitlab/cycle_analytics/production_stage.rb
index 9f387a0294533..451720c81214f 100644
--- a/lib/gitlab/cycle_analytics/production_stage.rb
+++ b/lib/gitlab/cycle_analytics/production_stage.rb
@@ -16,7 +16,7 @@ def name
       end
 
       def legend
-        "Related Issues"
+        N_("Related Issues")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/review_stage.rb b/lib/gitlab/cycle_analytics/review_stage.rb
index 4744be834de27..08e895acad628 100644
--- a/lib/gitlab/cycle_analytics/review_stage.rb
+++ b/lib/gitlab/cycle_analytics/review_stage.rb
@@ -14,7 +14,7 @@ def name
       end
 
       def legend
-        "Relative Merged Requests"
+        N_("Relative Merged Requests")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/staging_stage.rb b/lib/gitlab/cycle_analytics/staging_stage.rb
index 3cdbe04fbaf1d..706a5d14d3f35 100644
--- a/lib/gitlab/cycle_analytics/staging_stage.rb
+++ b/lib/gitlab/cycle_analytics/staging_stage.rb
@@ -15,7 +15,7 @@ def name
       end
 
       def legend
-        "Relative Deployed Builds"
+        N_("Relative Deployed Builds")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/test_stage.rb b/lib/gitlab/cycle_analytics/test_stage.rb
index e96943833bc46..887cf3e09416f 100644
--- a/lib/gitlab/cycle_analytics/test_stage.rb
+++ b/lib/gitlab/cycle_analytics/test_stage.rb
@@ -14,7 +14,7 @@ def name
       end
 
       def legend
-        "Relative Builds Trigger by Commits"
+        N_("Relative Builds Trigger by Commits")
       end
 
       def description
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index 747d446ebc41f..0a5db8ab1e5e7 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -17,9 +17,15 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
+msgid "Commits"
+msgstr ""
+
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
+msgid "Deploys"
+msgstr ""
+
 msgid "Deutsch"
 msgstr ""
 
@@ -41,6 +47,9 @@ msgstr ""
 msgid "Median"
 msgstr ""
 
+msgid "New Issues"
+msgstr ""
+
 msgid "Not available"
 msgstr ""
 
@@ -56,6 +65,24 @@ msgstr ""
 msgid "Read more"
 msgstr ""
 
+msgid "Related Commits"
+msgstr ""
+
+msgid "Related Issues"
+msgstr ""
+
+msgid "Related Merge Requests"
+msgstr ""
+
+msgid "Relative Builds Trigger by Commits"
+msgstr ""
+
+msgid "Relative Deployed Builds"
+msgstr ""
+
+msgid "Relative Merged Requests"
+msgstr ""
+
 msgid "Showing %d event"
 msgid_plural "Showing %d events"
 msgstr[0] ""
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index d2ecc6cd73aff..fac5c3ddb0273 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -17,9 +17,15 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
+msgid "Commits"
+msgstr ""
+
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
+msgid "Deploys"
+msgstr ""
+
 msgid "Deutsch"
 msgstr ""
 
@@ -41,6 +47,9 @@ msgstr ""
 msgid "Median"
 msgstr ""
 
+msgid "New Issues"
+msgstr ""
+
 msgid "Not available"
 msgstr ""
 
@@ -56,6 +65,24 @@ msgstr ""
 msgid "Read more"
 msgstr ""
 
+msgid "Related Commits"
+msgstr ""
+
+msgid "Related Issues"
+msgstr ""
+
+msgid "Related Merge Requests"
+msgstr ""
+
+msgid "Relative Builds Trigger by Commits"
+msgstr ""
+
+msgid "Relative Deployed Builds"
+msgstr ""
+
+msgid "Relative Merged Requests"
+msgstr ""
+
 msgid "Showing %d event"
 msgid_plural "Showing %d events"
 msgstr[0] ""
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 404ab89c7cd94..8bb4176dbf45a 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-04-26 16:07-0500\n"
+"PO-Revision-Date: 2017-04-26 20:01-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -17,9 +17,15 @@ msgstr ""
 "Last-Translator: \n"
 "X-Generator: Poedit 2.0.1\n"
 
+msgid "Commits"
+msgstr "Cambios"
+
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr "Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."
 
+msgid "Deploys"
+msgstr "Despliegues"
+
 msgid "Deutsch"
 msgstr "Alemán"
 
@@ -41,6 +47,9 @@ msgstr "Últimos 90 días"
 msgid "Median"
 msgstr "Mediana"
 
+msgid "New Issues"
+msgstr "Nuevos temas"
+
 msgid "Not available"
 msgstr "No disponible"
 
@@ -56,6 +65,26 @@ msgstr "Estado del Pipeline"
 msgid "Read more"
 msgstr "Leer más"
 
+msgid "Related Commits"
+msgstr "Cambios Relacionados"
+
+msgid "Related Issues"
+msgstr "Temas Relacionados"
+
+msgid "Related Merge Requests"
+msgstr "Solicitudes de fusión Relacionadas"
+
+#, fuzzy
+msgid "Relative Builds Trigger by Commits"
+msgstr "Builds Relacionadas Originadas por Cambios"
+
+#, fuzzy
+msgid "Relative Deployed Builds"
+msgstr "Builds Desplegadas Relacionadas"
+
+msgid "Relative Merged Requests"
+msgstr "Solicitudes de fusión Relacionadas"
+
 msgid "Showing %d event"
 msgid_plural "Showing %d events"
 msgstr[0] "Mostrando %d evento"
@@ -68,29 +97,28 @@ msgid "Stage"
 msgstr "Etapa"
 
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
-msgstr "La etapa de codificación muestra el tiempo desde la primera confirmación hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."
+msgstr "La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."
 
 msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr "La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."
 
 msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
-msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primera confirmación."
+msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."
 
 msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
-msgstr "La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirá automáticamente una vez haya finalizado por completo el ciclo de idea a producción."
+msgstr "La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."
 
 msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
 msgstr "La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."
 
 msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
-msgstr "La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirá automáticamente una vez que se implementa en producción por primera vez."
+msgstr "La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."
 
-#, fuzzy
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
-msgstr "La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirá automáticamente luego de que el primer pipeline termine de ejecutarse."
+msgstr "La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."
 
 msgid "Total Time"
-msgstr "Tiempo total"
+msgstr "Tiempo Total"
 
 msgid "Want to see the data? Please ask administrator for access."
 msgstr "¿Quieres ver los datos? Por favor pedir acceso al administrador."
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index ab9ab88b24e4e..9e2614cb8ba8f 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-26 12:42-0500\n"
-"PO-Revision-Date: 2017-04-26 12:42-0500\n"
+"POT-Creation-Date: 2017-04-26 19:52-0500\n"
+"PO-Revision-Date: 2017-04-26 19:52-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -18,9 +18,15 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
 
+msgid "Commits"
+msgstr ""
+
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
+msgid "Deploys"
+msgstr ""
+
 msgid "Deutsch"
 msgstr ""
 
@@ -42,6 +48,9 @@ msgstr ""
 msgid "Median"
 msgstr ""
 
+msgid "New Issues"
+msgstr ""
+
 msgid "Not available"
 msgstr ""
 
@@ -57,6 +66,24 @@ msgstr ""
 msgid "Read more"
 msgstr ""
 
+msgid "Related Commits"
+msgstr ""
+
+msgid "Related Issues"
+msgstr ""
+
+msgid "Related Merge Requests"
+msgstr ""
+
+msgid "Relative Builds Trigger by Commits"
+msgstr ""
+
+msgid "Relative Deployed Builds"
+msgstr ""
+
+msgid "Relative Merged Requests"
+msgstr ""
+
 msgid "Showing %d event"
 msgid_plural "Showing %d events"
 msgstr[0] ""
diff --git a/locale/unfound_translations.rb b/locale/unfound_translations.rb
new file mode 100644
index 0000000000000..a88300ba43f4b
--- /dev/null
+++ b/locale/unfound_translations.rb
@@ -0,0 +1,3 @@
+N_('New Issues')
+N_('Commits')
+N_('Deploys')
-- 
GitLab


From b3c80094e0db4d390522daef7a845ff2c5744bf3 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Thu, 27 Apr 2017 14:47:00 +0100
Subject: [PATCH 087/363] Added some missing text that needs translating

---
 .../components/limit_warning_component.js     |  4 ++--
 app/assets/javascripts/locale/de/app.js       |  2 +-
 app/assets/javascripts/locale/en/app.js       |  2 +-
 app/assets/javascripts/locale/es/app.js       |  2 +-
 .../projects/cycle_analytics/show.html.haml   | 10 +++++-----
 locale/de/gitlab.po                           | 15 +++++++++++++++
 locale/en/gitlab.po                           | 15 +++++++++++++++
 locale/es/gitlab.po                           | 16 ++++++++++++++++
 locale/gitlab.pot                             | 19 +++++++++++++++++--
 9 files changed, 73 insertions(+), 12 deletions(-)

diff --git a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
index 72ada943fea0f..6e07560040c95 100644
--- a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
@@ -6,10 +6,10 @@ export default {
     },
   },
   template: `
-    <span v-if="count === 50" class="events-info pull-right">
+    <span v-if="count === 50 || true" class="events-info pull-right">
       <i class="fa fa-warning has-tooltip"
           aria-hidden="true"
-          title="Limited to showing 50 events at most"
+          :title="__('Limited to showing 50 events at most')"
           data-placement="top"></i>
       {{ n__('Showing %d event', 'Showing %d events', 50) }}
     </span>
diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index ba6adc7334578..f59109f247f99 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index d5d5b00505898..ec64122d1a95f 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index c47a743e2107a..64a4449c260e0 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-26 20:01-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Deploys":["Despliegues"],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-26 20:01-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Deploys":["Despliegues"],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Limited to showing 50 events at most":[""],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["por"],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
\ No newline at end of file
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index 9710c7a090597..b088dfe344158 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -53,19 +53,19 @@
               %li.stage-header
                 %span.stage-name
                   {{ __('Stage') }}
-                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The phase of the development lifecycle.", "aria-hidden" => "true" }
+                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The phase of the development lifecycle."), "aria-hidden" => "true" }
               %li.median-header
                 %span.stage-name
                   {{ __('Median') }}
-                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.", "aria-hidden" => "true" }
+                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."), "aria-hidden" => "true" }
               %li.event-header
                 %span.stage-name
-                  {{ __(currentStage ? __(currentStage.legend) : __('Related Issues')) }}
-                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The collection of events added to the data gathered for that stage.", "aria-hidden" => "true" }
+                  {{ currentStage ? currentStage.legend : __('Related Issues') }}
+                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The collection of events added to the data gathered for that stage."), "aria-hidden" => "true" }
               %li.total-time-header
                 %span.stage-name
                   {{ __('Total Time') }}
-                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The time taken by each data entry gathered by that stage.", "aria-hidden" => "true" }
+                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The time taken by each data entry gathered by that stage."), "aria-hidden" => "true" }
         .stage-panel-body
           %nav.stage-nav
             %ul
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index 0a5db8ab1e5e7..bbdd4e4a0be80 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -44,6 +44,9 @@ msgstr ""
 msgid "Last 90 days"
 msgstr ""
 
+msgid "Limited to showing 50 events at most"
+msgstr ""
+
 msgid "Median"
 msgstr ""
 
@@ -97,9 +100,15 @@ msgstr ""
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
+msgid "The collection of events added to the data gathered for that stage."
+msgstr ""
+
 msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr ""
 
+msgid "The phase of the development lifecycle."
+msgstr ""
+
 msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr ""
 
@@ -115,6 +124,12 @@ msgstr ""
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
+msgid "The time taken by each data entry gathered by that stage."
+msgstr ""
+
+msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
+msgstr ""
+
 msgid "Total Time"
 msgstr ""
 
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index fac5c3ddb0273..4942149dd1fe4 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -44,6 +44,9 @@ msgstr ""
 msgid "Last 90 days"
 msgstr ""
 
+msgid "Limited to showing 50 events at most"
+msgstr ""
+
 msgid "Median"
 msgstr ""
 
@@ -97,9 +100,15 @@ msgstr ""
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
+msgid "The collection of events added to the data gathered for that stage."
+msgstr ""
+
 msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr ""
 
+msgid "The phase of the development lifecycle."
+msgstr ""
+
 msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr ""
 
@@ -115,6 +124,12 @@ msgstr ""
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
+msgid "The time taken by each data entry gathered by that stage."
+msgstr ""
+
+msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
+msgstr ""
+
 msgid "Total Time"
 msgstr ""
 
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 8bb4176dbf45a..bad2b6b72a25f 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -44,6 +44,9 @@ msgstr "Últimos 30 días"
 msgid "Last 90 days"
 msgstr "Últimos 90 días"
 
+msgid "Limited to showing 50 events at most"
+msgstr ""
+
 msgid "Median"
 msgstr "Mediana"
 
@@ -99,9 +102,15 @@ msgstr "Etapa"
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr "La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."
 
+msgid "The collection of events added to the data gathered for that stage."
+msgstr ""
+
 msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr "La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."
 
+msgid "The phase of the development lifecycle."
+msgstr ""
+
 msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."
 
@@ -117,6 +126,13 @@ msgstr "La etapa de puesta en escena muestra el tiempo entre la fusión  y el de
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr "La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."
 
+#, fuzzy
+msgid "The time taken by each data entry gathered by that stage."
+msgstr "por"
+
+msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
+msgstr ""
+
 msgid "Total Time"
 msgstr "Tiempo Total"
 
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index 9e2614cb8ba8f..faac5f13aedec 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-26 19:52-0500\n"
-"PO-Revision-Date: 2017-04-26 19:52-0500\n"
+"POT-Creation-Date: 2017-04-27 14:41+0100\n"
+"PO-Revision-Date: 2017-04-27 14:41+0100\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -45,6 +45,9 @@ msgstr ""
 msgid "Last 90 days"
 msgstr ""
 
+msgid "Limited to showing 50 events at most"
+msgstr ""
+
 msgid "Median"
 msgstr ""
 
@@ -98,9 +101,15 @@ msgstr ""
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
+msgid "The collection of events added to the data gathered for that stage."
+msgstr ""
+
 msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr ""
 
+msgid "The phase of the development lifecycle."
+msgstr ""
+
 msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr ""
 
@@ -116,6 +125,12 @@ msgstr ""
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr ""
 
+msgid "The time taken by each data entry gathered by that stage."
+msgstr ""
+
+msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
+msgstr ""
+
 msgid "Total Time"
 msgstr ""
 
-- 
GitLab


From abde62b53ed993b4ceec778d4fb839fa0132c165 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 27 Apr 2017 17:06:55 +0100
Subject: [PATCH 088/363] [ci skip] Add comment to regex

---
 app/assets/javascripts/droplab/constants.js | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/app/assets/javascripts/droplab/constants.js b/app/assets/javascripts/droplab/constants.js
index 0579b5de45938..868d47e91b3e4 100644
--- a/app/assets/javascripts/droplab/constants.js
+++ b/app/assets/javascripts/droplab/constants.js
@@ -2,8 +2,9 @@ const DATA_TRIGGER = 'data-dropdown-trigger';
 const DATA_DROPDOWN = 'data-dropdown';
 const SELECTED_CLASS = 'droplab-item-selected';
 const ACTIVE_CLASS = 'droplab-item-active';
-const TEMPLATE_REGEX = /\{\{(.+?)\}\}/g;
 const IGNORE_CLASS = 'droplab-item-ignore';
+// Matches `{{anything}}` and `{{ everything }}`.
+const TEMPLATE_REGEX = /\{\{(.+?)\}\}/g;
 
 export {
   DATA_TRIGGER,
-- 
GitLab


From 22ecf73e823f01e155b2db2cb072120f80b9d234 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 27 Apr 2017 11:51:04 -0500
Subject: [PATCH 089/363] Adjust regex used in order to parse inline content.

e.g: %h4 {{ __('foo') }}

Problem was it was being parsed as: %h4 = _('foo')
And the correct content should be: %h4= _('foo')
---
 app/assets/javascripts/locale/de/app.js         | 2 +-
 app/assets/javascripts/locale/en/app.js         | 2 +-
 app/assets/javascripts/locale/es/app.js         | 2 +-
 config/initializers/gettext_rails_i18n_patch.rb | 4 ++--
 locale/de/gitlab.po                             | 6 ++++++
 locale/en/gitlab.po                             | 6 ++++++
 locale/es/gitlab.po                             | 8 +++++++-
 locale/gitlab.pot                               | 6 ++++++
 8 files changed, 30 insertions(+), 6 deletions(-)

diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index f59109f247f99..9fd02783bdb97 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index ec64122d1a95f..8bc82f121714a 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index 64a4449c260e0..dfe33cb6eb92c 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-26 20:01-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Deploys":["Despliegues"],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Limited to showing 50 events at most":[""],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["por"],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-27 11:45-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Deploys":["Despliegues"],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Limited to showing 50 events at most":[""],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["por"],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
index 6785f361eda77..616f6c45b0add 100644
--- a/config/initializers/gettext_rails_i18n_patch.rb
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -1,7 +1,7 @@
 require 'gettext_i18n_rails/haml_parser'
 require 'gettext_i18n_rails_js/parser/javascript'
 
-VUE_TRANSLATE_REGEX = /{{ (N|n|s)?__\((.*)\) }}/
+VUE_TRANSLATE_REGEX = /((%[\w.-]+)(?:\s))?{{ (N|n|s)?__\((.*)\) }}/
 
 module GettextI18nRails
   class HamlParser
@@ -15,7 +15,7 @@ class HamlParser
     #
     # Overwrites: https://github.com/grosser/gettext_i18n_rails/blob/8396387a431e0f8ead72fc1cd425cad2fa4992f2/lib/gettext_i18n_rails/haml_parser.rb#L9
     def self.convert_to_code(text)
-      text.gsub!(VUE_TRANSLATE_REGEX, "= \\1_(\\2)")
+      text.gsub!(VUE_TRANSLATE_REGEX, "\\2= \\3_(\\4)")
 
       old_convert_to_code(text)
     end
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index bbdd4e4a0be80..108273898d03c 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -136,6 +136,12 @@ msgstr ""
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
 
+msgid "We don't have enough data to show this stage."
+msgstr ""
+
+msgid "You need permission."
+msgstr ""
+
 msgid "by"
 msgstr ""
 
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 4942149dd1fe4..b33e941a4f339 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -136,6 +136,12 @@ msgstr ""
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
 
+msgid "We don't have enough data to show this stage."
+msgstr ""
+
+msgid "You need permission."
+msgstr ""
+
 msgid "by"
 msgstr ""
 
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index bad2b6b72a25f..f3d77cffecdf3 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-04-26 20:01-0500\n"
+"PO-Revision-Date: 2017-04-27 11:45-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -139,6 +139,12 @@ msgstr "Tiempo Total"
 msgid "Want to see the data? Please ask administrator for access."
 msgstr "¿Quieres ver los datos? Por favor pedir acceso al administrador."
 
+msgid "We don't have enough data to show this stage."
+msgstr "No hay suficientes datos para mostrar en esta etapa."
+
+msgid "You need permission."
+msgstr "Necesitas permisos."
+
 msgid "by"
 msgstr "por"
 
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index faac5f13aedec..9d3be20e13224 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -137,6 +137,12 @@ msgstr ""
 msgid "Want to see the data? Please ask administrator for access."
 msgstr ""
 
+msgid "We don't have enough data to show this stage."
+msgstr ""
+
+msgid "You need permission."
+msgstr ""
+
 msgid "by"
 msgstr ""
 
-- 
GitLab


From f74d0ba195079260d3b3217cd72b2fd9b0483c00 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 27 Apr 2017 17:58:51 +0100
Subject: [PATCH 090/363] Fix webpack config conflicts bad resolution

---
 config/webpack.config.js | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/config/webpack.config.js b/config/webpack.config.js
index f61e65c0e5bb0..8b5354809e746 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -46,6 +46,7 @@ var config = {
     network:              './network/network_bundle.js',
     notebook_viewer:      './blob/notebook_viewer.js',
     pdf_viewer:           './blob/pdf_viewer.js',
+    pipelines:            './pipelines/index.js',
     balsamiq_viewer:      './blob/balsamiq_viewer.js',
     profile:              './profile/profile_bundle.js',
     protected_branches:   './protected_branches/protected_branches_bundle.js',
@@ -129,6 +130,7 @@ var config = {
         'merge_conflicts',
         'notebook_viewer',
         'pdf_viewer',
+        'pipelines',
         'balsamiq_viewer',
       ],
       minChunks: function(module, count) {
-- 
GitLab


From 657664f9801c90ded9ede327a7c8b11016809a29 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 27 Apr 2017 17:56:21 -0500
Subject: [PATCH 091/363] Add more spanish translations.

---
 app/assets/javascripts/locale/de/app.js       |  2 +-
 app/assets/javascripts/locale/en/app.js       |  2 +-
 app/assets/javascripts/locale/es/app.js       |  2 +-
 .../projects/cycle_analytics/show.html.haml   |  4 +-
 locale/de/gitlab.po                           | 18 +++++++++
 locale/en/gitlab.po                           | 18 +++++++++
 locale/es/gitlab.po                           | 37 +++++++++++++++----
 locale/gitlab.pot                             | 22 ++++++++++-
 locale/unfound_translations.rb                |  8 +++-
 9 files changed, 98 insertions(+), 15 deletions(-)

diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index 9fd02783bdb97..1b10c19c48e78 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index 8bc82f121714a..2754e177653b0 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index dfe33cb6eb92c..5d1cfddf76abf 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-27 11:45-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Deploys":["Despliegues"],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Limited to showing 50 events at most":[""],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["por"],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-27 19:15-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Cycle|Code":["Código"],"Cycle|Issue":["Tema"],"Cycle|Plan":["Planificación"],"Cycle|Review":["Revisión"],"Cycle|Staging":["Etapa"],"Cycle|Test":["Pruebas"],"Deploys":["Despliegues"],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Limited to showing 50 events at most":["Limitado a mostrar máximo 50 eventos"],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
\ No newline at end of file
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index b088dfe344158..a1a227d2fe9d0 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -60,7 +60,7 @@
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."), "aria-hidden" => "true" }
               %li.event-header
                 %span.stage-name
-                  {{ currentStage ? currentStage.legend : __('Related Issues') }}
+                  {{ currentStage ? __(currentStage.legend) : __('Related Issues') }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The collection of events added to the data gathered for that stage."), "aria-hidden" => "true" }
               %li.total-time-header
                 %span.stage-name
@@ -71,7 +71,7 @@
             %ul
               %li.stage-nav-item{ ':class' => '{ active: stage.active }', '@click' => 'selectStage(stage)', "v-for" => "stage in state.stages" }
                 .stage-nav-item-cell.stage-name
-                  {{ stage.title }}
+                  {{ s__('Cycle', stage.title) }}
                 .stage-nav-item-cell.stage-median
                   %template{ "v-if" => "stage.isUserAllowed" }
                     %span{ "v-if" => "stage.value" }
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index 108273898d03c..843648c25f1fe 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -23,6 +23,24 @@ msgstr ""
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
+msgid "Cycle|Code"
+msgstr ""
+
+msgid "Cycle|Issue"
+msgstr ""
+
+msgid "Cycle|Plan"
+msgstr ""
+
+msgid "Cycle|Review"
+msgstr ""
+
+msgid "Cycle|Staging"
+msgstr ""
+
+msgid "Cycle|Test"
+msgstr ""
+
 msgid "Deploys"
 msgstr ""
 
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index b33e941a4f339..e98931685d753 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -23,6 +23,24 @@ msgstr ""
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
+msgid "Cycle|Code"
+msgstr ""
+
+msgid "Cycle|Issue"
+msgstr ""
+
+msgid "Cycle|Plan"
+msgstr ""
+
+msgid "Cycle|Review"
+msgstr ""
+
+msgid "Cycle|Staging"
+msgstr ""
+
+msgid "Cycle|Test"
+msgstr ""
+
 msgid "Deploys"
 msgstr ""
 
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index f3d77cffecdf3..644007f514928 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-04-27 11:45-0500\n"
+"PO-Revision-Date: 2017-04-27 19:15-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -23,6 +23,30 @@ msgstr "Cambios"
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr "Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."
 
+#, fuzzy
+msgid "Cycle|Code"
+msgstr "Código"
+
+#, fuzzy
+msgid "Cycle|Issue"
+msgstr "Tema"
+
+#, fuzzy
+msgid "Cycle|Plan"
+msgstr "Planificación"
+
+#, fuzzy
+msgid "Cycle|Review"
+msgstr "Revisión"
+
+#, fuzzy
+msgid "Cycle|Staging"
+msgstr "Etapa"
+
+#, fuzzy
+msgid "Cycle|Test"
+msgstr "Pruebas"
+
 msgid "Deploys"
 msgstr "Despliegues"
 
@@ -45,7 +69,7 @@ msgid "Last 90 days"
 msgstr "Últimos 90 días"
 
 msgid "Limited to showing 50 events at most"
-msgstr ""
+msgstr "Limitado a mostrar máximo 50 eventos"
 
 msgid "Median"
 msgstr "Mediana"
@@ -103,13 +127,13 @@ msgid "The coding stage shows the time from the first commit to creating the mer
 msgstr "La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."
 
 msgid "The collection of events added to the data gathered for that stage."
-msgstr ""
+msgstr "La colección de eventos agregados a los datos recopilados para esa etapa."
 
 msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
 msgstr "La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."
 
 msgid "The phase of the development lifecycle."
-msgstr ""
+msgstr "La etapa del ciclo de vida del desarrollo."
 
 msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
 msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."
@@ -126,12 +150,11 @@ msgstr "La etapa de puesta en escena muestra el tiempo entre la fusión  y el de
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr "La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."
 
-#, fuzzy
 msgid "The time taken by each data entry gathered by that stage."
-msgstr "por"
+msgstr "El tiempo utilizado por cada entrada de datos obtenido por esa etapa."
 
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
-msgstr ""
+msgstr "El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."
 
 msgid "Total Time"
 msgstr "Tiempo Total"
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index 9d3be20e13224..1b60f9dad7a57 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-27 14:41+0100\n"
-"PO-Revision-Date: 2017-04-27 14:41+0100\n"
+"POT-Creation-Date: 2017-04-27 19:13-0500\n"
+"PO-Revision-Date: 2017-04-27 19:13-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -24,6 +24,24 @@ msgstr ""
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
+msgid "Cycle|Code"
+msgstr ""
+
+msgid "Cycle|Issue"
+msgstr ""
+
+msgid "Cycle|Plan"
+msgstr ""
+
+msgid "Cycle|Review"
+msgstr ""
+
+msgid "Cycle|Staging"
+msgstr ""
+
+msgid "Cycle|Test"
+msgstr ""
+
 msgid "Deploys"
 msgstr ""
 
diff --git a/locale/unfound_translations.rb b/locale/unfound_translations.rb
index a88300ba43f4b..e38b49b48a17c 100644
--- a/locale/unfound_translations.rb
+++ b/locale/unfound_translations.rb
@@ -1,3 +1,9 @@
-N_('New Issues')
 N_('Commits')
+N_('Cycle|Code')
+N_('Cycle|Issue')
+N_('Cycle|Plan')
+N_('Cycle|Review')
+N_('Cycle|Staging')
+N_('Cycle|Test')
 N_('Deploys')
+N_('New Issues')
-- 
GitLab


From a922d90414fea5b05152717a72f169c0d9ef7d09 Mon Sep 17 00:00:00 2001
From: Jacob Schatz <jschatz1@gmail.com>
Date: Thu, 27 Apr 2017 22:47:50 -0400
Subject: [PATCH 092/363] Add Vue with async deploy keys

---
 app/assets/javascripts/dispatcher.js          |  3 +
 .../settings/settings_repository.js           | 92 +++++++++++++++++++
 .../projects/deploy_keys/_index.html.haml     | 33 +++----
 3 files changed, 107 insertions(+), 21 deletions(-)
 create mode 100644 app/assets/javascripts/settings/settings_repository.js

diff --git a/app/assets/javascripts/dispatcher.js b/app/assets/javascripts/dispatcher.js
index b20673cd03c8d..2300daf07274a 100644
--- a/app/assets/javascripts/dispatcher.js
+++ b/app/assets/javascripts/dispatcher.js
@@ -48,6 +48,7 @@ import BlobForkSuggestion from './blob/blob_fork_suggestion';
 import UserCallout from './user_callout';
 import { ProtectedTagCreate, ProtectedTagEditList } from './protected_tags';
 import ShortcutsWiki from './shortcuts_wiki';
+import SettingsDeployKeys from './settings/settings_repository';
 
 const ShortcutsBlob = require('./shortcuts_blob');
 
@@ -343,6 +344,8 @@ const ShortcutsBlob = require('./shortcuts_blob');
           // Initialize Protected Tag Settings
           new ProtectedTagCreate();
           new ProtectedTagEditList();
+          // Initialize deploy key ajax call
+          new SettingsDeployKeys();
           break;
         case 'projects:ci_cd:show':
           new gl.ProjectVariables();
diff --git a/app/assets/javascripts/settings/settings_repository.js b/app/assets/javascripts/settings/settings_repository.js
new file mode 100644
index 0000000000000..4c6f769f53301
--- /dev/null
+++ b/app/assets/javascripts/settings/settings_repository.js
@@ -0,0 +1,92 @@
+/* eslint-disable no-new */
+import Vue from 'vue';
+import VueResource from 'vue-resource';
+
+Vue.use(VueResource);
+
+export default class SettingsDeployKeys {
+  constructor() {
+    this.initVue();
+  }
+
+  deployKeyRowTemplate() {
+    return `
+      <li>
+        <div class="pull-left append-right-10 hidden-xs">
+          <i aria-hidden="true" class="fa fa-key key-icon"></i>
+        </div>
+        <div class="deploy-key-content key-list-item-info">
+          <strong class="title">
+            {{deployKey.title}}
+          </strong>
+          <div class="description">
+            {{deployKey.fingerprint}}
+          </div>
+        </div>
+        <div class="deploy-key-content prepend-left-default deploy-key-projects">
+          <a class="label deploy-project-label" :href="project.full_path" v-for="project in deployKey.projects">{{project.full_name}}</a>
+        </div>
+        <div class="deploy-key-content">
+          <span class="key-created-at">
+            created {{deployKey.created_at}}
+          </span>
+          <div class="visible-xs-block visible-sm-block"></div>
+          <a v-if="!enabled" class="btn btn-sm prepend-left-10" rel="nofollow" data-method="put" href="enableURL">Enable
+</a>
+          <a v-else-if="deployKey.destroyed_when_orphaned && deployKey.almost_orphaned" class="btn btn-warning btn-sm prepend-left-10" rel="nofollow" data-method="put" :href="removeURL">Remove</a>
+          <a v-else class="btn btn-warning btn-sm prepend-left-10" rel="nofollow" data-method="put" :href="disableURL">Disable</a>
+        </div>
+      </li>`
+  }
+
+  deployKeyRowComponent() {
+    const self = this;
+    return {
+      props: {
+        deployKey: Object,
+        enabled: Boolean
+      },
+
+      computed: {
+        disableURL() {
+          return self.disableEndpoint.replace(':id', this.deployKey.id);
+        },
+
+        enableURL() {
+         return self.enableEndpoint.replace(':id', this.deployKey.id); 
+        }
+      },
+
+      template: this.deployKeyRowTemplate()
+    }
+  }
+
+  initVue() {
+    const self = this;
+    const el = document.getElementById('js-deploy-keys');
+    const endpoint = el.dataset.endpoint;
+    this.jsonEndpoint = `${endpoint}.json`;
+    this.disableEndpoint = `${endpoint}/:id/disable`;
+    this.enableEndpoint = `${endpoint}/:id/enable`;
+    new Vue({
+      el: el,
+      components: {
+        deployKeyRow: self.deployKeyRowComponent()
+      },
+      data () {
+        return {
+          enabledKeys: [],
+          availableKeys: []
+        }
+      },
+      created () {
+        this.$http.get(self.jsonEndpoint)
+          .then((res) => {
+            const keys = JSON.parse(res.body);
+            this.enabledKeys = keys.enabled_keys;
+            this.availableKeys = keys.available_project_keys;
+          });
+      }
+    })
+  }
+} 
\ No newline at end of file
diff --git a/app/views/projects/deploy_keys/_index.html.haml b/app/views/projects/deploy_keys/_index.html.haml
index 4cfbd9add009d..b35cd356aa507 100644
--- a/app/views/projects/deploy_keys/_index.html.haml
+++ b/app/views/projects/deploy_keys/_index.html.haml
@@ -10,25 +10,16 @@
     = render @deploy_keys.form_partial_path
   .col-lg-9.col-lg-offset-3
     %hr
-  .col-lg-9.col-lg-offset-3.append-bottom-default.deploy-keys
+  #js-deploy-keys.col-lg-9.col-lg-offset-3.append-bottom-default{data:{endpoint: namespace_project_deploy_keys_path}}
     %h5.prepend-top-0
-      Enabled deploy keys for this project (#{@deploy_keys.enabled_keys_size})
-    - if @deploy_keys.any_keys_enabled?
-      %ul.well-list
-        = render partial: 'projects/deploy_keys/deploy_key', collection: @deploy_keys.enabled_keys, as: :deploy_key
-    - else
-      .settings-message.text-center
-        No deploy keys found. Create one with the form above.
-    %h5.prepend-top-default
-      Deploy keys from projects you have access to (#{@deploy_keys.available_project_keys_size})
-    - if @deploy_keys.any_available_project_keys_enabled?
-      %ul.well-list
-        = render partial: 'projects/deploy_keys/deploy_key', collection: @deploy_keys.available_project_keys, as: :deploy_key
-    - else
-      .settings-message.text-center
-        No deploy keys from your projects could be found. Create one with the form above or add existing one below.
-    - if @deploy_keys.any_available_public_keys_enabled?
-      %h5.prepend-top-default
-        Public deploy keys available to any project (#{@deploy_keys.available_public_keys_size})
-      %ul.well-list
-        = render partial: 'projects/deploy_keys/deploy_key', collection: @deploy_keys.available_public_keys, as: :deploy_key
+      Enabled deploy keys for this project ({{enabledKeys.length}})
+    %ul.well-list{"v-if" => "enabledKeys.length"}
+      %deploy-key-row{"v-for" => "deployKey in enabledKeys", ":deploy-key" => "deployKey", ":enabled" =>"true"}
+    .settings-message.text-center{"v-else" => true}
+      No deploy keys found. Create one with the form above.
+    %h5.prepend-top-0
+      Deploy keys from projects you have access to ({{availableKeys.length}})
+    %ul.well-list{"v-if" => "availableKeys.length"}
+      %deploy-key-row{"v-for" => "deployKey in availableKeys", ":deploy-key" => "deployKey", ":enabled" =>"false"}
+    .settings-message.text-center{"v-else" => true}
+      No deploy keys found. Create one with the form above.
-- 
GitLab


From b5b33200ccef223e089edf6ad8cdd0ecc3e737da Mon Sep 17 00:00:00 2001
From: Jacob Schatz <jschatz1@gmail.com>
Date: Thu, 27 Apr 2017 23:05:22 -0400
Subject: [PATCH 093/363] Add timeago

---
 app/assets/javascripts/settings/settings_repository.js | 8 ++++++--
 1 file changed, 6 insertions(+), 2 deletions(-)

diff --git a/app/assets/javascripts/settings/settings_repository.js b/app/assets/javascripts/settings/settings_repository.js
index 4c6f769f53301..9d6a60baa3ad4 100644
--- a/app/assets/javascripts/settings/settings_repository.js
+++ b/app/assets/javascripts/settings/settings_repository.js
@@ -28,12 +28,12 @@ export default class SettingsDeployKeys {
         </div>
         <div class="deploy-key-content">
           <span class="key-created-at">
-            created {{deployKey.created_at}}
+            created {{timeagoDate}}
           </span>
           <div class="visible-xs-block visible-sm-block"></div>
           <a v-if="!enabled" class="btn btn-sm prepend-left-10" rel="nofollow" data-method="put" href="enableURL">Enable
 </a>
-          <a v-else-if="deployKey.destroyed_when_orphaned && deployKey.almost_orphaned" class="btn btn-warning btn-sm prepend-left-10" rel="nofollow" data-method="put" :href="removeURL">Remove</a>
+          <a v-else-if="deployKey.destroyed_when_orphaned && deployKey.almost_orphaned" class="btn btn-warning btn-sm prepend-left-10" rel="nofollow" data-method="put" :href="disableURL">Remove</a>
           <a v-else class="btn btn-warning btn-sm prepend-left-10" rel="nofollow" data-method="put" :href="disableURL">Disable</a>
         </div>
       </li>`
@@ -48,6 +48,10 @@ export default class SettingsDeployKeys {
       },
 
       computed: {
+        timeagoDate() {
+          return gl.utils.getTimeago().format(this.deployKey.createdAt, 'gl_en');
+        },
+
         disableURL() {
           return self.disableEndpoint.replace(':id', this.deployKey.id);
         },
-- 
GitLab


From d14397ff956e6f56bedf7737f51f8da154a9da64 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 28 Apr 2017 10:49:28 +0100
Subject: [PATCH 094/363] Fixed context translation Fixes CSS widths when
 translating

---
 .../components/limit_warning_component.js            |  2 +-
 app/assets/javascripts/locale/index.js               |  5 ++++-
 app/assets/stylesheets/pages/cycle_analytics.scss    | 12 ++++--------
 3 files changed, 9 insertions(+), 10 deletions(-)

diff --git a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
index 6e07560040c95..c43a45067553b 100644
--- a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
@@ -6,7 +6,7 @@ export default {
     },
   },
   template: `
-    <span v-if="count === 50 || true" class="events-info pull-right">
+    <span v-if="count === 50" class="events-info pull-right">
       <i class="fa fa-warning has-tooltip"
           aria-hidden="true"
           :title="__('Limited to showing 50 events at most')"
diff --git a/app/assets/javascripts/locale/index.js b/app/assets/javascripts/locale/index.js
index 3907b0e2aba34..ca3ed69fbb3ba 100644
--- a/app/assets/javascripts/locale/index.js
+++ b/app/assets/javascripts/locale/index.js
@@ -13,7 +13,10 @@ const lang = document.querySelector('html').getAttribute('lang') || 'en';
 const locale = new Jed(locales[lang]);
 const gettext = locale.gettext.bind(locale);
 const ngettext = locale.ngettext.bind(locale);
-const pgettext = locale.pgettext.bind(locale);
+const pgettext = (context, key) => {
+  const joinedKey = [context, key].join('|');
+  return gettext(joinedKey).split('|').pop();
+};
 
 export { lang };
 export { gettext as __ };
diff --git a/app/assets/stylesheets/pages/cycle_analytics.scss b/app/assets/stylesheets/pages/cycle_analytics.scss
index ad3dbc7ac4832..182909627bb48 100644
--- a/app/assets/stylesheets/pages/cycle_analytics.scss
+++ b/app/assets/stylesheets/pages/cycle_analytics.scss
@@ -213,7 +213,7 @@
     }
 
     .stage-nav-item {
-      display: block;
+      display: flex;
       line-height: 65px;
       border-top: 1px solid transparent;
       border-bottom: 1px solid transparent;
@@ -247,14 +247,10 @@
       }
 
       .stage-nav-item-cell {
-        float: left;
-
-        &.stage-name {
-          width: 65%;
-        }
-
         &.stage-median {
-          width: 35%;
+          margin-left: auto;
+          margin-right: $gl-padding;
+          min-width: calc(35% - #{$gl-padding});
         }
       }
 
-- 
GitLab


From 79d50538c4c633dd35b6182ba4d08d4eba7576fc Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 28 Apr 2017 11:27:52 +0100
Subject: [PATCH 095/363] Fixed avatar not displaying in issue boards

This happens when Gravatar is disabled in the admin settings, the avatar is returned as null & then frontend didn't do anything about it.

Closes #31428
---
 app/assets/javascripts/boards/models/user.js          | 3 ++-
 app/assets/javascripts/boards/utils/default_avatar.js | 1 +
 app/helpers/boards_helper.rb                          | 1 +
 changelogs/unreleased/issue-boards-no-avatar.yml      | 4 ++++
 4 files changed, 8 insertions(+), 1 deletion(-)
 create mode 100644 app/assets/javascripts/boards/utils/default_avatar.js
 create mode 100644 changelogs/unreleased/issue-boards-no-avatar.yml

diff --git a/app/assets/javascripts/boards/models/user.js b/app/assets/javascripts/boards/models/user.js
index 8e9de4d4cbb69..875c21a37d362 100644
--- a/app/assets/javascripts/boards/models/user.js
+++ b/app/assets/javascripts/boards/models/user.js
@@ -1,11 +1,12 @@
 /* eslint-disable no-unused-vars */
+import defaultAvatar from '../utils/default_avatar';
 
 class ListUser {
   constructor(user) {
     this.id = user.id;
     this.name = user.name;
     this.username = user.username;
-    this.avatar = user.avatar_url;
+    this.avatar = user.avatar_url || defaultAvatar();
   }
 }
 
diff --git a/app/assets/javascripts/boards/utils/default_avatar.js b/app/assets/javascripts/boards/utils/default_avatar.js
new file mode 100644
index 0000000000000..062ffec6dce0d
--- /dev/null
+++ b/app/assets/javascripts/boards/utils/default_avatar.js
@@ -0,0 +1 @@
+export default () => document.getElementById('board-app').dataset.defaultAvatar;
diff --git a/app/helpers/boards_helper.rb b/app/helpers/boards_helper.rb
index f43827da446d0..e2df52e383323 100644
--- a/app/helpers/boards_helper.rb
+++ b/app/helpers/boards_helper.rb
@@ -9,6 +9,7 @@ def board_data
       issue_link_base: namespace_project_issues_path(@project.namespace, @project),
       root_path: root_path,
       bulk_update_path: bulk_update_namespace_project_issues_path(@project.namespace, @project),
+      default_avatar: image_path(default_avatar)
     }
   end
 end
diff --git a/changelogs/unreleased/issue-boards-no-avatar.yml b/changelogs/unreleased/issue-boards-no-avatar.yml
new file mode 100644
index 0000000000000..a2dd53b3f2fe8
--- /dev/null
+++ b/changelogs/unreleased/issue-boards-no-avatar.yml
@@ -0,0 +1,4 @@
+---
+title: Fixed avatar not display on issue boards when Gravatar is disabled
+merge_request:
+author:
-- 
GitLab


From 817aa1b1160bcc1f3be828ad3c0c66fe403e2557 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 11:32:56 +0100
Subject: [PATCH 096/363] added balsamiq blobviewer class

---
 app/models/blob_viewer/balsamiq.rb | 12 ++++++++++++
 1 file changed, 12 insertions(+)
 create mode 100644 app/models/blob_viewer/balsamiq.rb

diff --git a/app/models/blob_viewer/balsamiq.rb b/app/models/blob_viewer/balsamiq.rb
new file mode 100644
index 0000000000000..8b0c0c049a124
--- /dev/null
+++ b/app/models/blob_viewer/balsamiq.rb
@@ -0,0 +1,12 @@
+module BalsamiqViewer
+  class Balsamiq < Base
+    include Rich
+    include ClientSide
+
+    self.partial_name = 'balsamiq'
+    self.extensions = %w(bmpr)
+    self.binary = true
+    self.switcher_icon = 'file-image-o'
+    self.switcher_title = 'preview'
+  end
+end
-- 
GitLab


From a67a648b0055334dbcfe0b20e01cd7b1401d2a0a Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 11:42:30 +0100
Subject: [PATCH 097/363] Moved partial

---
 app/models/blob_viewer/balsamiq.rb                              | 2 +-
 .../blob/{_bmpr.html.haml => viewers/_balsamiq.html.haml}       | 0
 2 files changed, 1 insertion(+), 1 deletion(-)
 rename app/views/projects/blob/{_bmpr.html.haml => viewers/_balsamiq.html.haml} (100%)

diff --git a/app/models/blob_viewer/balsamiq.rb b/app/models/blob_viewer/balsamiq.rb
index 8b0c0c049a124..f982521db9973 100644
--- a/app/models/blob_viewer/balsamiq.rb
+++ b/app/models/blob_viewer/balsamiq.rb
@@ -1,4 +1,4 @@
-module BalsamiqViewer
+module BlobViewer
   class Balsamiq < Base
     include Rich
     include ClientSide
diff --git a/app/views/projects/blob/_bmpr.html.haml b/app/views/projects/blob/viewers/_balsamiq.html.haml
similarity index 100%
rename from app/views/projects/blob/_bmpr.html.haml
rename to app/views/projects/blob/viewers/_balsamiq.html.haml
-- 
GitLab


From fc954749eace61d448fe8cc04c7fc0126bdbcdee Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 12:15:07 +0100
Subject: [PATCH 098/363] Corrected styling, less hacky

---
 app/assets/stylesheets/framework/files.scss | 29 +++++----------------
 1 file changed, 6 insertions(+), 23 deletions(-)

diff --git a/app/assets/stylesheets/framework/files.scss b/app/assets/stylesheets/framework/files.scss
index 3737ae3e653f6..a95ce4cc963e9 100644
--- a/app/assets/stylesheets/framework/files.scss
+++ b/app/assets/stylesheets/framework/files.scss
@@ -162,31 +162,14 @@
     }
 
     .list-inline.previews {
-      display: inline-block;
+      display: flex;
+      flex-wrap: wrap;
+      justify-content: center;
+      align-content: flex-start;
+      align-items: baseline;
 
       .preview {
-        display: inline-block;
-        height: 280px;
-        min-width: 260px;
-        margin: 0;
-        padding: 16px;
-        @include make-xs-column(3);
-      }
-
-      .panel {
-        margin: 0;
-        height: 100%;
-
-        .panel-body {
-          padding: $gl-padding 0 0;
-          text-align: center;
-        }
-      }
-
-      .img-thumbnail {
-        max-height: 195px;
-        max-width: 195px;
-        padding: 0;
+        padding: $gl-padding;
       }
     }
   }
-- 
GitLab


From 284d4f76fee9f593cb67f3f2978ad4f49ef03c13 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 13:23:34 +0100
Subject: [PATCH 099/363] Attempted adding separate clientside_sentry settings

---
 .../admin/application_settings_controller.rb      |  2 ++
 app/helpers/sentry_helper.rb                      | 12 ++++--------
 app/models/application_setting.rb                 |  4 ++++
 .../admin/application_settings/_form.html.haml    | 15 +++++++++++++++
 app/views/layouts/_head.html.haml                 |  2 +-
 config/application.rb                             |  2 ++
 db/schema.rb                                      | 13 ++++++-------
 lib/api/settings.rb                               |  5 +++++
 lib/api/v3/settings.rb                            |  6 +++++-
 lib/gitlab/gon_helper.rb                          |  2 +-
 10 files changed, 45 insertions(+), 18 deletions(-)

diff --git a/app/controllers/admin/application_settings_controller.rb b/app/controllers/admin/application_settings_controller.rb
index 643993d035e43..152d7baad4967 100644
--- a/app/controllers/admin/application_settings_controller.rb
+++ b/app/controllers/admin/application_settings_controller.rb
@@ -133,6 +133,8 @@ def application_setting_params_ce
       :signup_enabled,
       :sentry_dsn,
       :sentry_enabled,
+      :clientside_sentry_dsn,
+      :clientside_sentry_enabled,
       :send_user_confirmation_email,
       :shared_runners_enabled,
       :shared_runners_text,
diff --git a/app/helpers/sentry_helper.rb b/app/helpers/sentry_helper.rb
index 4b07f71bcea02..8e80f83ae6147 100644
--- a/app/helpers/sentry_helper.rb
+++ b/app/helpers/sentry_helper.rb
@@ -7,13 +7,9 @@ def sentry_context
     Gitlab::Sentry.context(current_user)
   end
 
-  def sentry_dsn_public
-    sentry_dsn = ApplicationSetting.current.sentry_dsn
-
-    return unless sentry_dsn
-
-    uri = URI.parse(sentry_dsn)
-    uri.password = nil
-    uri.to_s
+  def clientside_sentry_enabled?
+    current_application_settings.clientside_sentry_enabled
   end
+
+  delegate :clientside_sentry_dsn, to: :current_application_settings
 end
diff --git a/app/models/application_setting.rb b/app/models/application_setting.rb
index dd1a692296883..4a1e7d6e59eef 100644
--- a/app/models/application_setting.rb
+++ b/app/models/application_setting.rb
@@ -60,6 +60,10 @@ class ApplicationSetting < ActiveRecord::Base
             presence: true,
             if: :sentry_enabled
 
+  validates :clientside_sentry_dsn,
+            presence: true,
+            if: :clientside_sentry_enabled
+
   validates :akismet_api_key,
             presence: true,
             if: :akismet_enabled
diff --git a/app/views/admin/application_settings/_form.html.haml b/app/views/admin/application_settings/_form.html.haml
index 0dc1103eecef5..5cc99e0833fcb 100644
--- a/app/views/admin/application_settings/_form.html.haml
+++ b/app/views/admin/application_settings/_form.html.haml
@@ -411,6 +411,21 @@
       .col-sm-10
         = f.text_field :sentry_dsn, class: 'form-control'
 
+    .form-group
+      .col-sm-offset-2.col-sm-10
+        .checkbox
+          = f.label :clientside_sentry_enabled do
+            = f.check_box :clientside_sentry_enabled
+            Enable Clientside Sentry
+          .help-block
+            Sentry can also be used for reporting and logging clientside exceptions.
+            %a{ href: 'https://sentry.io/for/javascript/', target: '_blank', rel: 'noopener noreferrer' } https://sentry.io/for/javascript/
+
+    .form-group
+      = f.label :sentry_dsn, 'Sentry DSN', class: 'control-label col-sm-2'
+      .col-sm-10
+        = f.text_field :sentry_dsn, class: 'form-control'
+
   %fieldset
     %legend Repository Storage
     .form-group
diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index 8aef5cbdc044f..1579afa646174 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -33,7 +33,7 @@
   = webpack_bundle_tag "runtime"
   = webpack_bundle_tag "common"
   = webpack_bundle_tag "main"
-  = webpack_bundle_tag "raven" if sentry_enabled?
+  = webpack_bundle_tag "raven" if clientside_sentry_enabled?
 
   - if content_for?(:page_specific_javascripts)
     = yield :page_specific_javascripts
diff --git a/config/application.rb b/config/application.rb
index f2ecc4ce77c7e..0ab8a24a5dff8 100644
--- a/config/application.rb
+++ b/config/application.rb
@@ -55,6 +55,7 @@ class Application < Rails::Application
     # - Webhook URLs (:hook)
     # - GitLab-shell secret token (:secret_token)
     # - Sentry DSN (:sentry_dsn)
+    # - Clientside Sentry DSN (:clientside_sentry_dsn)
     # - Deploy keys (:key)
     config.filter_parameters += %i(
       authentication_token
@@ -71,6 +72,7 @@ class Application < Rails::Application
       runners_token
       secret_token
       sentry_dsn
+      clientside_sentry_dsn
       variables
     )
 
diff --git a/db/schema.rb b/db/schema.rb
index ff00951d5f6bb..592f1c253eaa2 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170424142900) do
+ActiveRecord::Schema.define(version: 20170426175636) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -63,7 +63,6 @@
     t.boolean "shared_runners_enabled", default: true, null: false
     t.integer "max_artifacts_size", default: 100, null: false
     t.string "runners_registration_token"
-    t.integer "max_pages_size", default: 100, null: false
     t.boolean "require_two_factor_authentication", default: false
     t.integer "two_factor_grace_period", default: 48
     t.boolean "metrics_enabled", default: false
@@ -112,15 +111,16 @@
     t.boolean "html_emails_enabled", default: true
     t.string "plantuml_url"
     t.boolean "plantuml_enabled"
+    t.integer "max_pages_size", default: 100, null: false
     t.integer "terminal_max_session_time", default: 0, null: false
     t.string "default_artifacts_expire_in", default: "0", null: false
     t.integer "unique_ips_limit_per_user"
     t.integer "unique_ips_limit_time_window"
     t.boolean "unique_ips_limit_enabled", default: false, null: false
     t.decimal "polling_interval_multiplier", default: 1.0, null: false
-    t.integer "cached_markdown_version"
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
+    t.integer "cached_markdown_version"
   end
 
   create_table "audit_events", force: :cascade do |t|
@@ -731,8 +731,8 @@
     t.integer "visibility_level", default: 20, null: false
     t.boolean "request_access_enabled", default: false, null: false
     t.datetime "deleted_at"
-    t.text "description_html"
     t.boolean "lfs_enabled"
+    t.text "description_html"
     t.integer "parent_id"
     t.boolean "require_two_factor_authentication", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
@@ -965,9 +965,9 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.integer "cached_markdown_version"
   end
 
@@ -1070,7 +1070,6 @@
     t.string "line_code"
     t.string "note_type"
     t.text "position"
-    t.string "in_reply_to_discussion_id"
   end
 
   add_index "sent_notifications", ["reply_key"], name: "index_sent_notifications_on_reply_key", unique: true, using: :btree
@@ -1320,10 +1319,10 @@
     t.string "organization"
     t.boolean "authorized_projects_populated"
     t.boolean "ghost"
-    t.date "last_activity_on"
     t.boolean "notified_of_own_activity"
     t.boolean "require_two_factor_authentication_from_group", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
+    t.date "last_activity_on"
   end
 
   add_index "users", ["admin"], name: "index_users_on_admin", using: :btree
diff --git a/lib/api/settings.rb b/lib/api/settings.rb
index d01c7f2703b96..82f513c984ec1 100644
--- a/lib/api/settings.rb
+++ b/lib/api/settings.rb
@@ -58,6 +58,7 @@ def current_settings
         :restricted_visibility_levels,
         :send_user_confirmation_email,
         :sentry_enabled,
+        :clientside_sentry_enabled,
         :session_expire_delay,
         :shared_runners_enabled,
         :sidekiq_throttling_enabled,
@@ -138,6 +139,10 @@ def current_settings
       given sentry_enabled: ->(val) { val } do
         requires :sentry_dsn, type: String, desc: 'Sentry Data Source Name'
       end
+      optional :clientside_sentry_enabled, type: Boolean, desc: 'Sentry can also be used for reporting and logging clientside exceptions. https://sentry.io/for/javascript/'
+      given clientside_sentry_enabled: ->(val) { val } do
+        requires :clientside_sentry_dsn, type: String, desc: 'Clientside Sentry Data Source Name'
+      end
       optional :repository_storage, type: String, desc: 'Storage paths for new projects'
       optional :repository_checks_enabled, type: Boolean, desc: "GitLab will periodically run 'git fsck' in all project and wiki repositories to look for silent disk corruption issues."
       optional :koding_enabled, type: Boolean, desc: 'Enable Koding'
diff --git a/lib/api/v3/settings.rb b/lib/api/v3/settings.rb
index 748d6b97d4f94..67a85e13d8c48 100644
--- a/lib/api/v3/settings.rb
+++ b/lib/api/v3/settings.rb
@@ -89,6 +89,10 @@ def current_settings
         given sentry_enabled: ->(val) { val } do
           requires :sentry_dsn, type: String, desc: 'Sentry Data Source Name'
         end
+        optional :clientside_sentry_enabled, type: Boolean, desc: 'Sentry can also be used for reporting and logging clientside exceptions. https://sentry.io/for/javascript/'
+        given clientside_sentry_enabled: ->(val) { val } do
+          requires :clientside_sentry_dsn, type: String, desc: 'Clientside Sentry Data Source Name'
+        end
         optional :repository_storage, type: String, desc: 'Storage paths for new projects'
         optional :repository_checks_enabled, type: Boolean, desc: "GitLab will periodically run 'git fsck' in all project and wiki repositories to look for silent disk corruption issues."
         optional :koding_enabled, type: Boolean, desc: 'Enable Koding'
@@ -120,7 +124,7 @@ def current_settings
                         :home_page_url, :after_sign_out_path, :sign_in_text, :help_page_text,
                         :shared_runners_enabled, :max_artifacts_size, :max_pages_size, :container_registry_token_expire_delay,
                         :metrics_enabled, :sidekiq_throttling_enabled, :recaptcha_enabled,
-                        :akismet_enabled, :admin_notification_email, :sentry_enabled,
+                        :akismet_enabled, :admin_notification_email, :sentry_enabled, :clientside_sentry_enabled,
                         :repository_storage, :repository_checks_enabled, :koding_enabled, :plantuml_enabled,
                         :version_check_enabled, :email_author_in_body, :html_emails_enabled,
                         :housekeeping_enabled, :terminal_max_session_time
diff --git a/lib/gitlab/gon_helper.rb b/lib/gitlab/gon_helper.rb
index 4de504e9bf984..a8a2715e8480c 100644
--- a/lib/gitlab/gon_helper.rb
+++ b/lib/gitlab/gon_helper.rb
@@ -12,7 +12,7 @@ def add_gon_variables
       gon.user_color_scheme      = Gitlab::ColorSchemes.for_user(current_user).css_class
       gon.katex_css_url          = ActionController::Base.helpers.asset_path('katex.css')
       gon.katex_js_url           = ActionController::Base.helpers.asset_path('katex.js')
-      gon.sentry_dsn             = sentry_dsn_public if sentry_enabled?
+      gon.sentry_dsn             = Gitlab.config.clientside_sentry_dsn if clientside_sentry_enabled?
       gon.gitlab_url             = Gitlab.config.gitlab.url
       gon.is_production          = Rails.env.production?
 
-- 
GitLab


From cb4b2e31a75e0f31e49f739c9008451d53908a0e Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 13:42:26 +0100
Subject: [PATCH 100/363] Added migration

---
 ...3910_add_clientside_sentry_to_application_settings.rb | 9 +++++++++
 db/schema.rb                                             | 5 ++++-
 2 files changed, 13 insertions(+), 1 deletion(-)
 create mode 100644 db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb

diff --git a/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb b/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
new file mode 100644
index 0000000000000..2b085b889eb87
--- /dev/null
+++ b/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
@@ -0,0 +1,9 @@
+# rubocop:disable all
+class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
+  def change
+    change_table :application_settings do |t|
+      t.boolean :clientside_sentry_enabled, default: false
+      t.string :clientside_sentry_dsn
+    end
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 592f1c253eaa2..a9940be22ffe8 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170426175636) do
+ActiveRecord::Schema.define(version: 20170428123910) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -121,6 +121,8 @@
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
     t.integer "cached_markdown_version"
+    t.boolean "clientside_sentry_enabled", default: false
+    t.string "clientside_sentry_dsn"
   end
 
   create_table "audit_events", force: :cascade do |t|
@@ -296,6 +298,7 @@
     t.boolean "locked", default: false, null: false
   end
 
+  add_index "ci_runners", ["contacted_at"], name: "index_ci_runners_on_contacted_at", using: :btree
   add_index "ci_runners", ["is_shared"], name: "index_ci_runners_on_is_shared", using: :btree
   add_index "ci_runners", ["locked"], name: "index_ci_runners_on_locked", using: :btree
   add_index "ci_runners", ["token"], name: "index_ci_runners_on_token", using: :btree
-- 
GitLab


From 8e8be5d1710f246bbb43c6c0bae82dead1139fde Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 13:51:45 +0100
Subject: [PATCH 101/363] Fixed view to correct property

---
 app/helpers/sentry_helper.rb                  |  6 -----
 .../application_settings/_form.html.haml      |  4 ++--
 app/views/layouts/_head.html.haml             |  2 +-
 lib/gitlab/gon_helper.rb                      |  2 +-
 spec/helpers/sentry_helper_spec.rb            | 22 -------------------
 5 files changed, 4 insertions(+), 32 deletions(-)
 delete mode 100644 spec/helpers/sentry_helper_spec.rb

diff --git a/app/helpers/sentry_helper.rb b/app/helpers/sentry_helper.rb
index 8e80f83ae6147..3d255df66a0c3 100644
--- a/app/helpers/sentry_helper.rb
+++ b/app/helpers/sentry_helper.rb
@@ -6,10 +6,4 @@ def sentry_enabled?
   def sentry_context
     Gitlab::Sentry.context(current_user)
   end
-
-  def clientside_sentry_enabled?
-    current_application_settings.clientside_sentry_enabled
-  end
-
-  delegate :clientside_sentry_dsn, to: :current_application_settings
 end
diff --git a/app/views/admin/application_settings/_form.html.haml b/app/views/admin/application_settings/_form.html.haml
index 5cc99e0833fcb..6030c8b1dfa25 100644
--- a/app/views/admin/application_settings/_form.html.haml
+++ b/app/views/admin/application_settings/_form.html.haml
@@ -422,9 +422,9 @@
             %a{ href: 'https://sentry.io/for/javascript/', target: '_blank', rel: 'noopener noreferrer' } https://sentry.io/for/javascript/
 
     .form-group
-      = f.label :sentry_dsn, 'Sentry DSN', class: 'control-label col-sm-2'
+      = f.label :clientside_sentry_dsn, 'Clientside Sentry DSN', class: 'control-label col-sm-2'
       .col-sm-10
-        = f.text_field :sentry_dsn, class: 'form-control'
+        = f.text_field :clientside_sentry_dsn, class: 'form-control'
 
   %fieldset
     %legend Repository Storage
diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index 1579afa646174..b768ac5a36cfd 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -33,7 +33,7 @@
   = webpack_bundle_tag "runtime"
   = webpack_bundle_tag "common"
   = webpack_bundle_tag "main"
-  = webpack_bundle_tag "raven" if clientside_sentry_enabled?
+  = webpack_bundle_tag "raven" if Gitlab.config.clientside_sentry_enabled
 
   - if content_for?(:page_specific_javascripts)
     = yield :page_specific_javascripts
diff --git a/lib/gitlab/gon_helper.rb b/lib/gitlab/gon_helper.rb
index a8a2715e8480c..e7ae1d1669861 100644
--- a/lib/gitlab/gon_helper.rb
+++ b/lib/gitlab/gon_helper.rb
@@ -12,7 +12,7 @@ def add_gon_variables
       gon.user_color_scheme      = Gitlab::ColorSchemes.for_user(current_user).css_class
       gon.katex_css_url          = ActionController::Base.helpers.asset_path('katex.css')
       gon.katex_js_url           = ActionController::Base.helpers.asset_path('katex.js')
-      gon.sentry_dsn             = Gitlab.config.clientside_sentry_dsn if clientside_sentry_enabled?
+      gon.sentry_dsn             = Gitlab.config.clientside_sentry_dsn if Gitlab.config.clientside_sentry_enabled
       gon.gitlab_url             = Gitlab.config.gitlab.url
       gon.is_production          = Rails.env.production?
 
diff --git a/spec/helpers/sentry_helper_spec.rb b/spec/helpers/sentry_helper_spec.rb
deleted file mode 100644
index ff218235cd1d3..0000000000000
--- a/spec/helpers/sentry_helper_spec.rb
+++ /dev/null
@@ -1,22 +0,0 @@
-require 'spec_helper'
-
-describe SentryHelper do
-  describe '#sentry_dsn_public' do
-    it 'returns nil if no sentry_dsn is set' do
-      mock_sentry_dsn(nil)
-
-      expect(helper.sentry_dsn_public).to eq nil
-    end
-
-    it 'returns the uri string with no password if sentry_dsn is set' do
-      mock_sentry_dsn('https://test:dsn@host/path')
-
-      expect(helper.sentry_dsn_public).to eq 'https://test@host/path'
-    end
-  end
-
-  def mock_sentry_dsn(value)
-    allow_message_expectations_on_nil
-    allow(ApplicationSetting.current).to receive(:sentry_dsn).and_return(value)
-  end
-end
-- 
GitLab


From 0e8afe1343497165bf609a840306ce26c6136a7b Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 14:00:20 +0100
Subject: [PATCH 102/363] Remove unneeded helper include

---
 lib/gitlab/gon_helper.rb | 2 --
 1 file changed, 2 deletions(-)

diff --git a/lib/gitlab/gon_helper.rb b/lib/gitlab/gon_helper.rb
index e7ae1d1669861..34f74232db62e 100644
--- a/lib/gitlab/gon_helper.rb
+++ b/lib/gitlab/gon_helper.rb
@@ -1,5 +1,3 @@
-include SentryHelper
-
 module Gitlab
   module GonHelper
     def add_gon_variables
-- 
GitLab


From 97633d561681a5c0f55472c1372bedca5fe3bffe Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 16:01:22 +0100
Subject: [PATCH 103/363] Fixed specs

---
 app/assets/javascripts/raven/raven_config.js |  2 +-
 app/views/layouts/_head.html.haml            |  5 +++-
 config/application.rb                        |  1 +
 lib/gitlab/gon_helper.rb                     |  2 +-
 spec/features/raven_js_spec.rb               |  4 +--
 spec/javascripts/raven/raven_config_spec.js  | 28 +++++++++++---------
 6 files changed, 24 insertions(+), 18 deletions(-)

diff --git a/app/assets/javascripts/raven/raven_config.js b/app/assets/javascripts/raven/raven_config.js
index 6c20cd9aabdb7..43e55bc541512 100644
--- a/app/assets/javascripts/raven/raven_config.js
+++ b/app/assets/javascripts/raven/raven_config.js
@@ -61,7 +61,7 @@ const RavenConfig = {
       environment: this.options.isProduction ? 'production' : 'development',
       ignoreErrors: this.IGNORE_ERRORS,
       ignoreUrls: this.IGNORE_URLS,
-      shouldSendCallback: this.shouldSendSample,
+      shouldSendCallback: this.shouldSendSample.bind(this),
     }).install();
   },
 
diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index b768ac5a36cfd..6123440b6e58f 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -33,7 +33,10 @@
   = webpack_bundle_tag "runtime"
   = webpack_bundle_tag "common"
   = webpack_bundle_tag "main"
-  = webpack_bundle_tag "raven" if Gitlab.config.clientside_sentry_enabled
+  = webpack_bundle_tag "raven" if current_application_settings.clientside_sentry_enabled
+
+  = 'LUKETEST'
+  = current_application_settings.clientside_sentry_enabled
 
   - if content_for?(:page_specific_javascripts)
     = yield :page_specific_javascripts
diff --git a/config/application.rb b/config/application.rb
index 0ab8a24a5dff8..46652637c1f36 100644
--- a/config/application.rb
+++ b/config/application.rb
@@ -72,6 +72,7 @@ class Application < Rails::Application
       runners_token
       secret_token
       sentry_dsn
+      clientside_sentry_enabled
       clientside_sentry_dsn
       variables
     )
diff --git a/lib/gitlab/gon_helper.rb b/lib/gitlab/gon_helper.rb
index 34f74232db62e..848b3352c63b7 100644
--- a/lib/gitlab/gon_helper.rb
+++ b/lib/gitlab/gon_helper.rb
@@ -10,7 +10,7 @@ def add_gon_variables
       gon.user_color_scheme      = Gitlab::ColorSchemes.for_user(current_user).css_class
       gon.katex_css_url          = ActionController::Base.helpers.asset_path('katex.css')
       gon.katex_js_url           = ActionController::Base.helpers.asset_path('katex.js')
-      gon.sentry_dsn             = Gitlab.config.clientside_sentry_dsn if Gitlab.config.clientside_sentry_enabled
+      gon.sentry_dsn             = current_application_settings.clientside_sentry_dsn if current_application_settings.clientside_sentry_enabled
       gon.gitlab_url             = Gitlab.config.gitlab.url
       gon.is_production          = Rails.env.production?
 
diff --git a/spec/features/raven_js_spec.rb b/spec/features/raven_js_spec.rb
index 74df52d80a73e..0fe1c28bf8d3d 100644
--- a/spec/features/raven_js_spec.rb
+++ b/spec/features/raven_js_spec.rb
@@ -10,8 +10,7 @@
   end
 
   it 'should load raven if sentry is enabled' do
-    allow_any_instance_of(SentryHelper).to receive_messages(sentry_dsn_public: 'https://key@domain.com/id',
-                                                            sentry_enabled?: true)
+    stub_application_setting(clientside_sentry_dsn: 'https://key@domain.com/id', clientside_sentry_enabled: true)
 
     visit new_user_session_path
 
@@ -19,6 +18,7 @@
   end
 
   def has_requested_raven
+    p page.driver.network_traffic
     page.driver.network_traffic.one? {|request| request.url.end_with?(raven_path)}
   end
 end
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index 78251b4f61b6c..b4c35420f9094 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -77,6 +77,7 @@ describe('RavenConfig', () => {
   describe('configure', () => {
     let options;
     let raven;
+    let ravenConfig;
 
     beforeEach(() => {
       options = {
@@ -85,22 +86,25 @@ describe('RavenConfig', () => {
         isProduction: true,
       };
 
+      ravenConfig = jasmine.createSpyObj('ravenConfig', ['shouldSendSample']);
       raven = jasmine.createSpyObj('raven', ['install']);
 
       spyOn(Raven, 'config').and.returnValue(raven);
 
-      RavenConfig.configure.call({
-        options,
-      });
+      ravenConfig.options = options;
+      ravenConfig.IGNORE_ERRORS = 'ignore_errors';
+      ravenConfig.IGNORE_URLS = 'ignore_urls';
+
+      RavenConfig.configure.call(ravenConfig);
     });
 
     it('should call Raven.config', () => {
       expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
         whitelistUrls: options.whitelistUrls,
         environment: 'production',
-        ignoreErrors: Raven.IGNORE_ERRORS,
-        ignoreUrls: Raven.IGNORE_URLS,
-        shouldSendCallback: Raven.shouldSendSample,
+        ignoreErrors: ravenConfig.IGNORE_ERRORS,
+        ignoreUrls: ravenConfig.IGNORE_URLS,
+        shouldSendCallback: jasmine.any(Function),
       });
     });
 
@@ -109,18 +113,16 @@ describe('RavenConfig', () => {
     });
 
     it('should set .environment to development if isProduction is false', () => {
-      options.isProduction = false;
+      ravenConfig.options.isProduction = false;
 
-      RavenConfig.configure.call({
-        options,
-      });
+      RavenConfig.configure.call(ravenConfig);
 
       expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
         whitelistUrls: options.whitelistUrls,
         environment: 'development',
-        ignoreErrors: Raven.IGNORE_ERRORS,
-        ignoreUrls: Raven.IGNORE_URLS,
-        shouldSendCallback: Raven.shouldSendSample,
+        ignoreErrors: ravenConfig.IGNORE_ERRORS,
+        ignoreUrls: ravenConfig.IGNORE_URLS,
+        shouldSendCallback: jasmine.any(Function),
       });
     });
   });
-- 
GitLab


From 89eaeb11d2a29b28b04f6494545ee1368462c237 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 16:04:07 +0100
Subject: [PATCH 104/363] Revert schema.db to master

---
 db/schema.rb | 17 ++++++-----------
 1 file changed, 6 insertions(+), 11 deletions(-)

diff --git a/db/schema.rb b/db/schema.rb
index 81cd32e2d8302..b938657a18655 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,11 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-<<<<<<< HEAD
-ActiveRecord::Schema.define(version: 20170428123910) do
-=======
 ActiveRecord::Schema.define(version: 20170426181740) do
->>>>>>> origin/master
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -67,6 +63,7 @@
     t.boolean "shared_runners_enabled", default: true, null: false
     t.integer "max_artifacts_size", default: 100, null: false
     t.string "runners_registration_token"
+    t.integer "max_pages_size", default: 100, null: false
     t.boolean "require_two_factor_authentication", default: false
     t.integer "two_factor_grace_period", default: 48
     t.boolean "metrics_enabled", default: false
@@ -115,18 +112,15 @@
     t.boolean "html_emails_enabled", default: true
     t.string "plantuml_url"
     t.boolean "plantuml_enabled"
-    t.integer "max_pages_size", default: 100, null: false
     t.integer "terminal_max_session_time", default: 0, null: false
     t.string "default_artifacts_expire_in", default: "0", null: false
     t.integer "unique_ips_limit_per_user"
     t.integer "unique_ips_limit_time_window"
     t.boolean "unique_ips_limit_enabled", default: false, null: false
     t.decimal "polling_interval_multiplier", default: 1.0, null: false
+    t.integer "cached_markdown_version"
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
-    t.integer "cached_markdown_version"
-    t.boolean "clientside_sentry_enabled", default: false
-    t.string "clientside_sentry_dsn"
   end
 
   create_table "audit_events", force: :cascade do |t|
@@ -738,8 +732,8 @@
     t.integer "visibility_level", default: 20, null: false
     t.boolean "request_access_enabled", default: false, null: false
     t.datetime "deleted_at"
-    t.boolean "lfs_enabled"
     t.text "description_html"
+    t.boolean "lfs_enabled"
     t.integer "parent_id"
     t.boolean "require_two_factor_authentication", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
@@ -972,9 +966,9 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.integer "cached_markdown_version"
   end
 
@@ -1077,6 +1071,7 @@
     t.string "line_code"
     t.string "note_type"
     t.text "position"
+    t.string "in_reply_to_discussion_id"
   end
 
   add_index "sent_notifications", ["reply_key"], name: "index_sent_notifications_on_reply_key", unique: true, using: :btree
@@ -1326,10 +1321,10 @@
     t.string "organization"
     t.boolean "authorized_projects_populated"
     t.boolean "ghost"
+    t.date "last_activity_on"
     t.boolean "notified_of_own_activity"
     t.boolean "require_two_factor_authentication_from_group", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
-    t.date "last_activity_on"
   end
 
   add_index "users", ["admin"], name: "index_users_on_admin", using: :btree
-- 
GitLab


From eacf8b74c5f57bd594485b9fe20e91c933d8b9bb Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 16:07:59 +0100
Subject: [PATCH 105/363] Applied schema update with patch

---
 db/schema.rb | 15 ++++++++-------
 1 file changed, 8 insertions(+), 7 deletions(-)

diff --git a/db/schema.rb b/db/schema.rb
index b938657a18655..a9940be22ffe8 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170426181740) do
+ActiveRecord::Schema.define(version: 20170428123910) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -63,7 +63,6 @@
     t.boolean "shared_runners_enabled", default: true, null: false
     t.integer "max_artifacts_size", default: 100, null: false
     t.string "runners_registration_token"
-    t.integer "max_pages_size", default: 100, null: false
     t.boolean "require_two_factor_authentication", default: false
     t.integer "two_factor_grace_period", default: 48
     t.boolean "metrics_enabled", default: false
@@ -112,15 +111,18 @@
     t.boolean "html_emails_enabled", default: true
     t.string "plantuml_url"
     t.boolean "plantuml_enabled"
+    t.integer "max_pages_size", default: 100, null: false
     t.integer "terminal_max_session_time", default: 0, null: false
     t.string "default_artifacts_expire_in", default: "0", null: false
     t.integer "unique_ips_limit_per_user"
     t.integer "unique_ips_limit_time_window"
     t.boolean "unique_ips_limit_enabled", default: false, null: false
     t.decimal "polling_interval_multiplier", default: 1.0, null: false
-    t.integer "cached_markdown_version"
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
+    t.integer "cached_markdown_version"
+    t.boolean "clientside_sentry_enabled", default: false
+    t.string "clientside_sentry_dsn"
   end
 
   create_table "audit_events", force: :cascade do |t|
@@ -732,8 +734,8 @@
     t.integer "visibility_level", default: 20, null: false
     t.boolean "request_access_enabled", default: false, null: false
     t.datetime "deleted_at"
-    t.text "description_html"
     t.boolean "lfs_enabled"
+    t.text "description_html"
     t.integer "parent_id"
     t.boolean "require_two_factor_authentication", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
@@ -966,9 +968,9 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.integer "cached_markdown_version"
   end
 
@@ -1071,7 +1073,6 @@
     t.string "line_code"
     t.string "note_type"
     t.text "position"
-    t.string "in_reply_to_discussion_id"
   end
 
   add_index "sent_notifications", ["reply_key"], name: "index_sent_notifications_on_reply_key", unique: true, using: :btree
@@ -1321,10 +1322,10 @@
     t.string "organization"
     t.boolean "authorized_projects_populated"
     t.boolean "ghost"
-    t.date "last_activity_on"
     t.boolean "notified_of_own_activity"
     t.boolean "require_two_factor_authentication_from_group", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
+    t.date "last_activity_on"
   end
 
   add_index "users", ["admin"], name: "index_users_on_admin", using: :btree
-- 
GitLab


From f6ebb90a7bcc984a958f2e5cf01f8f1716c7f1de Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 16:16:41 +0100
Subject: [PATCH 106/363] Deleted spinner and spec

---
 app/assets/javascripts/spinner.js |  29 ---------
 spec/javascripts/spinner_spec.js  | 101 ------------------------------
 2 files changed, 130 deletions(-)
 delete mode 100644 app/assets/javascripts/spinner.js
 delete mode 100644 spec/javascripts/spinner_spec.js

diff --git a/app/assets/javascripts/spinner.js b/app/assets/javascripts/spinner.js
deleted file mode 100644
index 6b5ac89a57641..0000000000000
--- a/app/assets/javascripts/spinner.js
+++ /dev/null
@@ -1,29 +0,0 @@
-class Spinner {
-  constructor(renderable) {
-    this.renderable = renderable;
-
-    this.container = Spinner.createContainer();
-  }
-
-  start() {
-    this.renderable.innerHTML = '';
-    this.renderable.appendChild(this.container);
-  }
-
-  stop() {
-    this.container.remove();
-  }
-
-  static createContainer() {
-    const container = document.createElement('div');
-    container.classList.add('loading');
-
-    container.innerHTML = Spinner.TEMPLATE;
-
-    return container;
-  }
-}
-
-Spinner.TEMPLATE = '<i class="fa fa-spinner fa-spin"></i>';
-
-export default Spinner;
diff --git a/spec/javascripts/spinner_spec.js b/spec/javascripts/spinner_spec.js
deleted file mode 100644
index f550285e0f76d..0000000000000
--- a/spec/javascripts/spinner_spec.js
+++ /dev/null
@@ -1,101 +0,0 @@
-import Spinner from '~/spinner';
-import ClassSpecHelper from './helpers/class_spec_helper';
-
-describe('Spinner', () => {
-  let renderable;
-  let container;
-  let spinner;
-
-  describe('class constructor', () => {
-    beforeEach(() => {
-      renderable = {};
-      container = {};
-
-      spyOn(Spinner, 'createContainer').and.returnValue(container);
-
-      spinner = new Spinner(renderable);
-    });
-
-    it('should set .renderable', () => {
-      expect(spinner.renderable).toBe(renderable);
-    });
-
-    it('should call Spinner.createContainer', () => {
-      expect(Spinner.createContainer).toHaveBeenCalled();
-    });
-
-    it('should set .container', () => {
-      expect(spinner.container).toBe(container);
-    });
-  });
-
-  describe('start', () => {
-    beforeEach(() => {
-      renderable = jasmine.createSpyObj('renderable', ['appendChild']);
-      container = {};
-
-      spinner = {
-        renderable,
-        container,
-      };
-
-      Spinner.prototype.start.call(spinner);
-    });
-
-    it('should set .innerHTML to an empty string', () => {
-      expect(renderable.innerHTML).toEqual('');
-    });
-
-    it('should call .appendChild', () => {
-      expect(renderable.appendChild).toHaveBeenCalledWith(container);
-    });
-  });
-
-  describe('stop', () => {
-    beforeEach(() => {
-      container = jasmine.createSpyObj('container', ['remove']);
-
-      spinner = {
-        container,
-      };
-
-      Spinner.prototype.stop.call(spinner);
-    });
-
-    it('should call .remove', () => {
-      expect(container.remove).toHaveBeenCalled();
-    });
-  });
-
-  describe('createContainer', () => {
-    let createContainer;
-
-    beforeEach(() => {
-      container = {
-        classList: jasmine.createSpyObj('classList', ['add']),
-      };
-
-      spyOn(document, 'createElement').and.returnValue(container);
-
-      createContainer = Spinner.createContainer();
-    });
-
-    ClassSpecHelper.itShouldBeAStaticMethod(Spinner, 'createContainer');
-
-    it('should call document.createElement', () => {
-      expect(document.createElement).toHaveBeenCalledWith('div');
-    });
-
-    it('should call classList.add', () => {
-      expect(container.classList.add).toHaveBeenCalledWith('loading');
-    });
-
-    it('should return the container element', () => {
-      expect(createContainer).toBe(container);
-    });
-
-    it('should set the container .innerHTML to Spinner.TEMPLATE', () => {
-      expect(container.innerHTML).toBe(Spinner.TEMPLATE);
-    });
-  });
-});
-- 
GitLab


From f13a6882d06d2a17c0f67b40cbf8f30edd289610 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 16:18:08 +0100
Subject: [PATCH 107/363] Removed puts

---
 spec/features/raven_js_spec.rb | 1 -
 1 file changed, 1 deletion(-)

diff --git a/spec/features/raven_js_spec.rb b/spec/features/raven_js_spec.rb
index 0fe1c28bf8d3d..e8fa49c18cbd5 100644
--- a/spec/features/raven_js_spec.rb
+++ b/spec/features/raven_js_spec.rb
@@ -18,7 +18,6 @@
   end
 
   def has_requested_raven
-    p page.driver.network_traffic
     page.driver.network_traffic.one? {|request| request.url.end_with?(raven_path)}
   end
 end
-- 
GitLab


From 1e2246e73ed45de4161c47a842dd68616373898a Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 16:21:50 +0100
Subject: [PATCH 108/363] Removed Spinner usage

---
 app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js | 2 --
 1 file changed, 2 deletions(-)

diff --git a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
index 5bcd7d5eccc07..1778f9cba5171 100644
--- a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
+++ b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
@@ -2,13 +2,11 @@
 
 import sqljs from 'sql.js';
 import { template as _template } from 'underscore';
-import Spinner from '../../spinner';
 
 class BalsamiqViewer {
   constructor(viewer) {
     this.viewer = viewer;
     this.endpoint = this.viewer.dataset.endpoint;
-    this.spinner = new Spinner(this.viewer);
   }
 
   loadFile() {
-- 
GitLab


From e11a702afceee7f5edeb8c93a4192fa05209b091 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 28 Apr 2017 17:06:19 +0100
Subject: [PATCH 109/363] Re-wrote to match our docs - still not 100% sure but
 closer than it was

---
 .../deploy_keys/components/action_btn.vue     |  53 +++++++++
 .../deploy_keys/components/app.vue            | 105 ++++++++++++++++++
 .../deploy_keys/components/key.vue            |  85 ++++++++++++++
 .../deploy_keys/components/keys.vue           |  52 +++++++++
 .../javascripts/deploy_keys/eventhub.js       |   3 +
 app/assets/javascripts/deploy_keys/index.js   |  21 ++++
 .../javascripts/deploy_keys/service/index.js  |  34 ++++++
 .../javascripts/deploy_keys/store/index.js    |  13 +++
 app/assets/javascripts/dispatcher.js          |   3 -
 .../settings/settings_repository.js           |  96 ----------------
 .../projects/deploy_keys_controller.rb        |  11 +-
 app/serializers/project_entity.rb             |   4 +-
 .../projects/deploy_keys/_index.html.haml     |  14 +--
 .../settings/repository/show.html.haml        |   4 +
 config/webpack.config.js                      |   1 +
 15 files changed, 384 insertions(+), 115 deletions(-)
 create mode 100644 app/assets/javascripts/deploy_keys/components/action_btn.vue
 create mode 100644 app/assets/javascripts/deploy_keys/components/app.vue
 create mode 100644 app/assets/javascripts/deploy_keys/components/key.vue
 create mode 100644 app/assets/javascripts/deploy_keys/components/keys.vue
 create mode 100644 app/assets/javascripts/deploy_keys/eventhub.js
 create mode 100644 app/assets/javascripts/deploy_keys/index.js
 create mode 100644 app/assets/javascripts/deploy_keys/service/index.js
 create mode 100644 app/assets/javascripts/deploy_keys/store/index.js
 delete mode 100644 app/assets/javascripts/settings/settings_repository.js

diff --git a/app/assets/javascripts/deploy_keys/components/action_btn.vue b/app/assets/javascripts/deploy_keys/components/action_btn.vue
new file mode 100644
index 0000000000000..7da2915a45e38
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/components/action_btn.vue
@@ -0,0 +1,53 @@
+<script>
+  import eventHub from '../eventhub';
+
+  export default {
+    data() {
+      return {
+        isLoading: false,
+      };
+    },
+    props: {
+      deployKey: {
+        type: Object,
+        required: true,
+      },
+      type: {
+        type: String,
+        required: true,
+      },
+      btnCssClass: {
+        type: String,
+        required: false,
+        default: 'btn-default',
+      },
+    },
+    methods: {
+      doAction() {
+        this.isLoading = true;
+
+        eventHub.$emit(`${this.type}.key`, this.deployKey);
+      },
+    },
+    computed: {
+      text() {
+        return `${this.type.charAt(0).toUpperCase()}${this.type.slice(1)}`;
+      },
+    },
+  };
+</script>
+
+<template>
+  <button
+    class="btn btn-sm prepend-left-10"
+    :class="[{ disabled: isLoading }, btnCssClass]"
+    :disabled="isLoading"
+    @click="doAction">
+    {{ text }}
+    <i
+      v-if="isLoading"
+      class="fa fa-spinner fa-spin"
+      aria-hidden="true">
+    </i>
+  </button>
+</template>
diff --git a/app/assets/javascripts/deploy_keys/components/app.vue b/app/assets/javascripts/deploy_keys/components/app.vue
new file mode 100644
index 0000000000000..ff54a0f241a2f
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/components/app.vue
@@ -0,0 +1,105 @@
+<script>
+  /* global Flash */
+  import eventHub from '../eventhub';
+  import DeployKeysService from '../service';
+  import DeployKeysStore from '../store';
+  import keysPanel from './keys.vue';
+
+  export default {
+    data() {
+      const store = new DeployKeysStore();
+
+      return {
+        isLoading: false,
+        store,
+      };
+    },
+    props: {
+      endpoint: {
+        type: String,
+        required: true,
+      },
+    },
+    computed: {
+      hasKeys() {
+        return Object.keys(this.keys).length;
+      },
+      keys() {
+        return this.store.keys;
+      },
+    },
+    components: {
+      keysPanel,
+    },
+    methods: {
+      fetchKeys() {
+        this.isLoading = true;
+        this.store.keys = {};
+
+        this.service.getKeys()
+          .then((data) => {
+            this.isLoading = false;
+            this.store.keys = data;
+          })
+          .catch(() => new Flash('Error getting deploy keys'));
+      },
+      enableKey(deployKey) {
+        this.service.enableKey(deployKey.id)
+          .then(() => this.fetchKeys())
+          .catch(() => new Flash('Error enabling deploy key'));
+      },
+      removeKey(deployKey) {
+        this.disableKey(deployKey);
+      },
+      disableKey(deployKey) {
+        // eslint-disable-next-line no-alert
+        if (confirm('You are going to remove this deploy key. Are you sure?')) {
+          this.service.disableKey(deployKey.id)
+            .then(() => this.fetchKeys())
+            .catch(() => new Flash('Error removing deploy key'));
+        }
+      },
+    },
+    created() {
+      this.service = new DeployKeysService(this.endpoint);
+
+      eventHub.$on('enable.key', this.enableKey);
+      eventHub.$on('remove.key', this.removeKey);
+      eventHub.$on('disable.key', this.disableKey);
+    },
+    mounted() {
+      this.fetchKeys();
+    },
+    beforeDestroy() {
+      eventHub.$off('enable.key', this.enableKey);
+      eventHub.$off('remove.key', this.removeKey);
+      eventHub.$off('disable.key', this.disableKey);
+    },
+  };
+</script>
+
+<template>
+  <div class="col-lg-9 col-lg-offset-3 append-bottom-default">
+    <div
+      class="text-center"
+      v-if="isLoading && !hasKeys">
+      <i
+        class="fa fa-spinner fa-spin fa-2x">
+      </i>
+    </div>
+    <div v-else-if="hasKeys">
+      <keys-panel
+        title="Enabled deploy keys for this project"
+        :keys="keys.enabled_keys"
+        :store="store" />
+      <keys-panel
+        title="Deploy keys from projects you have access to"
+        :keys="keys.available_project_keys"
+        :store="store" />
+      <keys-panel
+        title="Public deploy keys available to any project"
+        :keys="keys.public_keys"
+        :store="store" />
+    </div>
+  </div>
+</template>
diff --git a/app/assets/javascripts/deploy_keys/components/key.vue b/app/assets/javascripts/deploy_keys/components/key.vue
new file mode 100644
index 0000000000000..af842a3bb391c
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/components/key.vue
@@ -0,0 +1,85 @@
+<script>
+  import actionBtn from './action_btn.vue';
+
+  export default {
+    props: {
+      deployKey: {
+        type: Object,
+        required: true,
+      },
+      enabled: {
+        type: Boolean,
+        required: false,
+        default: true,
+      },
+      store: {
+        type: Object,
+        required: true,
+      },
+    },
+    components: {
+      actionBtn,
+    },
+    computed: {
+      timeagoDate() {
+        return gl.utils.getTimeago().format(this.deployKey.created_at);
+      },
+    },
+    methods: {
+      isEnabled(id) {
+        return this.store.findEnabledKey(id) !== undefined;
+      },
+    },
+  };
+</script>
+
+<template>
+  <div>
+    <div class="pull-left append-right-10 hidden-xs">
+      <i
+        aria-hidden="true"
+        class="fa fa-key key-icon">
+      </i>
+    </div>
+    <div class="deploy-key-content key-list-item-info">
+      <strong class="title">
+        {{ deployKey.title }}
+      </strong>
+      <div class="description">
+        {{ deployKey.fingerprint }}
+      </div>
+      <div
+        v-if="deployKey.can_push"
+        class="write-access-allowed">
+        Write access allowed
+      </div>
+    </div>
+    <div class="deploy-key-content prepend-left-default deploy-key-projects">
+      <a
+        class="label deploy-project-label"
+        :href="project.full_path"
+        v-for="project in deployKey.projects">
+        {{ project.full_name }}
+      </a>
+    </div>
+    <div class="deploy-key-content">
+      <span class="key-created-at">
+        created {{ timeagoDate }}
+      </span>
+      <action-btn
+        v-if="!isEnabled(deployKey.id)"
+        :deploy-key="deployKey"
+        type="enable"/>
+      <action-btn
+        v-else-if="deployKey.destroyed_when_orphaned && deployKey.almost_orphaned"
+        :deploy-key="deployKey"
+        btn-css-class="btn-warning"
+        type="remove" />
+      <action-btn
+        v-else
+        :deploy-key="deployKey"
+        btn-css-class="btn-warning"
+        type="disable" />
+    </div>
+  </div>
+</template>
diff --git a/app/assets/javascripts/deploy_keys/components/keys.vue b/app/assets/javascripts/deploy_keys/components/keys.vue
new file mode 100644
index 0000000000000..2470831eddfef
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/components/keys.vue
@@ -0,0 +1,52 @@
+<script>
+  import key from './key.vue';
+
+  export default {
+    props: {
+      title: {
+        type: String,
+        required: true,
+      },
+      keys: {
+        type: Array,
+        required: true,
+      },
+      showHelpBox: {
+        type: Boolean,
+        required: false,
+        default: true,
+      },
+      store: {
+        type: Object,
+        required: true,
+      },
+    },
+    components: {
+      key,
+    },
+  };
+</script>
+
+<template>
+  <div>
+    <h5>
+      {{ title }}
+      ({{ keys.length }})
+    </h5>
+    <ul class="well-list"
+      v-if="keys.length">
+      <li
+        v-for="deployKey in keys"
+        key="deployKey.id">
+        <key
+          :deploy-key="deployKey"
+          :store="store" />
+      </li>
+    </ul>
+    <div
+      class="settings-message text-center"
+      v-else-if="showHelpBox">
+      No deploy keys found. Create one with the form above.
+    </div>
+  </div>
+</template>
diff --git a/app/assets/javascripts/deploy_keys/eventhub.js b/app/assets/javascripts/deploy_keys/eventhub.js
new file mode 100644
index 0000000000000..0948c2e53524a
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/eventhub.js
@@ -0,0 +1,3 @@
+import Vue from 'vue';
+
+export default new Vue();
diff --git a/app/assets/javascripts/deploy_keys/index.js b/app/assets/javascripts/deploy_keys/index.js
new file mode 100644
index 0000000000000..a5f232f950a61
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/index.js
@@ -0,0 +1,21 @@
+import Vue from 'vue';
+import deployKeysApp from './components/app.vue';
+
+document.addEventListener('DOMContentLoaded', () => new Vue({
+  el: document.getElementById('js-deploy-keys'),
+  data() {
+    return {
+      endpoint: this.$options.el.dataset.endpoint,
+    };
+  },
+  components: {
+    deployKeysApp,
+  },
+  render(createElement) {
+    return createElement('deploy-keys-app', {
+      props: {
+        endpoint: this.endpoint,
+      },
+    });
+  },
+}));
diff --git a/app/assets/javascripts/deploy_keys/service/index.js b/app/assets/javascripts/deploy_keys/service/index.js
new file mode 100644
index 0000000000000..fe6dbaa949809
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/service/index.js
@@ -0,0 +1,34 @@
+import Vue from 'vue';
+import VueResource from 'vue-resource';
+
+Vue.use(VueResource);
+
+export default class DeployKeysService {
+  constructor(endpoint) {
+    this.endpoint = endpoint;
+
+    this.resource = Vue.resource(`${this.endpoint}{/id}`, {}, {
+      enable: {
+        method: 'PUT',
+        url: `${this.endpoint}{/id}/enable`,
+      },
+      disable: {
+        method: 'PUT',
+        url: `${this.endpoint}{/id}/disable`,
+      },
+    });
+  }
+
+  getKeys() {
+    return this.resource.get()
+      .then(response => response.json());
+  }
+
+  enableKey(id) {
+    return this.resource.enable({ id }, {});
+  }
+
+  disableKey(id) {
+    return this.resource.disable({ id }, {});
+  }
+}
diff --git a/app/assets/javascripts/deploy_keys/store/index.js b/app/assets/javascripts/deploy_keys/store/index.js
new file mode 100644
index 0000000000000..7177d9bed7fbc
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/store/index.js
@@ -0,0 +1,13 @@
+export default class DeployKeysStore {
+  constructor() {
+    this.keys = {};
+  }
+
+  findEnabledKey(id) {
+    return this.keys.enabled_keys.find(key => key.id === id);
+  }
+
+  removeKeyForType(deployKey, type) {
+    this.keys[type] = this.keys[type].filter(key => key.id !== deployKey.id);
+  }
+}
diff --git a/app/assets/javascripts/dispatcher.js b/app/assets/javascripts/dispatcher.js
index f7402792c59eb..d3d75c4bf4a15 100644
--- a/app/assets/javascripts/dispatcher.js
+++ b/app/assets/javascripts/dispatcher.js
@@ -48,7 +48,6 @@ import BlobForkSuggestion from './blob/blob_fork_suggestion';
 import UserCallout from './user_callout';
 import { ProtectedTagCreate, ProtectedTagEditList } from './protected_tags';
 import ShortcutsWiki from './shortcuts_wiki';
-import SettingsDeployKeys from './settings/settings_repository';
 import BlobViewer from './blob/viewer/index';
 
 const ShortcutsBlob = require('./shortcuts_blob');
@@ -346,8 +345,6 @@ const ShortcutsBlob = require('./shortcuts_blob');
           // Initialize Protected Tag Settings
           new ProtectedTagCreate();
           new ProtectedTagEditList();
-          // Initialize deploy key ajax call
-          new SettingsDeployKeys();
           break;
         case 'projects:ci_cd:show':
           new gl.ProjectVariables();
diff --git a/app/assets/javascripts/settings/settings_repository.js b/app/assets/javascripts/settings/settings_repository.js
deleted file mode 100644
index 9d6a60baa3ad4..0000000000000
--- a/app/assets/javascripts/settings/settings_repository.js
+++ /dev/null
@@ -1,96 +0,0 @@
-/* eslint-disable no-new */
-import Vue from 'vue';
-import VueResource from 'vue-resource';
-
-Vue.use(VueResource);
-
-export default class SettingsDeployKeys {
-  constructor() {
-    this.initVue();
-  }
-
-  deployKeyRowTemplate() {
-    return `
-      <li>
-        <div class="pull-left append-right-10 hidden-xs">
-          <i aria-hidden="true" class="fa fa-key key-icon"></i>
-        </div>
-        <div class="deploy-key-content key-list-item-info">
-          <strong class="title">
-            {{deployKey.title}}
-          </strong>
-          <div class="description">
-            {{deployKey.fingerprint}}
-          </div>
-        </div>
-        <div class="deploy-key-content prepend-left-default deploy-key-projects">
-          <a class="label deploy-project-label" :href="project.full_path" v-for="project in deployKey.projects">{{project.full_name}}</a>
-        </div>
-        <div class="deploy-key-content">
-          <span class="key-created-at">
-            created {{timeagoDate}}
-          </span>
-          <div class="visible-xs-block visible-sm-block"></div>
-          <a v-if="!enabled" class="btn btn-sm prepend-left-10" rel="nofollow" data-method="put" href="enableURL">Enable
-</a>
-          <a v-else-if="deployKey.destroyed_when_orphaned && deployKey.almost_orphaned" class="btn btn-warning btn-sm prepend-left-10" rel="nofollow" data-method="put" :href="disableURL">Remove</a>
-          <a v-else class="btn btn-warning btn-sm prepend-left-10" rel="nofollow" data-method="put" :href="disableURL">Disable</a>
-        </div>
-      </li>`
-  }
-
-  deployKeyRowComponent() {
-    const self = this;
-    return {
-      props: {
-        deployKey: Object,
-        enabled: Boolean
-      },
-
-      computed: {
-        timeagoDate() {
-          return gl.utils.getTimeago().format(this.deployKey.createdAt, 'gl_en');
-        },
-
-        disableURL() {
-          return self.disableEndpoint.replace(':id', this.deployKey.id);
-        },
-
-        enableURL() {
-         return self.enableEndpoint.replace(':id', this.deployKey.id); 
-        }
-      },
-
-      template: this.deployKeyRowTemplate()
-    }
-  }
-
-  initVue() {
-    const self = this;
-    const el = document.getElementById('js-deploy-keys');
-    const endpoint = el.dataset.endpoint;
-    this.jsonEndpoint = `${endpoint}.json`;
-    this.disableEndpoint = `${endpoint}/:id/disable`;
-    this.enableEndpoint = `${endpoint}/:id/enable`;
-    new Vue({
-      el: el,
-      components: {
-        deployKeyRow: self.deployKeyRowComponent()
-      },
-      data () {
-        return {
-          enabledKeys: [],
-          availableKeys: []
-        }
-      },
-      created () {
-        this.$http.get(self.jsonEndpoint)
-          .then((res) => {
-            const keys = JSON.parse(res.body);
-            this.enabledKeys = keys.enabled_keys;
-            this.availableKeys = keys.available_project_keys;
-          });
-      }
-    })
-  }
-} 
\ No newline at end of file
diff --git a/app/controllers/projects/deploy_keys_controller.rb b/app/controllers/projects/deploy_keys_controller.rb
index a47e15a192bff..f27089b8590d0 100644
--- a/app/controllers/projects/deploy_keys_controller.rb
+++ b/app/controllers/projects/deploy_keys_controller.rb
@@ -32,7 +32,10 @@ def create
   def enable
     Projects::EnableDeployKeyService.new(@project, current_user, params).execute
 
-    redirect_to_repository_settings(@project)
+    respond_to do |format|
+      format.html { redirect_to_repository_settings(@project) }
+      format.json { head :ok }
+    end
   end
 
   def disable
@@ -40,7 +43,11 @@ def disable
     return render_404 unless deploy_key_project
 
     deploy_key_project.destroy!
-    redirect_to_repository_settings(@project)
+
+    respond_to do |format|
+      format.html { redirect_to_repository_settings(@project) }
+      format.json { head :ok }
+    end
   end
 
   protected
diff --git a/app/serializers/project_entity.rb b/app/serializers/project_entity.rb
index 6f8061f753095..a471a7e6a8825 100644
--- a/app/serializers/project_entity.rb
+++ b/app/serializers/project_entity.rb
@@ -1,9 +1,11 @@
 class ProjectEntity < Grape::Entity
+  include RequestAwareEntity
+  
   expose :id
   expose :name
 
   expose :full_path do |project|
-    project.full_path
+    namespace_project_path(project.namespace, project)
   end
 
   expose :full_name do |project|
diff --git a/app/views/projects/deploy_keys/_index.html.haml b/app/views/projects/deploy_keys/_index.html.haml
index b35cd356aa507..74756b58439a0 100644
--- a/app/views/projects/deploy_keys/_index.html.haml
+++ b/app/views/projects/deploy_keys/_index.html.haml
@@ -10,16 +10,4 @@
     = render @deploy_keys.form_partial_path
   .col-lg-9.col-lg-offset-3
     %hr
-  #js-deploy-keys.col-lg-9.col-lg-offset-3.append-bottom-default{data:{endpoint: namespace_project_deploy_keys_path}}
-    %h5.prepend-top-0
-      Enabled deploy keys for this project ({{enabledKeys.length}})
-    %ul.well-list{"v-if" => "enabledKeys.length"}
-      %deploy-key-row{"v-for" => "deployKey in enabledKeys", ":deploy-key" => "deployKey", ":enabled" =>"true"}
-    .settings-message.text-center{"v-else" => true}
-      No deploy keys found. Create one with the form above.
-    %h5.prepend-top-0
-      Deploy keys from projects you have access to ({{availableKeys.length}})
-    %ul.well-list{"v-if" => "availableKeys.length"}
-      %deploy-key-row{"v-for" => "deployKey in availableKeys", ":deploy-key" => "deployKey", ":enabled" =>"false"}
-    .settings-message.text-center{"v-else" => true}
-      No deploy keys found. Create one with the form above.
+  #js-deploy-keys{ data: { endpoint: namespace_project_deploy_keys_path } }
diff --git a/app/views/projects/settings/repository/show.html.haml b/app/views/projects/settings/repository/show.html.haml
index 5402320cb66d0..4e59033c4a34e 100644
--- a/app/views/projects/settings/repository/show.html.haml
+++ b/app/views/projects/settings/repository/show.html.haml
@@ -1,6 +1,10 @@
 - page_title "Repository"
 = render "projects/settings/head"
 
+- content_for :page_specific_javascripts do
+  = page_specific_javascript_bundle_tag('common_vue')
+  = page_specific_javascript_bundle_tag('deploy_keys')
+
 = render @deploy_keys
 = render "projects/protected_branches/index"
 = render "projects/protected_tags/index"
diff --git a/config/webpack.config.js b/config/webpack.config.js
index cb0a57a3a410f..1721d275685fb 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -26,6 +26,7 @@ var config = {
     common_d3:            ['d3'],
     cycle_analytics:      './cycle_analytics/cycle_analytics_bundle.js',
     commit_pipelines:     './commit/pipelines/pipelines_bundle.js',
+    deploy_keys:          './deploy_keys/index.js',
     diff_notes:           './diff_notes/diff_notes_bundle.js',
     environments:         './environments/environments_bundle.js',
     environments_folder:  './environments/folder/environments_folder_bundle.js',
-- 
GitLab


From fde9732a27065467f4fab0ba0a6bd1cfed9b092b Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 28 Apr 2017 17:11:46 +0100
Subject: [PATCH 110/363] Fixed public keys always rendering even when empty

---
 app/assets/javascripts/deploy_keys/components/app.vue | 1 +
 1 file changed, 1 insertion(+)

diff --git a/app/assets/javascripts/deploy_keys/components/app.vue b/app/assets/javascripts/deploy_keys/components/app.vue
index ff54a0f241a2f..e0443e2e0df63 100644
--- a/app/assets/javascripts/deploy_keys/components/app.vue
+++ b/app/assets/javascripts/deploy_keys/components/app.vue
@@ -97,6 +97,7 @@
         :keys="keys.available_project_keys"
         :store="store" />
       <keys-panel
+        v-if="keys.public_keys.length"
         title="Public deploy keys available to any project"
         :keys="keys.public_keys"
         :store="store" />
-- 
GitLab


From 0b7824d433ec5029da17114389f425f816b7be74 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 28 Apr 2017 17:22:57 +0100
Subject: [PATCH 111/363] Updated some Vue specific JS

---
 .../javascripts/deploy_keys/components/action_btn.vue  |  3 ++-
 app/assets/javascripts/deploy_keys/components/app.vue  | 10 +++++-----
 app/assets/javascripts/deploy_keys/components/key.vue  |  4 ++--
 .../components/{keys.vue => keys_panel.vue}            |  0
 app/assets/javascripts/deploy_keys/store/index.js      |  4 ----
 changelogs/unreleased/29667-deploy-keys.yml            |  4 ----
 changelogs/unreleased/deploy-keys-load-async.yml       |  4 ++++
 7 files changed, 13 insertions(+), 16 deletions(-)
 rename app/assets/javascripts/deploy_keys/components/{keys.vue => keys_panel.vue} (100%)
 delete mode 100644 changelogs/unreleased/29667-deploy-keys.yml
 create mode 100644 changelogs/unreleased/deploy-keys-load-async.yml

diff --git a/app/assets/javascripts/deploy_keys/components/action_btn.vue b/app/assets/javascripts/deploy_keys/components/action_btn.vue
index 7da2915a45e38..3ff3a9d977e77 100644
--- a/app/assets/javascripts/deploy_keys/components/action_btn.vue
+++ b/app/assets/javascripts/deploy_keys/components/action_btn.vue
@@ -47,7 +47,8 @@
     <i
       v-if="isLoading"
       class="fa fa-spinner fa-spin"
-      aria-hidden="true">
+      aria-hidden="true"
+      aria-label="Loading">
     </i>
   </button>
 </template>
diff --git a/app/assets/javascripts/deploy_keys/components/app.vue b/app/assets/javascripts/deploy_keys/components/app.vue
index e0443e2e0df63..ee2f85bde5388 100644
--- a/app/assets/javascripts/deploy_keys/components/app.vue
+++ b/app/assets/javascripts/deploy_keys/components/app.vue
@@ -3,15 +3,13 @@
   import eventHub from '../eventhub';
   import DeployKeysService from '../service';
   import DeployKeysStore from '../store';
-  import keysPanel from './keys.vue';
+  import keysPanel from './keys_panel.vue';
 
   export default {
     data() {
-      const store = new DeployKeysStore();
-
       return {
         isLoading: false,
-        store,
+        store: new DeployKeysStore(),
       };
     },
     props: {
@@ -84,7 +82,9 @@
       class="text-center"
       v-if="isLoading && !hasKeys">
       <i
-        class="fa fa-spinner fa-spin fa-2x">
+        class="fa fa-spinner fa-spin fa-2x"
+        aria-hidden="true"
+        aria-label="Loading deploy keys">
       </i>
     </div>
     <div v-else-if="hasKeys">
diff --git a/app/assets/javascripts/deploy_keys/components/key.vue b/app/assets/javascripts/deploy_keys/components/key.vue
index af842a3bb391c..1b47cceba78a1 100644
--- a/app/assets/javascripts/deploy_keys/components/key.vue
+++ b/app/assets/javascripts/deploy_keys/components/key.vue
@@ -56,9 +56,9 @@
     </div>
     <div class="deploy-key-content prepend-left-default deploy-key-projects">
       <a
+        v-for="project in deployKey.projects"
         class="label deploy-project-label"
-        :href="project.full_path"
-        v-for="project in deployKey.projects">
+        :href="project.full_path">
         {{ project.full_name }}
       </a>
     </div>
diff --git a/app/assets/javascripts/deploy_keys/components/keys.vue b/app/assets/javascripts/deploy_keys/components/keys_panel.vue
similarity index 100%
rename from app/assets/javascripts/deploy_keys/components/keys.vue
rename to app/assets/javascripts/deploy_keys/components/keys_panel.vue
diff --git a/app/assets/javascripts/deploy_keys/store/index.js b/app/assets/javascripts/deploy_keys/store/index.js
index 7177d9bed7fbc..6210361af26e8 100644
--- a/app/assets/javascripts/deploy_keys/store/index.js
+++ b/app/assets/javascripts/deploy_keys/store/index.js
@@ -6,8 +6,4 @@ export default class DeployKeysStore {
   findEnabledKey(id) {
     return this.keys.enabled_keys.find(key => key.id === id);
   }
-
-  removeKeyForType(deployKey, type) {
-    this.keys[type] = this.keys[type].filter(key => key.id !== deployKey.id);
-  }
 }
diff --git a/changelogs/unreleased/29667-deploy-keys.yml b/changelogs/unreleased/29667-deploy-keys.yml
deleted file mode 100644
index 0f202ebf1ee56..0000000000000
--- a/changelogs/unreleased/29667-deploy-keys.yml
+++ /dev/null
@@ -1,4 +0,0 @@
----
-title: Project deploy keys json end point
-merge_request:
-author:
diff --git a/changelogs/unreleased/deploy-keys-load-async.yml b/changelogs/unreleased/deploy-keys-load-async.yml
new file mode 100644
index 0000000000000..e90910278e807
--- /dev/null
+++ b/changelogs/unreleased/deploy-keys-load-async.yml
@@ -0,0 +1,4 @@
+---
+title: Deploy keys load are loaded async
+merge_request:
+author:
-- 
GitLab


From 2ef7c6f070e52f68b10cba09d10a4db9ab8b2537 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 17:32:50 +0100
Subject: [PATCH 112/363] Fixed specs

---
 .../blob/balsamiq/balsamiq_viewer.js          |  58 ++++++----
 .../projects/blob/viewers/_balsamiq.html.haml |   2 +-
 .../blob/balsamiq/balsamiq_viewer_spec.js     | 105 ++++--------------
 3 files changed, 56 insertions(+), 109 deletions(-)

diff --git a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
index 1778f9cba5171..cdbfe36ca1cc3 100644
--- a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
+++ b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
@@ -3,6 +3,15 @@
 import sqljs from 'sql.js';
 import { template as _template } from 'underscore';
 
+const PREVIEW_TEMPLATE = _template(`
+  <div class="panel panel-default">
+    <div class="panel-heading"><%- name %></div>
+    <div class="panel-body">
+      <img class="img-thumbnail" src="data:image/png;base64,<%- image %>"/>
+    </div>
+  </div>
+`);
+
 class BalsamiqViewer {
   constructor(viewer) {
     this.viewer = viewer;
@@ -18,23 +27,23 @@ class BalsamiqViewer {
     xhr.onload = this.renderFile.bind(this);
     xhr.onerror = BalsamiqViewer.onError;
 
-    this.spinner.start();
-
     xhr.send();
   }
 
   renderFile(loadEvent) {
-    this.spinner.stop();
-
     const container = document.createElement('ul');
 
     this.initDatabase(loadEvent.target.response);
 
     const previews = this.getPreviews();
-    const renderedPreviews = previews.map(preview => this.renderPreview(preview));
+    previews.forEach((preview) => {
+      const renderedPreview = this.renderPreview(preview);
 
-    container.innerHTML = renderedPreviews.join('');
-    container.classList.add('list-inline', 'previews');
+      container.appendChild(renderedPreview);
+    });
+
+    container.classList.add('list-inline');
+    container.classList.add('previews');
 
     this.viewer.appendChild(container);
   }
@@ -51,8 +60,10 @@ class BalsamiqViewer {
     return thumbnails[0].values.map(BalsamiqViewer.parsePreview);
   }
 
-  getTitle(resourceID) {
-    return this.database.exec(`SELECT * FROM resources WHERE id = '${resourceID}'`);
+  getResource(resourceID) {
+    const resources = this.database.exec(`SELECT * FROM resources WHERE id = '${resourceID}'`);
+
+    return resources[0];
   }
 
   renderPreview(preview) {
@@ -61,15 +72,15 @@ class BalsamiqViewer {
     previewElement.classList.add('preview');
     previewElement.innerHTML = this.renderTemplate(preview);
 
-    return previewElement.outerHTML;
+    return previewElement;
   }
 
   renderTemplate(preview) {
-    const title = this.getTitle(preview.resourceID);
-    const name = BalsamiqViewer.parseTitle(title);
+    const resource = this.getResource(preview.resourceID);
+    const name = BalsamiqViewer.parseTitle(resource);
     const image = preview.image;
 
-    const template = BalsamiqViewer.PREVIEW_TEMPLATE({
+    const template = PREVIEW_TEMPLATE({
       name,
       image,
     });
@@ -81,8 +92,16 @@ class BalsamiqViewer {
     return JSON.parse(preview[1]);
   }
 
-  static parseTitle(title) {
-    return JSON.parse(title[0].values[0][2]).name;
+  /*
+   * resource = {
+   *   columns: ['ID', 'BRANCHID', 'ATTRIBUTES', 'DATA'],
+   *   values: [['id', 'branchId', 'attributes', 'data']],
+   * }
+   *
+   * 'attributes' being a JSON string containing the `name` property.
+   */
+  static parseTitle(resource) {
+    return JSON.parse(resource.values[0][2]).name;
   }
 
   static onError() {
@@ -92,13 +111,4 @@ class BalsamiqViewer {
   }
 }
 
-BalsamiqViewer.PREVIEW_TEMPLATE = _template(`
-  <div class="panel panel-default">
-    <div class="panel-heading"><%- name %></div>
-    <div class="panel-body">
-      <img class="img-thumbnail" src="data:image/png;base64,<%- image %>"/>
-    </div>
-  </div>
-`);
-
 export default BalsamiqViewer;
diff --git a/app/views/projects/blob/viewers/_balsamiq.html.haml b/app/views/projects/blob/viewers/_balsamiq.html.haml
index 573b24ae44f7f..28670e7de9755 100644
--- a/app/views/projects/blob/viewers/_balsamiq.html.haml
+++ b/app/views/projects/blob/viewers/_balsamiq.html.haml
@@ -1,4 +1,4 @@
 - content_for :page_specific_javascripts do
   = page_specific_javascript_bundle_tag('balsamiq_viewer')
 
-.file-content.balsamiq-viewer#js-balsamiq-viewer{ data: { endpoint: namespace_project_raw_path(@project.namespace, @project, @id) } }
+.file-content.balsamiq-viewer#js-balsamiq-viewer{ data: { endpoint: blob_raw_url } }
diff --git a/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js b/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
index 557eb721a2b8c..85816ee1f1161 100644
--- a/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
+++ b/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
@@ -1,6 +1,5 @@
 import sqljs from 'sql.js';
 import BalsamiqViewer from '~/blob/balsamiq/balsamiq_viewer';
-import * as spinnerSrc from '~/spinner';
 import ClassSpecHelper from '../../helpers/class_spec_helper';
 
 describe('BalsamiqViewer', () => {
@@ -17,8 +16,6 @@ describe('BalsamiqViewer', () => {
         },
       };
 
-      spyOn(spinnerSrc, 'default');
-
       balsamiqViewer = new BalsamiqViewer(viewer);
     });
 
@@ -29,38 +26,23 @@ describe('BalsamiqViewer', () => {
     it('should set .endpoint', () => {
       expect(balsamiqViewer.endpoint).toBe(endpoint);
     });
-
-    it('should instantiate Spinner', () => {
-      expect(spinnerSrc.default).toHaveBeenCalledWith(viewer);
-    });
-
-    it('should set .spinner', () => {
-      expect(balsamiqViewer.spinner).toEqual(jasmine.any(spinnerSrc.default));
-    });
   });
 
   describe('loadFile', () => {
     let xhr;
-    let spinner;
 
     beforeEach(() => {
       endpoint = 'endpoint';
       xhr = jasmine.createSpyObj('xhr', ['open', 'send']);
-      spinner = jasmine.createSpyObj('spinner', ['start']);
 
       balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['renderFile']);
       balsamiqViewer.endpoint = endpoint;
-      balsamiqViewer.spinner = spinner;
 
       spyOn(window, 'XMLHttpRequest').and.returnValue(xhr);
 
       BalsamiqViewer.prototype.loadFile.call(balsamiqViewer);
     });
 
-    it('should instantiate XMLHttpRequest', () => {
-      expect(window.XMLHttpRequest).toHaveBeenCalled();
-    });
-
     it('should call .open', () => {
       expect(xhr.open).toHaveBeenCalledWith('GET', endpoint, true);
     });
@@ -69,25 +51,12 @@ describe('BalsamiqViewer', () => {
       expect(xhr.responseType).toBe('arraybuffer');
     });
 
-    it('should set .onload', () => {
-      expect(xhr.onload).toEqual(jasmine.any(Function));
-    });
-
-    it('should set .onerror', () => {
-      expect(xhr.onerror).toBe(BalsamiqViewer.onError);
-    });
-
-    it('should call spinner.start', () => {
-      expect(spinner.start).toHaveBeenCalled();
-    });
-
     it('should call .send', () => {
       expect(xhr.send).toHaveBeenCalled();
     });
   });
 
   describe('renderFile', () => {
-    let spinner;
     let container;
     let loadEvent;
     let previews;
@@ -95,12 +64,10 @@ describe('BalsamiqViewer', () => {
     beforeEach(() => {
       loadEvent = { target: { response: {} } };
       viewer = jasmine.createSpyObj('viewer', ['appendChild']);
-      spinner = jasmine.createSpyObj('spinner', ['stop']);
-      previews = [0, 1, 2];
+      previews = [document.createElement('ul'), document.createElement('ul')];
 
       balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['initDatabase', 'getPreviews', 'renderPreview']);
       balsamiqViewer.viewer = viewer;
-      balsamiqViewer.spinner = spinner;
 
       balsamiqViewer.getPreviews.and.returnValue(previews);
       balsamiqViewer.renderPreview.and.callFake(preview => preview);
@@ -111,10 +78,6 @@ describe('BalsamiqViewer', () => {
       BalsamiqViewer.prototype.renderFile.call(balsamiqViewer, loadEvent);
     });
 
-    it('should call spinner.stop', () => {
-      expect(spinner.stop).toHaveBeenCalled();
-    });
-
     it('should call .initDatabase', () => {
       expect(balsamiqViewer.initDatabase).toHaveBeenCalledWith(loadEvent.target.response);
     });
@@ -126,15 +89,15 @@ describe('BalsamiqViewer', () => {
     it('should call .renderPreview for each preview', () => {
       const allArgs = balsamiqViewer.renderPreview.calls.allArgs();
 
-      expect(allArgs.length).toBe(3);
+      expect(allArgs.length).toBe(2);
 
       previews.forEach((preview, i) => {
         expect(allArgs[i][0]).toBe(preview);
       });
     });
 
-    it('should set .innerHTML', () => {
-      expect(container.innerHTML).toBe('012');
+    it('should set the container HTML', () => {
+      expect(container.innerHTML).toBe('<ul></ul><ul></ul>');
     });
 
     it('should add inline preview classes', () => {
@@ -216,16 +179,16 @@ describe('BalsamiqViewer', () => {
     });
   });
 
-  describe('getTitle', () => {
+  describe('getResource', () => {
     let database;
     let resourceID;
     let resource;
-    let getTitle;
+    let getResource;
 
     beforeEach(() => {
       database = jasmine.createSpyObj('database', ['exec']);
       resourceID = 4;
-      resource = 'resource';
+      resource = ['resource'];
 
       balsamiqViewer = {
         database,
@@ -233,7 +196,7 @@ describe('BalsamiqViewer', () => {
 
       database.exec.and.returnValue(resource);
 
-      getTitle = BalsamiqViewer.prototype.getTitle.call(balsamiqViewer, resourceID);
+      getResource = BalsamiqViewer.prototype.getResource.call(balsamiqViewer, resourceID);
     });
 
     it('should call database.exec', () => {
@@ -241,7 +204,7 @@ describe('BalsamiqViewer', () => {
     });
 
     it('should return the selected resource', () => {
-      expect(getTitle).toBe(resource);
+      expect(getResource).toBe(resource[0]);
     });
   });
 
@@ -267,10 +230,6 @@ describe('BalsamiqViewer', () => {
       renderPreview = BalsamiqViewer.prototype.renderPreview.call(balsamiqViewer, preview);
     });
 
-    it('should call document.createElement', () => {
-      expect(document.createElement).toHaveBeenCalledWith('li');
-    });
-
     it('should call classList.add', () => {
       expect(previewElement.classList.add).toHaveBeenCalledWith('preview');
     });
@@ -283,22 +242,22 @@ describe('BalsamiqViewer', () => {
       expect(previewElement.innerHTML).toBe(innerHTML);
     });
 
-    it('should return .outerHTML', () => {
-      expect(renderPreview).toBe(previewElement.outerHTML);
+    it('should return element', () => {
+      expect(renderPreview).toBe(previewElement);
     });
   });
 
   describe('renderTemplate', () => {
     let preview;
     let name;
-    let title;
+    let resource;
     let template;
     let renderTemplate;
 
     beforeEach(() => {
       preview = { resourceID: 1, image: 'image' };
       name = 'name';
-      title = 'title';
+      resource = 'resource';
       template = `
         <div class="panel panel-default">
           <div class="panel-heading">name</div>
@@ -308,32 +267,24 @@ describe('BalsamiqViewer', () => {
         </div>
       `;
 
-      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['getTitle']);
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['getResource']);
 
       spyOn(BalsamiqViewer, 'parseTitle').and.returnValue(name);
-      spyOn(BalsamiqViewer, 'PREVIEW_TEMPLATE').and.returnValue(template);
-      balsamiqViewer.getTitle.and.returnValue(title);
+      balsamiqViewer.getResource.and.returnValue(resource);
 
       renderTemplate = BalsamiqViewer.prototype.renderTemplate.call(balsamiqViewer, preview);
     });
 
-    it('should call .getTitle', () => {
-      expect(balsamiqViewer.getTitle).toHaveBeenCalledWith(preview.resourceID);
+    it('should call .getResource', () => {
+      expect(balsamiqViewer.getResource).toHaveBeenCalledWith(preview.resourceID);
     });
 
     it('should call .parseTitle', () => {
-      expect(BalsamiqViewer.parseTitle).toHaveBeenCalledWith(title);
-    });
-
-    it('should call .PREVIEW_TEMPLATE', () => {
-      expect(BalsamiqViewer.PREVIEW_TEMPLATE).toHaveBeenCalledWith({
-        name,
-        image: preview.image,
-      });
+      expect(BalsamiqViewer.parseTitle).toHaveBeenCalledWith(resource);
     });
 
     it('should return the template string', function () {
-      expect(renderTemplate.trim()).toBe(template.trim());
+      expect(renderTemplate.replace(/\s/g, '')).toEqual(template.replace(/\s/g, ''));
     });
   });
 
@@ -351,10 +302,6 @@ describe('BalsamiqViewer', () => {
 
     ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'parsePreview');
 
-    it('should call JSON.parse', () => {
-      expect(JSON.parse).toHaveBeenCalledWith(preview[1]);
-    });
-
     it('should return the parsed JSON', () => {
       expect(parsePreview).toEqual(JSON.parse('{ "id": 1 }'));
     });
@@ -365,7 +312,7 @@ describe('BalsamiqViewer', () => {
     let parseTitle;
 
     beforeEach(() => {
-      title = [{ values: [['{}', '{}', '{"name":"name"}']] }];
+      title = { values: [['{}', '{}', '{"name":"name"}']] };
 
       spyOn(JSON, 'parse').and.callThrough();
 
@@ -374,22 +321,16 @@ describe('BalsamiqViewer', () => {
 
     ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'parsePreview');
 
-    it('should call JSON.parse', () => {
-      expect(JSON.parse).toHaveBeenCalledWith(title[0].values[0][2]);
-    });
-
     it('should return the name value', () => {
       expect(parseTitle).toBe('name');
     });
   });
 
   describe('onError', () => {
-    let onError;
-
     beforeEach(() => {
       spyOn(window, 'Flash');
 
-      onError = BalsamiqViewer.onError();
+      BalsamiqViewer.onError();
     });
 
     ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'onError');
@@ -397,9 +338,5 @@ describe('BalsamiqViewer', () => {
     it('should instantiate Flash', () => {
       expect(window.Flash).toHaveBeenCalledWith('Balsamiq file could not be loaded.');
     });
-
-    it('should return Flash', () => {
-      expect(onError).toEqual(jasmine.any(window.Flash));
-    });
   });
 });
-- 
GitLab


From 6b6296e1cde034ed6a335890a44a2f7bc87d782d Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 17:50:41 +0100
Subject: [PATCH 113/363] Reverted false schema diffs

---
 db/schema.rb | 11 ++++++-----
 1 file changed, 6 insertions(+), 5 deletions(-)

diff --git a/db/schema.rb b/db/schema.rb
index a9940be22ffe8..06f1b079dbfc6 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -63,6 +63,7 @@
     t.boolean "shared_runners_enabled", default: true, null: false
     t.integer "max_artifacts_size", default: 100, null: false
     t.string "runners_registration_token"
+    t.integer "max_pages_size", default: 100, null: false
     t.boolean "require_two_factor_authentication", default: false
     t.integer "two_factor_grace_period", default: 48
     t.boolean "metrics_enabled", default: false
@@ -111,16 +112,15 @@
     t.boolean "html_emails_enabled", default: true
     t.string "plantuml_url"
     t.boolean "plantuml_enabled"
-    t.integer "max_pages_size", default: 100, null: false
     t.integer "terminal_max_session_time", default: 0, null: false
     t.string "default_artifacts_expire_in", default: "0", null: false
     t.integer "unique_ips_limit_per_user"
     t.integer "unique_ips_limit_time_window"
     t.boolean "unique_ips_limit_enabled", default: false, null: false
     t.decimal "polling_interval_multiplier", default: 1.0, null: false
+    t.integer "cached_markdown_version"
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
-    t.integer "cached_markdown_version"
     t.boolean "clientside_sentry_enabled", default: false
     t.string "clientside_sentry_dsn"
   end
@@ -734,8 +734,8 @@
     t.integer "visibility_level", default: 20, null: false
     t.boolean "request_access_enabled", default: false, null: false
     t.datetime "deleted_at"
-    t.boolean "lfs_enabled"
     t.text "description_html"
+    t.boolean "lfs_enabled"
     t.integer "parent_id"
     t.boolean "require_two_factor_authentication", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
@@ -968,9 +968,9 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.integer "cached_markdown_version"
   end
 
@@ -1073,6 +1073,7 @@
     t.string "line_code"
     t.string "note_type"
     t.text "position"
+    t.string "in_reply_to_discussion_id"
   end
 
   add_index "sent_notifications", ["reply_key"], name: "index_sent_notifications_on_reply_key", unique: true, using: :btree
@@ -1322,10 +1323,10 @@
     t.string "organization"
     t.boolean "authorized_projects_populated"
     t.boolean "ghost"
+    t.date "last_activity_on"
     t.boolean "notified_of_own_activity"
     t.boolean "require_two_factor_authentication_from_group", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
-    t.date "last_activity_on"
   end
 
   add_index "users", ["admin"], name: "index_users_on_admin", using: :btree
-- 
GitLab


From ca17e258a2233e18912529f33646db27edc58e25 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 17:52:18 +0100
Subject: [PATCH 114/363] Removed debug message

---
 app/views/layouts/_head.html.haml | 1 -
 1 file changed, 1 deletion(-)

diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index 6123440b6e58f..fa4d760a20eb1 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -35,7 +35,6 @@
   = webpack_bundle_tag "main"
   = webpack_bundle_tag "raven" if current_application_settings.clientside_sentry_enabled
 
-  = 'LUKETEST'
   = current_application_settings.clientside_sentry_enabled
 
   - if content_for?(:page_specific_javascripts)
-- 
GitLab


From 637ed8a21e9f9457d1b194f9c591a0813c20cc3e Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 18:09:55 +0100
Subject: [PATCH 115/363] Added migration downtime tag

---
 ...0428123910_add_clientside_sentry_to_application_settings.rb | 3 +++
 1 file changed, 3 insertions(+)

diff --git a/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb b/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
index 2b085b889eb87..380060b18b391 100644
--- a/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
+++ b/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
@@ -1,5 +1,8 @@
 # rubocop:disable all
 class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
+  DOWNTIME = true
+  DOWNTIME_REASON = 'This migration requires downtime because we must add 2 new columns, 1 of which has a default value.'
+
   def change
     change_table :application_settings do |t|
       t.boolean :clientside_sentry_enabled, default: false
-- 
GitLab


From adff7923ca3088df9e07ccd60d6f9fa24fbdbfc5 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 28 Apr 2017 19:08:39 +0100
Subject: [PATCH 116/363] Removed unneeded spec

---
 .../projects/blobs/balsamiq_preview_spec.rb   | 24 -------------------
 1 file changed, 24 deletions(-)
 delete mode 100644 spec/features/projects/blobs/balsamiq_preview_spec.rb

diff --git a/spec/features/projects/blobs/balsamiq_preview_spec.rb b/spec/features/projects/blobs/balsamiq_preview_spec.rb
deleted file mode 100644
index 59475525b813f..0000000000000
--- a/spec/features/projects/blobs/balsamiq_preview_spec.rb
+++ /dev/null
@@ -1,24 +0,0 @@
-require 'spec_helper'
-
-feature 'Balsamiq preview', :feature, :js do
-  include TreeHelper
-
-  let(:user) { create(:user) }
-  let(:project) { create(:project) }
-  let(:branch) { 'add-balsamiq-file' }
-  let(:path) { 'files/images/balsamiq.bmpr' }
-
-  before do
-    project.add_master(user)
-    login_as user
-    visit namespace_project_blob_path(project.namespace, project, tree_join(branch, path))
-  end
-
-  it 'should show a loading icon' do
-    expect(find('.file-content')).to have_selector('.loading')
-  end
-
-  it 'should show a viewer container' do
-    expect(page).to have_selector('.balsamiq-viewer')
-  end
-end
-- 
GitLab


From 357ab10da34102499a32ad569d9b58c8d5b97d17 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 28 Apr 2017 20:59:20 +0100
Subject: [PATCH 117/363] Fixed some spinach & rspec tests

---
 .../javascripts/deploy_keys/components/app.vue   |  2 +-
 features/project/deploy_keys.feature             |  6 ++++++
 features/steps/project/deploy_keys.rb            | 16 +++++++++-------
 spec/features/projects/deploy_keys_spec.rb       | 12 ++++++++----
 spec/serializers/deploy_key_entity_spec.rb       |  9 ++++++++-
 5 files changed, 32 insertions(+), 13 deletions(-)

diff --git a/app/assets/javascripts/deploy_keys/components/app.vue b/app/assets/javascripts/deploy_keys/components/app.vue
index ee2f85bde5388..2ba5001cf56a0 100644
--- a/app/assets/javascripts/deploy_keys/components/app.vue
+++ b/app/assets/javascripts/deploy_keys/components/app.vue
@@ -77,7 +77,7 @@
 </script>
 
 <template>
-  <div class="col-lg-9 col-lg-offset-3 append-bottom-default">
+  <div class="col-lg-9 col-lg-offset-3 append-bottom-default deploy-keys">
     <div
       class="text-center"
       v-if="isLoading && !hasKeys">
diff --git a/features/project/deploy_keys.feature b/features/project/deploy_keys.feature
index 960b4100ee506..6f1ed9ff5b6b6 100644
--- a/features/project/deploy_keys.feature
+++ b/features/project/deploy_keys.feature
@@ -3,28 +3,33 @@ Feature: Project Deploy Keys
     Given I sign in as a user
     And I own project "Shop"
 
+  @javascript
   Scenario: I should see deploy keys list
     Given project has deploy key
     When I visit project deploy keys page
     Then I should see project deploy key
 
+  @javascript
   Scenario: I should see project deploy keys
     Given other projects have deploy keys
     When I visit project deploy keys page
     Then I should see other project deploy key
     And I should only see the same deploy key once
 
+  @javascript
   Scenario: I should see public deploy keys
     Given public deploy key exists
     When I visit project deploy keys page
     Then I should see public deploy key
 
+  @javascript
   Scenario: I add new deploy key
     Given I visit project deploy keys page
     And I submit new deploy key
     Then I should be on deploy keys page
     And I should see newly created deploy key
 
+  @javascript
   Scenario: I attach other project deploy key to project
     Given other projects have deploy keys
     And I visit project deploy keys page
@@ -32,6 +37,7 @@ Feature: Project Deploy Keys
     Then I should be on deploy keys page
     And I should see newly created deploy key
 
+  @javascript
   Scenario: I attach public deploy key to project
     Given public deploy key exists
     And I visit project deploy keys page
diff --git a/features/steps/project/deploy_keys.rb b/features/steps/project/deploy_keys.rb
index ec59a2c094e79..8ad9d4a47419c 100644
--- a/features/steps/project/deploy_keys.rb
+++ b/features/steps/project/deploy_keys.rb
@@ -8,19 +8,19 @@ class Spinach::Features::ProjectDeployKeys < Spinach::FeatureSteps
   end
 
   step 'I should see project deploy key' do
-    page.within '.deploy-keys' do
+    page.within(find('.deploy-keys')) do
       expect(page).to have_content deploy_key.title
     end
   end
 
   step 'I should see other project deploy key' do
-    page.within '.deploy-keys' do
+    page.within(find('.deploy-keys')) do
       expect(page).to have_content other_deploy_key.title
     end
   end
 
   step 'I should see public deploy key' do
-    page.within '.deploy-keys' do
+    page.within(find('.deploy-keys')) do
       expect(page).to have_content public_deploy_key.title
     end
   end
@@ -40,7 +40,8 @@ class Spinach::Features::ProjectDeployKeys < Spinach::FeatureSteps
   end
 
   step 'I should see newly created deploy key' do
-    page.within '.deploy-keys' do
+    @project.reload
+    page.within(find('.deploy-keys')) do
       expect(page).to have_content(deploy_key.title)
     end
   end
@@ -56,7 +57,7 @@ class Spinach::Features::ProjectDeployKeys < Spinach::FeatureSteps
   end
 
   step 'I should only see the same deploy key once' do
-    page.within '.deploy-keys' do
+    page.within(find('.deploy-keys')) do
       expect(page).to have_selector('ul li', count: 1)
     end
   end
@@ -66,8 +67,9 @@ class Spinach::Features::ProjectDeployKeys < Spinach::FeatureSteps
   end
 
   step 'I click attach deploy key' do
-    page.within '.deploy-keys' do
-      click_link 'Enable'
+    page.within(find('.deploy-keys')) do
+      click_button 'Enable'
+      expect(page).not_to have_selector('.fa-spinner')
     end
   end
 
diff --git a/spec/features/projects/deploy_keys_spec.rb b/spec/features/projects/deploy_keys_spec.rb
index 0b997f130ea6a..06abfbbc86b78 100644
--- a/spec/features/projects/deploy_keys_spec.rb
+++ b/spec/features/projects/deploy_keys_spec.rb
@@ -1,6 +1,6 @@
 require 'spec_helper'
 
-describe 'Project deploy keys', feature: true do
+describe 'Project deploy keys', :js, :feature do
   let(:user) { create(:user) }
   let(:project) { create(:project_empty_repo) }
 
@@ -17,9 +17,13 @@
     it 'removes association between project and deploy key' do
       visit namespace_project_settings_repository_path(project.namespace, project)
 
-      page.within '.deploy-keys' do
-        expect { click_on 'Remove' }
-          .to change { project.deploy_keys.count }.by(-1)
+      page.within(find('.deploy-keys')) do
+        expect(page).to have_selector('.deploy-keys li', count: 1)
+
+        click_on 'Remove'
+
+        expect(page).not_to have_selector('.fa-spinner', count: 0)
+        expect(page).to have_selector('.deploy-keys li', count: 0)
       end
     end
   end
diff --git a/spec/serializers/deploy_key_entity_spec.rb b/spec/serializers/deploy_key_entity_spec.rb
index cc3fb193f1be2..e73fbe190ca18 100644
--- a/spec/serializers/deploy_key_entity_spec.rb
+++ b/spec/serializers/deploy_key_entity_spec.rb
@@ -1,6 +1,8 @@
 require 'spec_helper'
 
 describe DeployKeyEntity do
+  include RequestAwareEntity
+  
   let(:user) { create(:user) }
   let(:project) { create(:empty_project, :internal)}
   let(:project_private) { create(:empty_project, :private)}
@@ -22,7 +24,12 @@
       created_at: deploy_key.created_at,
       updated_at: deploy_key.updated_at,
       projects: [
-        { id: project.id, name: project.name, full_path: project.full_path, full_name: project.full_name }
+        {
+          id: project.id,
+          name: project.name,
+          full_path: namespace_project_path(project.namespace, project),
+          full_name: project.full_name
+        }
       ]
     }
 
-- 
GitLab


From dc3d778fac4f9b0d4fbcb574f9ef3b0cd11ad640 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Sun, 30 Apr 2017 12:02:17 +0100
Subject: [PATCH 118/363] Fix failing specs

---
 spec/features/projects/commit/cherry_pick_spec.rb    |  2 +-
 .../projects/environments/environment_spec.rb        |  2 +-
 spec/features/projects/merge_request_button_spec.rb  | 12 ++++++------
 3 files changed, 8 insertions(+), 8 deletions(-)

diff --git a/spec/features/projects/commit/cherry_pick_spec.rb b/spec/features/projects/commit/cherry_pick_spec.rb
index 5d64d42fd6158..8c86a6742c069 100644
--- a/spec/features/projects/commit/cherry_pick_spec.rb
+++ b/spec/features/projects/commit/cherry_pick_spec.rb
@@ -75,7 +75,7 @@
       wait_for_ajax
 
       page.within('#modal-cherry-pick-commit .dropdown-menu .dropdown-content') do
-        click_link 'feature'
+        click_link "'test'"
       end
 
       page.within('#modal-cherry-pick-commit') do
diff --git a/spec/features/projects/environments/environment_spec.rb b/spec/features/projects/environments/environment_spec.rb
index acc3efe04e672..49a4944e8740a 100644
--- a/spec/features/projects/environments/environment_spec.rb
+++ b/spec/features/projects/environments/environment_spec.rb
@@ -200,7 +200,7 @@
     end
 
     scenario 'user deletes the branch with running environment' do
-      visit namespace_project_branches_path(project.namespace, project)
+      visit namespace_project_branches_path(project.namespace, project, page: 2)
 
       remove_branch_with_hooks(project, user, 'feature') do
         page.within('.js-branch-feature') { find('a.btn-remove').click }
diff --git a/spec/features/projects/merge_request_button_spec.rb b/spec/features/projects/merge_request_button_spec.rb
index 05f3162f13cd9..2c5642f9a8c4c 100644
--- a/spec/features/projects/merge_request_button_spec.rb
+++ b/spec/features/projects/merge_request_button_spec.rb
@@ -25,7 +25,7 @@
       it 'shows Create merge request button' do
         href = new_namespace_project_merge_request_path(project.namespace,
                                                         project,
-                                                        merge_request: { source_branch: 'feature',
+                                                        merge_request: { source_branch: "'test'",
                                                                          target_branch: 'master' })
 
         visit url
@@ -69,7 +69,7 @@
         it 'shows Create merge request button' do
           href = new_namespace_project_merge_request_path(forked_project.namespace,
                                                           forked_project,
-                                                          merge_request: { source_branch: 'feature',
+                                                          merge_request: { source_branch: "'test'",
                                                                            target_branch: 'master' })
 
           visit fork_url
@@ -93,16 +93,16 @@
   context 'on compare page' do
     it_behaves_like 'Merge request button only shown when allowed' do
       let(:label) { 'Create merge request' }
-      let(:url) { namespace_project_compare_path(project.namespace, project, from: 'master', to: 'feature') }
-      let(:fork_url) { namespace_project_compare_path(forked_project.namespace, forked_project, from: 'master', to: 'feature') }
+      let(:url) { namespace_project_compare_path(project.namespace, project, from: 'master', to: "'test'") }
+      let(:fork_url) { namespace_project_compare_path(forked_project.namespace, forked_project, from: 'master', to: "'test'") }
     end
   end
 
   context 'on commits page' do
     it_behaves_like 'Merge request button only shown when allowed' do
       let(:label) { 'Create merge request' }
-      let(:url) { namespace_project_commits_path(project.namespace, project, 'feature') }
-      let(:fork_url) { namespace_project_commits_path(forked_project.namespace, forked_project, 'feature') }
+      let(:url) { namespace_project_commits_path(project.namespace, project, "'test'") }
+      let(:fork_url) { namespace_project_commits_path(forked_project.namespace, forked_project, "'test'") }
     end
   end
 end
-- 
GitLab


From 6baaa8a98e10ff93ba3f481052bb68fdafb6e2c1 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Mon, 1 May 2017 13:38:57 +0200
Subject: [PATCH 119/363] Add new ability check for stopping environment

---
 app/policies/environment_policy.rb           | 13 ++++-
 app/services/ci/stop_environments_service.rb |  9 +---
 spec/policies/environment_policy_spec.rb     | 57 ++++++++++++++++++++
 3 files changed, 71 insertions(+), 8 deletions(-)
 create mode 100644 spec/policies/environment_policy_spec.rb

diff --git a/app/policies/environment_policy.rb b/app/policies/environment_policy.rb
index f4219569161e9..0b976e5664d04 100644
--- a/app/policies/environment_policy.rb
+++ b/app/policies/environment_policy.rb
@@ -1,5 +1,16 @@
 class EnvironmentPolicy < BasePolicy
+
+  alias_method :environment, :subject
+
   def rules
-    delegate! @subject.project
+    delegate! environment.project
+
+    if environment.stop_action?
+      delegate! environment.stop_action
+    end
+
+    if can?(:create_deployment) && can?(:play_build)
+      can! :stop_environment
+    end
   end
 end
diff --git a/app/services/ci/stop_environments_service.rb b/app/services/ci/stop_environments_service.rb
index bd9735fc0accf..43c9a065fcf75 100644
--- a/app/services/ci/stop_environments_service.rb
+++ b/app/services/ci/stop_environments_service.rb
@@ -5,12 +5,11 @@ class StopEnvironmentsService < BaseService
     def execute(branch_name)
       @ref = branch_name
 
-      return unless has_ref?
-      return unless can?(current_user, :create_deployment, project)
+      return unless @ref.present?
 
       environments.each do |environment|
         next unless environment.stop_action?
-        next unless can?(current_user, :play_build, environment.stop_action)
+        next unless can?(current_user, :stop_environment, environment)
 
         environment.stop_with_action!(current_user)
       end
@@ -18,10 +17,6 @@ def execute(branch_name)
 
     private
 
-    def has_ref?
-      @ref.present?
-    end
-
     def environments
       @environments ||= EnvironmentsFinder
         .new(project, current_user, ref: @ref, recently_updated: true)
diff --git a/spec/policies/environment_policy_spec.rb b/spec/policies/environment_policy_spec.rb
new file mode 100644
index 0000000000000..f43caffa94633
--- /dev/null
+++ b/spec/policies/environment_policy_spec.rb
@@ -0,0 +1,57 @@
+require 'spec_helper'
+
+describe Ci::EnvironmentPolicy do
+  let(:user) { create(:user) }
+  let(:project) { create(:project) }
+
+  let(:environment) do
+    create(:environment, :with_review_app, project: project)
+  end
+
+  let(:policies) do
+    described_class.abilities(user, environment).to_set
+  end
+
+  describe '#rules' do
+    context 'when user does not have access to the project' do
+      let(:project) { create(:project, :private) }
+
+      it 'does not include ability to stop environment' do
+        expect(policies).not_to include :stop_environment
+      end
+    end
+
+    context 'when anonymous user has access to the project' do
+      let(:project) { create(:project, :public) }
+
+      it 'does not include ability to stop environment' do
+        expect(policies).not_to include :stop_environment
+      end
+    end
+
+    context 'when team member has access to the project' do
+      let(:project) { create(:project, :public) }
+
+      before do
+        project.add_master(user)
+      end
+
+      context 'when team member has ability to stop environment' do
+        it 'does includes ability to stop environment' do
+          expect(policies).to include :stop_environment
+        end
+      end
+
+      context 'when team member has no ability to stop environment' do
+        before do
+          create(:protected_branch, :no_one_can_push,
+                 name: 'master', project: project)
+        end
+
+        it 'does not include ability to stop environment' do
+          expect(policies).not_to include :stop_environment
+        end
+      end
+    end
+  end
+end
-- 
GitLab


From 4f2cc5951f3d2862a44fe124ef9a879162028e62 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Mon, 1 May 2017 14:29:20 +0200
Subject: [PATCH 120/363] Extend action tooltop to show info about abilities

---
 lib/gitlab/ci/status/build/play.rb            |  6 +-
 .../gitlab/ci/status/build/factory_spec.rb    |  2 +-
 spec/lib/gitlab/ci/status/build/play_spec.rb  | 65 ++++++++++---------
 3 files changed, 42 insertions(+), 31 deletions(-)

diff --git a/lib/gitlab/ci/status/build/play.rb b/lib/gitlab/ci/status/build/play.rb
index 4c893f62925e3..8130c7d1c90ef 100644
--- a/lib/gitlab/ci/status/build/play.rb
+++ b/lib/gitlab/ci/status/build/play.rb
@@ -6,7 +6,11 @@ class Play < SimpleDelegator
           include Status::Extended
 
           def label
-            'manual play action'
+            if has_action?
+              'manual play action'
+            else
+              'manual play action (not allowed)'
+            end
           end
 
           def has_action?
diff --git a/spec/lib/gitlab/ci/status/build/factory_spec.rb b/spec/lib/gitlab/ci/status/build/factory_spec.rb
index 2ab67127b1edd..2de00c289450c 100644
--- a/spec/lib/gitlab/ci/status/build/factory_spec.rb
+++ b/spec/lib/gitlab/ci/status/build/factory_spec.rb
@@ -216,7 +216,7 @@
         expect(status.group).to eq 'manual'
         expect(status.icon).to eq 'icon_status_manual'
         expect(status.favicon).to eq 'favicon_status_manual'
-        expect(status.label).to eq 'manual play action'
+        expect(status.label).to include 'manual play action'
         expect(status).to have_details
         expect(status.action_path).to include 'play'
       end
diff --git a/spec/lib/gitlab/ci/status/build/play_spec.rb b/spec/lib/gitlab/ci/status/build/play_spec.rb
index 95f1388a9b971..abefdbe7c66ca 100644
--- a/spec/lib/gitlab/ci/status/build/play_spec.rb
+++ b/spec/lib/gitlab/ci/status/build/play_spec.rb
@@ -1,51 +1,58 @@
 require 'spec_helper'
 
 describe Gitlab::Ci::Status::Build::Play do
-  let(:status) { double('core') }
-  let(:user) { double('user') }
+  let(:user) { create(:user) }
+  let(:build) { create(:ci_build, :manual) }
+  let(:status) { Gitlab::Ci::Status::Core.new(build, user) }
 
   subject { described_class.new(status) }
 
-  describe '#label' do
-    it { expect(subject.label).to eq 'manual play action' }
-  end
-
-  describe 'action details' do
-    let(:user) { create(:user) }
-    let(:build) { create(:ci_build, :manual) }
-    let(:status) { Gitlab::Ci::Status::Core.new(build, user) }
+  context 'when user is allowed to update build' do
+    context 'when user can push to branch' do
+      before { build.project.add_master(user) }
 
-    describe '#has_action?' do
-      context 'when user is allowed to update build' do
-        context 'when user can push to branch' do
-          before { build.project.add_master(user) }
+      describe '#has_action?' do
+        it { is_expected.to have_action }
+      end
 
-          it { is_expected.to have_action }
+      describe '#label' do
+        it 'has a label that says it is a manual action' do
+          expect(subject.label).to eq 'manual play action'
         end
+      end
+    end
 
-        context 'when user can not push to the branch' do
-          before { build.project.add_developer(user) }
+    context 'when user can not push to the branch' do
+      before { build.project.add_developer(user) }
 
-          it { is_expected.not_to have_action }
-        end
+      describe 'has_action?' do
+        it { is_expected.not_to have_action }
       end
 
-      context 'when user is not allowed to update build' do
-        it { is_expected.not_to have_action }
+      describe '#label' do
+        it 'has a label that says user is not allowed to play it' do
+          expect(subject.label).to eq 'manual play action (not allowed)'
+        end
       end
     end
+  end
 
-    describe '#action_path' do
-      it { expect(subject.action_path).to include "#{build.id}/play" }
+  context 'when user is not allowed to update build' do
+    describe '#has_action?' do
+      it { is_expected.not_to have_action }
     end
+  end
 
-    describe '#action_icon' do
-      it { expect(subject.action_icon).to eq 'icon_action_play' }
-    end
+  describe '#action_path' do
+    it { expect(subject.action_path).to include "#{build.id}/play" }
+  end
 
-    describe '#action_title' do
-      it { expect(subject.action_title).to eq 'Play' }
-    end
+  describe '#action_icon' do
+    it { expect(subject.action_icon).to eq 'icon_action_play' }
+  end
+
+  describe '#action_title' do
+    it { expect(subject.action_title).to eq 'Play' }
   end
 
   describe '.matches?' do
-- 
GitLab


From e5df86f54d6165e0b3871678dd987956dddfa061 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Mon, 1 May 2017 14:30:50 +0200
Subject: [PATCH 121/363] Fix Rubocop offense in environments policy class

---
 app/policies/environment_policy.rb | 1 -
 spec/models/ci/build_spec.rb       | 2 +-
 2 files changed, 1 insertion(+), 2 deletions(-)

diff --git a/app/policies/environment_policy.rb b/app/policies/environment_policy.rb
index 0b976e5664d04..c0c9bbf2d9e9d 100644
--- a/app/policies/environment_policy.rb
+++ b/app/policies/environment_policy.rb
@@ -1,5 +1,4 @@
 class EnvironmentPolicy < BasePolicy
-
   alias_method :environment, :subject
 
   def rules
diff --git a/spec/models/ci/build_spec.rb b/spec/models/ci/build_spec.rb
index b2f9a61f7f3a2..5231ce28c9df1 100644
--- a/spec/models/ci/build_spec.rb
+++ b/spec/models/ci/build_spec.rb
@@ -911,7 +911,7 @@ def create_mr(build, pipeline, factory: :merge_request, created_at: Time.now)
       it { is_expected.to eq(environment) }
     end
 
-    context 'when  referenced with a variable' do
+    context 'when referenced with a variable' do
       let(:build) do
         create(:ci_build, pipeline: pipeline, environment: "foo-$CI_COMMIT_REF_NAME")
       end
-- 
GitLab


From 3584e74eacfc101c64fe1305bbf97515e86ebc88 Mon Sep 17 00:00:00 2001
From: Eric Eastwood <contact@ericeastwood.com>
Date: Tue, 25 Apr 2017 04:34:46 -0500
Subject: [PATCH 122/363] Fix MR target branch selector dropdown placement
 cut-off

Fix https://gitlab.com/gitlab-org/gitlab-ce/issues/31271

When the dropdown items are too wide, constrain the dropdown to the
offsetParent. Otherwise, let it naturally flow as Select2 wants.
---
 app/assets/javascripts/dispatcher.js          |  2 +
 .../issuable/auto_width_dropdown_select.js    | 38 +++++++++++++++++++
 .../stylesheets/pages/merge_requests.scss     |  4 ++
 .../issuable/form/_branch_chooser.html.haml   | 10 +++--
 4 files changed, 51 insertions(+), 3 deletions(-)
 create mode 100644 app/assets/javascripts/issuable/auto_width_dropdown_select.js

diff --git a/app/assets/javascripts/dispatcher.js b/app/assets/javascripts/dispatcher.js
index 15fe87f21ea7a..c0f11c0140575 100644
--- a/app/assets/javascripts/dispatcher.js
+++ b/app/assets/javascripts/dispatcher.js
@@ -49,6 +49,7 @@ import UserCallout from './user_callout';
 import { ProtectedTagCreate, ProtectedTagEditList } from './protected_tags';
 import ShortcutsWiki from './shortcuts_wiki';
 import BlobViewer from './blob/viewer/index';
+import AutoWidthDropdownSelect from './issuable/auto_width_dropdown_select';
 
 const ShortcutsBlob = require('./shortcuts_blob');
 
@@ -186,6 +187,7 @@ const ShortcutsBlob = require('./shortcuts_blob');
           new LabelsSelect();
           new MilestoneSelect();
           new gl.IssuableTemplateSelectors();
+          new AutoWidthDropdownSelect($('.js-target-branch-select')).init();
           break;
         case 'projects:tags:new':
           new ZenMode();
diff --git a/app/assets/javascripts/issuable/auto_width_dropdown_select.js b/app/assets/javascripts/issuable/auto_width_dropdown_select.js
new file mode 100644
index 0000000000000..2203a56315e86
--- /dev/null
+++ b/app/assets/javascripts/issuable/auto_width_dropdown_select.js
@@ -0,0 +1,38 @@
+let instanceCount = 0;
+
+class AutoWidthDropdownSelect {
+  constructor(selectElement) {
+    this.$selectElement = $(selectElement);
+    this.dropdownClass = `js-auto-width-select-dropdown-${instanceCount}`;
+    instanceCount += 1;
+  }
+
+  init() {
+    const dropdownClass = this.dropdownClass;
+    this.$selectElement.select2({
+      dropdownCssClass: dropdownClass,
+      dropdownCss() {
+        let resultantWidth = 'auto';
+        const $dropdown = $(`.${dropdownClass}`);
+
+        // We have to look at the parent because
+        // `offsetParent` on a `display: none;` is `null`
+        const offsetParentWidth = $(this).parent().offsetParent().width();
+        // Reset any width to let it naturally flow
+        $dropdown.css('width', 'auto');
+        if ($dropdown.outerWidth(false) > offsetParentWidth) {
+          resultantWidth = offsetParentWidth;
+        }
+
+        return {
+          width: resultantWidth,
+          maxWidth: offsetParentWidth,
+        };
+      },
+    });
+
+    return this;
+  }
+}
+
+export default AutoWidthDropdownSelect;
diff --git a/app/assets/stylesheets/pages/merge_requests.scss b/app/assets/stylesheets/pages/merge_requests.scss
index 6a419384a3472..05e7e565177f5 100644
--- a/app/assets/stylesheets/pages/merge_requests.scss
+++ b/app/assets/stylesheets/pages/merge_requests.scss
@@ -482,6 +482,10 @@
   }
 }
 
+.target-branch-select-dropdown-container {
+  position: relative;
+}
+
 .assign-to-me-link {
   padding-left: 12px;
   white-space: nowrap;
diff --git a/app/views/shared/issuable/form/_branch_chooser.html.haml b/app/views/shared/issuable/form/_branch_chooser.html.haml
index 2793e7bcff430..f57b4d899ce98 100644
--- a/app/views/shared/issuable/form/_branch_chooser.html.haml
+++ b/app/views/shared/issuable/form/_branch_chooser.html.haml
@@ -10,12 +10,16 @@
     = form.label :source_branch, class: 'control-label'
     .col-sm-10
       .issuable-form-select-holder
-        = form.select(:source_branch, [issuable.source_branch], {}, { class: 'source_branch select2 span2', disabled: true })
+        = form.select(:source_branch, [issuable.source_branch], {}, { class: 'source_branch select2', disabled: true })
 .form-group
   = form.label :target_branch, class: 'control-label'
-  .col-sm-10
+  .col-sm-10.target-branch-select-dropdown-container
     .issuable-form-select-holder
-      = form.select(:target_branch, issuable.target_branches, { include_blank: true }, { class: 'target_branch select2 span2', disabled: issuable.new_record?, data: { placeholder: "Select branch" }})
+      = form.select(:target_branch, issuable.target_branches,
+        { include_blank: true },
+        { class: 'target_branch js-target-branch-select',
+          disabled: issuable.new_record?,
+          data: { placeholder: "Select branch" }})
     - if issuable.new_record?
       &nbsp;
       = link_to 'Change branches', mr_change_branches_path(issuable)
-- 
GitLab


From 78a8be969f47af2fca2bff8b177f47238f8e7516 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Mon, 1 May 2017 15:45:50 -0600
Subject: [PATCH 123/363] make tab title legit

---
 .../issue_show/issue_title_description.vue         | 14 ++++++++++++--
 app/controllers/projects/issues_controller.rb      |  2 ++
 .../issue_show/issue_title_description_spec.js     |  6 ++++--
 spec/javascripts/issue_show/mock_data.js           |  3 +++
 4 files changed, 21 insertions(+), 4 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index e21667f2ac709..6fdf56fc93830 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -31,6 +31,7 @@ export default {
       poll,
       timeoutId: null,
       title: '<span></span>',
+      titleText: '',
       description: '<span></span>',
       descriptionText: '',
       descriptionChange: false,
@@ -40,16 +41,18 @@ export default {
   methods: {
     renderResponse(res) {
       const data = JSON.parse(res.body);
+      this.issueIID = data.issue_number;
       this.triggerAnimation(data);
     },
     updateTaskHTML(data) {
       this.taskStatus = data.task_status;
       document.querySelector('#task_status').innerText = this.taskStatus;
     },
-    elementsToVisualize(noTitleChange, noDescriptionChange) {
+    elementsToVisualize(noTitleChange, noDescriptionChange, data) {
       const elementStack = [];
 
       if (!noTitleChange) {
+        this.titleText = data.title_text;
         elementStack.push(this.$el.querySelector('.title'));
       }
 
@@ -66,11 +69,17 @@ export default {
 
       return elementStack;
     },
+    setTabTitle() {
+      const currentTabTitle = document.querySelector('title');
+      const currentTabTitleScope = currentTabTitle.innerText.split('·');
+      currentTabTitleScope[0] = `${this.titleText} (#${this.issueIID}) `;
+      currentTabTitle.innerText = currentTabTitleScope.join('·');
+    },
     animate(title, description, elementsToVisualize) {
       this.timeoutId = setTimeout(() => {
         this.title = title;
         this.description = description;
-        document.querySelector('title').innerText = title;
+        this.setTabTitle();
 
         elementsToVisualize.forEach((element) => {
           element.classList.remove('issue-realtime-pre-pulse');
@@ -99,6 +108,7 @@ export default {
       const elementsToVisualize = this.elementsToVisualize(
         noTitleChange,
         noDescriptionChange,
+        data,
       );
 
       this.animate(title, description, elementsToVisualize);
diff --git a/app/controllers/projects/issues_controller.rb b/app/controllers/projects/issues_controller.rb
index e5c1505ece65d..7932b92c00549 100644
--- a/app/controllers/projects/issues_controller.rb
+++ b/app/controllers/projects/issues_controller.rb
@@ -201,9 +201,11 @@ def rendered_title
 
     render json: {
       title: view_context.markdown_field(@issue, :title),
+      title_text: @issue.title,
       description: view_context.markdown_field(@issue, :description),
       description_text: @issue.description,
       task_status: @issue.task_status,
+      issue_number: @issue.iid,
     }
   end
 
diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index 756eda54f798c..30daa3e7eda07 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -13,7 +13,9 @@ const issueShowInterceptor = (request, next) => {
   }));
 };
 
-describe('Issue Title', () => {
+fdescribe('Issue Title', () => {
+  document.body.innerHTML = '<span id="task_status"></span>';
+
   const comps = {
     IssueTitleComponent: {},
   };
@@ -40,7 +42,7 @@ describe('Issue Title', () => {
     // need setTimeout because actual setTimeout in code :P
     setTimeout(() => {
       expect(document.querySelector('title').innerText)
-        .toContain('this is a title');
+        .toContain('this is a title (#1)');
 
       expect(issueShowComponent.$el.querySelector('.title').innerHTML)
         .toContain('<p>this is a title</p>');
diff --git a/spec/javascripts/issue_show/mock_data.js b/spec/javascripts/issue_show/mock_data.js
index 37e86cfe33d5b..b7b01391a18a8 100644
--- a/spec/javascripts/issue_show/mock_data.js
+++ b/spec/javascripts/issue_show/mock_data.js
@@ -1,5 +1,8 @@
 export default {
   title: '<p>this is a title</p>',
+  title_text: 'this is a title',
   description: '<p>this is a description!</p>',
   description_text: 'this is a description',
+  issue_number: 1,
+  task_status: '2/4 completed',
 };
-- 
GitLab


From a194e874e2bcd61b9ca5f87a6918a69eb8833aa9 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Mon, 1 May 2017 16:39:39 -0600
Subject: [PATCH 124/363] previous and current description

---
 .../issue_show/issue_title_description.vue    | 74 +++++++++++++------
 .../issue_title_description_spec.js           |  2 +-
 2 files changed, 54 insertions(+), 22 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 6fdf56fc93830..c7adec878a31f 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -29,36 +29,43 @@ export default {
 
     return {
       poll,
+      data: {},
+      current: true,
       timeoutId: null,
       title: '<span></span>',
       titleText: '',
       description: '<span></span>',
       descriptionText: '',
       descriptionChange: false,
+      previousDescription: null,
       taskStatus: '',
     };
   },
   methods: {
     renderResponse(res) {
       const data = JSON.parse(res.body);
-      this.issueIID = data.issue_number;
+      this.data = data;
+      this.issueIID = this.data.issue_number;
       this.triggerAnimation(data);
     },
-    updateTaskHTML(data) {
-      this.taskStatus = data.task_status;
+    updateTaskHTML() {
+      this.taskStatus = this.data.task_status;
       document.querySelector('#task_status').innerText = this.taskStatus;
     },
-    elementsToVisualize(noTitleChange, noDescriptionChange, data) {
+    elementsToVisualize(noTitleChange, noDescriptionChange) {
       const elementStack = [];
 
       if (!noTitleChange) {
-        this.titleText = data.title_text;
+        this.titleText = this.data.title_text;
         elementStack.push(this.$el.querySelector('.title'));
       }
 
       if (!noDescriptionChange) {
         // only change to true when we need to bind TaskLists the html of description
         this.descriptionChange = true;
+        if (this.description !== '<span></span>') {
+          this.previousDescription = this.description;
+        }
         elementStack.push(this.$el.querySelector('.wiki'));
       }
 
@@ -89,35 +96,49 @@ export default {
         clearTimeout(this.timeoutId);
       }, 0);
     },
-    triggerAnimation(data) {
+    triggerAnimation() {
       // always reset to false before checking the change
       this.descriptionChange = false;
 
-      const { title, description } = data;
-      this.descriptionText = data.description_text;
-      this.updateTaskHTML(data);
+      const { title, description } = this.data;
+      this.descriptionText = this.data.description_text;
+
+      this.updateTaskHTML();
+
+      const noTitleChange = this.title === title;
+      const noDescriptionChange = this.description === description;
+
       /**
       * since opacity is changed, even if there is no diff for Vue to update
       * we must check the title/description even on a 304 to ensure no visual change
       */
-      const noTitleChange = this.title === title;
-      const noDescriptionChange = this.description === description;
-
       if (noTitleChange && noDescriptionChange) return;
 
       const elementsToVisualize = this.elementsToVisualize(
         noTitleChange,
         noDescriptionChange,
-        data,
       );
 
       this.animate(title, description, elementsToVisualize);
     },
+    handleCurrentOrPrevious() {
+      this.descriptionChange = true;
+      this.current = !this.current;
+    },
   },
   computed: {
     descriptionClass() {
       return `description ${this.candescription} is-task-list-enabled`;
     },
+    showDescription() {
+      return this.current ? this.description : this.previousDescription;
+    },
+    previousOrCurrentButtonText() {
+      return this.current ? '<< Show Previous Decription' : 'Show Current Description >>';
+    },
+    prevCurrBtnClass() {
+      return this.current ? 'btn btn-sm btn-default' : 'btn btn-sm btn-primary';
+    },
   },
   created() {
     if (!Visibility.hidden()) {
@@ -135,14 +156,17 @@ export default {
   updated() {
     // if new html is injected (description changed) - bind TaskList and call renderGFM
     if (this.descriptionChange) {
-      const tl = new gl.TaskList({
-        dataType: 'issue',
-        fieldName: 'description',
-        selector: '.detail-page-description',
-      });
-
       $(this.$refs['issue-content-container-gfm-entry']).renderGFM();
-      return tl;
+
+      if (this.current) {
+        const tl = new gl.TaskList({
+          dataType: 'issue',
+          fieldName: 'description',
+          selector: '.detail-page-description',
+        });
+
+        return tl;
+      }
     }
     return null;
   },
@@ -156,9 +180,17 @@ export default {
       :class="descriptionClass"
       v-if="description"
     >
+      <div v-if="previousDescription">
+        <button
+          :class="prevCurrBtnClass"
+          @click="handleCurrentOrPrevious"
+        >
+          {{ previousOrCurrentButtonText }}
+        </button>
+      </div><br>
       <div
         class="wiki issue-realtime-trigger-pulse"
-        v-html="description"
+        v-html="showDescription"
         ref="issue-content-container-gfm-entry"
       >
       </div>
diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index 30daa3e7eda07..4c7d4748693dd 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -13,7 +13,7 @@ const issueShowInterceptor = (request, next) => {
   }));
 };
 
-fdescribe('Issue Title', () => {
+describe('Issue Title', () => {
   document.body.innerHTML = '<span id="task_status"></span>';
 
   const comps = {
-- 
GitLab


From dd9f848653d9635e7f17e608e8ff6cceca608068 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Mon, 1 May 2017 18:59:32 -0600
Subject: [PATCH 125/363] handle tasks and fix some specs

---
 .../issue_show/issue_title_description.vue    | 36 ++++++++++++-------
 spec/features/issues/award_spec.rb            |  6 ++++
 ...issuable_slash_commands_shared_examples.rb |  2 ++
 3 files changed, 31 insertions(+), 13 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index c7adec878a31f..dd8794188d783 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -20,7 +20,7 @@ export default {
       errorCallback: (err) => {
         if (process.env.NODE_ENV !== 'production') {
           // eslint-disable-next-line no-console
-          console.error('ISSUE SHOW REALTIME ERROR', err);
+          console.error('ISSUE SHOW REALTIME ERROR', err, err.stack);
         } else {
           throw new Error(err);
         }
@@ -29,7 +29,7 @@ export default {
 
     return {
       poll,
-      data: {},
+      apiData: {},
       current: true,
       timeoutId: null,
       title: '<span></span>',
@@ -43,29 +43,41 @@ export default {
   },
   methods: {
     renderResponse(res) {
-      const data = JSON.parse(res.body);
-      this.data = data;
-      this.issueIID = this.data.issue_number;
-      this.triggerAnimation(data);
+      this.apiData = JSON.parse(res.body);
+      this.issueIID = this.apiData.issue_number;
+      this.triggerAnimation();
     },
     updateTaskHTML() {
-      this.taskStatus = this.data.task_status;
-      document.querySelector('#task_status').innerText = this.taskStatus;
+      const tasks = document.querySelector('#task_status_short');
+      const zeroTasks = this.apiData.task_status.includes('0 of 0');
+
+      if (tasks && !zeroTasks) {
+        tasks.innerText = this.apiData.task_status;
+      } else if (this.apiData.task_status.includes('0 of 0')) {
+        $('#task_status_short').remove();
+      } else if (!tasks && !zeroTasks) {
+        $('.issuable-header').append(`
+          <span id="task_status_short" class="hidden-md hidden-lg">${this.apiData.task_status}</span>
+        `);
+      }
     },
     elementsToVisualize(noTitleChange, noDescriptionChange) {
       const elementStack = [];
 
       if (!noTitleChange) {
-        this.titleText = this.data.title_text;
+        this.titleText = this.apiData.title_text;
         elementStack.push(this.$el.querySelector('.title'));
       }
 
       if (!noDescriptionChange) {
         // only change to true when we need to bind TaskLists the html of description
         this.descriptionChange = true;
+        this.updateTaskHTML();
+
         if (this.description !== '<span></span>') {
           this.previousDescription = this.description;
         }
+
         elementStack.push(this.$el.querySelector('.wiki'));
       }
 
@@ -100,10 +112,8 @@ export default {
       // always reset to false before checking the change
       this.descriptionChange = false;
 
-      const { title, description } = this.data;
-      this.descriptionText = this.data.description_text;
-
-      this.updateTaskHTML();
+      const { title, description } = this.apiData;
+      this.descriptionText = this.apiData.description_text;
 
       const noTitleChange = this.title === title;
       const noDescriptionChange = this.description === description;
diff --git a/spec/features/issues/award_spec.rb b/spec/features/issues/award_spec.rb
index 401e1ea2b893e..08e3f99e29f0d 100644
--- a/spec/features/issues/award_spec.rb
+++ b/spec/features/issues/award_spec.rb
@@ -6,9 +6,12 @@
   let(:issue)     { create(:issue, project: project) }
 
   describe 'logged in' do
+    include WaitForVueResource
+
     before do
       login_as(user)
       visit namespace_project_issue_path(project.namespace, project, issue)
+      wait_for_vue_resource
     end
 
     it 'adds award to issue' do
@@ -38,8 +41,11 @@
   end
 
   describe 'logged out' do
+    include WaitForVueResource
+    
     before do
       visit namespace_project_issue_path(project.namespace, project, issue)
+      wait_for_vue_resource
     end
 
     it 'does not see award menu button' do
diff --git a/spec/support/features/issuable_slash_commands_shared_examples.rb b/spec/support/features/issuable_slash_commands_shared_examples.rb
index 5bbe36d9b7fb8..5e7eca1d987d4 100644
--- a/spec/support/features/issuable_slash_commands_shared_examples.rb
+++ b/spec/support/features/issuable_slash_commands_shared_examples.rb
@@ -3,6 +3,7 @@
 
 shared_examples 'issuable record that supports slash commands in its description and notes' do |issuable_type|
   include SlashCommandsHelpers
+  include WaitForVueResource
 
   let(:master) { create(:user) }
   let(:assignee) { create(:user, username: 'bob') }
@@ -18,6 +19,7 @@
     project.team << [assignee, :developer]
     project.team << [guest, :guest]
     login_with(master)
+    wait_for_vue_resource
   end
 
   after do
-- 
GitLab


From 60b27b1b3e76b4b848b044c1f64a6d27a67b9424 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Mon, 1 May 2017 19:05:07 -0600
Subject: [PATCH 126/363] formatting

---
 .../javascripts/issue_show/issue_title_description.vue       | 5 ++---
 1 file changed, 2 insertions(+), 3 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index dd8794188d783..97f22754393a1 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -38,7 +38,6 @@ export default {
       descriptionText: '',
       descriptionChange: false,
       previousDescription: null,
-      taskStatus: '',
     };
   },
   methods: {
@@ -53,12 +52,12 @@ export default {
 
       if (tasks && !zeroTasks) {
         tasks.innerText = this.apiData.task_status;
-      } else if (this.apiData.task_status.includes('0 of 0')) {
-        $('#task_status_short').remove();
       } else if (!tasks && !zeroTasks) {
         $('.issuable-header').append(`
           <span id="task_status_short" class="hidden-md hidden-lg">${this.apiData.task_status}</span>
         `);
+      } else if (zeroTasks) {
+        $('#task_status_short').remove();
       }
     },
     elementsToVisualize(noTitleChange, noDescriptionChange) {
-- 
GitLab


From 7021e867e41f86343fa8fe9cef75f89d466b2ef6 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Mon, 1 May 2017 19:48:25 -0600
Subject: [PATCH 127/363] use indexOf instead of includes

---
 app/assets/javascripts/issue_show/issue_title_description.vue | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 97f22754393a1..48b649cd56509 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -48,7 +48,7 @@ export default {
     },
     updateTaskHTML() {
       const tasks = document.querySelector('#task_status_short');
-      const zeroTasks = this.apiData.task_status.includes('0 of 0');
+      const zeroTasks = this.apiData.task_status.indexOf('0 of 0') >= 0;
 
       if (tasks && !zeroTasks) {
         tasks.innerText = this.apiData.task_status;
-- 
GitLab


From cfc03dab245843b924c353e50e15e2e59733048e Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Mon, 1 May 2017 20:56:04 -0600
Subject: [PATCH 128/363] fix rubocop offenses

---
 .../issue_show/issue_title_description.vue    |  3 +--
 spec/features/task_lists_spec.rb              | 23 +++++++------------
 2 files changed, 9 insertions(+), 17 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 48b649cd56509..7862923c00442 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -43,7 +43,6 @@ export default {
   methods: {
     renderResponse(res) {
       this.apiData = JSON.parse(res.body);
-      this.issueIID = this.apiData.issue_number;
       this.triggerAnimation();
     },
     updateTaskHTML() {
@@ -90,7 +89,7 @@ export default {
     setTabTitle() {
       const currentTabTitle = document.querySelector('title');
       const currentTabTitleScope = currentTabTitle.innerText.split('·');
-      currentTabTitleScope[0] = `${this.titleText} (#${this.issueIID}) `;
+      currentTabTitleScope[0] = `${this.titleText} (#${this.apiData.issue_number}) `;
       currentTabTitle.innerText = currentTabTitleScope.join('·');
     },
     animate(title, description, elementsToVisualize) {
diff --git a/spec/features/task_lists_spec.rb b/spec/features/task_lists_spec.rb
index 3858f4d89596f..38437b4250aca 100644
--- a/spec/features/task_lists_spec.rb
+++ b/spec/features/task_lists_spec.rb
@@ -62,15 +62,16 @@ def visit_issue(project, issue)
     visit namespace_project_issue_path(project.namespace, project, issue)
   end
 
-  describe 'for Issues' do
+  describe 'for Issues', js: true do
     include WaitForVueResource
 
+    before { wait_for_vue_resource }
+
     describe 'multiple tasks', js: true do
       let!(:issue) { create(:issue, description: markdown, author: user, project: project) }
 
       it 'renders' do
         visit_issue(project, issue)
-        wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 6)
@@ -79,7 +80,6 @@ def visit_issue(project, issue)
 
       it 'contains the required selectors' do
         visit_issue(project, issue)
-        wait_for_vue_resource
 
         container = '.detail-page-description .description.js-task-list-container'
 
@@ -92,14 +92,12 @@ def visit_issue(project, issue)
 
       it 'is only editable by author' do
         visit_issue(project, issue)
-        wait_for_vue_resource
 
         expect(page).to have_selector('.js-task-list-container')
 
         logout(:user)
         login_as(user2)
         visit current_path
-        wait_for_vue_resource
         expect(page).not_to have_selector('.js-task-list-container')
       end
 
@@ -109,12 +107,15 @@ def visit_issue(project, issue)
       end
     end
 
-    describe 'single incomplete task' do
+    describe 'single incomplete task', js: true do
+      include WaitForVueResource
+
       let!(:issue) { create(:issue, description: singleIncompleteMarkdown, author: user, project: project) }
 
+      before { wait_for_vue_resource }
+
       it 'renders' do
         visit_issue(project, issue)
-        wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 1)
@@ -132,7 +133,6 @@ def visit_issue(project, issue)
 
       it 'renders' do
         visit_issue(project, issue)
-        wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 1)
@@ -151,9 +151,6 @@ def visit_issue(project, issue)
       before { visit_issue(project, issue) }
 
       it 'renders' do
-
-        wait_for_vue_resource
-
         expect(page).to have_selector('ul.task-list',      count: 2)
         expect(page).to have_selector('li.task-list-item', count: 7)
         expect(page).to have_selector('ul input[checked]', count: 1)
@@ -163,8 +160,6 @@ def visit_issue(project, issue)
       it 'solves tasks' do
         expect(page).to have_content("2 of 7 tasks completed")
 
-        wait_for_vue_resource
-
         page.find('li.task-list-item', text: 'Task b').find('input').click
         page.find('li.task-list-item ul li.task-list-item', text: 'Task a.2').find('input').click
         page.find('li.task-list-item ol li.task-list-item', text: 'Task 1.1').find('input').click
@@ -173,8 +168,6 @@ def visit_issue(project, issue)
 
         visit_issue(project, issue) # reload to see new system notes
 
-        wait_for_vue_resource
-
         expect(page).to have_content('marked the task Task b as complete')
         expect(page).to have_content('marked the task Task a.2 as complete')
         expect(page).to have_content('marked the task Task 1.1 as complete')
-- 
GitLab


From e2bc67c91fc3ccbcbe5ed5b391d805b5da2f4d02 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Mon, 1 May 2017 20:59:52 -0600
Subject: [PATCH 129/363] fix task_list spec - not ocmpletely but progress

---
 spec/features/task_lists_spec.rb | 10 +++++-----
 1 file changed, 5 insertions(+), 5 deletions(-)

diff --git a/spec/features/task_lists_spec.rb b/spec/features/task_lists_spec.rb
index 38437b4250aca..f55c5aaf646be 100644
--- a/spec/features/task_lists_spec.rb
+++ b/spec/features/task_lists_spec.rb
@@ -62,12 +62,12 @@ def visit_issue(project, issue)
     visit namespace_project_issue_path(project.namespace, project, issue)
   end
 
-  describe 'for Issues', js: true do
-    include WaitForVueResource
-
-    before { wait_for_vue_resource }
-
+  describe 'for Issues', feature: true do
     describe 'multiple tasks', js: true do
+      include WaitForVueResource
+      
+      before { wait_for_vue_resource }
+
       let!(:issue) { create(:issue, description: markdown, author: user, project: project) }
 
       it 'renders' do
-- 
GitLab


From 96ea22208ffced8a8a82d182fe29172d0b60ce29 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Tue, 2 May 2017 08:29:39 +0100
Subject: [PATCH 130/363] Started specs [ci skip]

---
 .../deploy_keys/components/keys_panel.vue     |   2 +-
 .../deploy_keys/components/app_spec.js        | 129 ++++++++++++++++++
 spec/javascripts/fixtures/deploy_keys.rb      |  36 +++++
 3 files changed, 166 insertions(+), 1 deletion(-)
 create mode 100644 spec/javascripts/deploy_keys/components/app_spec.js
 create mode 100644 spec/javascripts/fixtures/deploy_keys.rb

diff --git a/app/assets/javascripts/deploy_keys/components/keys_panel.vue b/app/assets/javascripts/deploy_keys/components/keys_panel.vue
index 2470831eddfef..e5113fe5245ce 100644
--- a/app/assets/javascripts/deploy_keys/components/keys_panel.vue
+++ b/app/assets/javascripts/deploy_keys/components/keys_panel.vue
@@ -28,7 +28,7 @@
 </script>
 
 <template>
-  <div>
+  <div class="deploy-keys-panel">
     <h5>
       {{ title }}
       ({{ keys.length }})
diff --git a/spec/javascripts/deploy_keys/components/app_spec.js b/spec/javascripts/deploy_keys/components/app_spec.js
new file mode 100644
index 0000000000000..e061992b2bda8
--- /dev/null
+++ b/spec/javascripts/deploy_keys/components/app_spec.js
@@ -0,0 +1,129 @@
+import Vue from 'vue';
+import eventHub from '~/deploy_keys/eventhub';
+import deployKeysApp from '~/deploy_keys/components/app.vue';
+
+describe('Deploy keys app component', () => {
+  const data = getJSONFixture('deploy_keys/keys.json');
+  let vm;
+
+  const deployKeysResponse = (request, next) => {
+    next(request.respondWith(JSON.stringify(data), {
+      status: 200,
+    }));
+  };
+
+  beforeEach((done) => {
+    const Component = Vue.extend(deployKeysApp);
+
+    Vue.http.interceptors.push(deployKeysResponse);
+
+    vm = new Component({
+      propsData: {
+        endpoint: '/test',
+      },
+    }).$mount();
+
+    setTimeout(done);
+  });
+
+  afterEach(() => {
+    Vue.http.interceptors = _.without(Vue.http.interceptors, deployKeysResponse);
+  });
+
+  it('renders loading icon', (done) => {
+    vm.store.keys = {};
+    vm.isLoading = false;
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelectorAll('.deploy-keys-panel').length,
+      ).toBe(0);
+
+      expect(
+        vm.$el.querySelector('.fa-spinner'),
+      ).toBeDefined();
+
+      done();
+    });
+  });
+
+  it('renders keys panels', () => {
+    expect(
+      vm.$el.querySelectorAll('.deploy-keys-panel').length,
+    ).toBe(3);
+  });
+
+  it('does not render key panels when keys object is empty', (done) => {
+    vm.store.keys = {};
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelectorAll('.deploy-keys-panel').length,
+      ).toBe(0);
+
+      done();
+    });
+  });
+
+  it('does not render public panel when empty', (done) => {
+    vm.store.keys.public_keys = [];
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelectorAll('.deploy-keys-panel').length,
+      ).toBe(2);
+
+      done();
+    });
+  });
+
+  it('re-fetches deploy keys when enabling a key', (done) => {
+    const key = data.public_keys[0];
+
+    spyOn(vm.service, 'getKeys');
+    spyOn(vm.service, 'enableKey').and.callFake(() => new Promise((resolve) => {
+      resolve();
+
+      setTimeout(() => {
+        expect(vm.service.getKeys).toHaveBeenCalled();
+
+        done();
+      });
+    }));
+
+    eventHub.$emit('enable.key', key);
+
+    expect(vm.service.enableKey).toHaveBeenCalledWith(key.id);
+  });
+
+  it('re-fetches deploy keys when disabling a key', (done) => {
+    const key = data.public_keys[0];
+
+    spyOn(window, 'confirm').and.returnValue(true);
+    spyOn(vm.service, 'getKeys');
+    spyOn(vm.service, 'disableKey').and.callFake(() => new Promise((resolve) => {
+      resolve();
+
+      setTimeout(() => {
+        expect(vm.service.getKeys).toHaveBeenCalled();
+
+        done();
+      });
+    }));
+
+    eventHub.$emit('disable.key', key);
+
+    expect(vm.service.disableKey).toHaveBeenCalledWith(key.id);
+  });
+
+  it('calls disableKey when removing a key', () => {
+    const key = data.public_keys[0];
+
+    spyOn(window, 'confirm').and.returnValue(true);
+    spyOn(vm, 'disableKey');
+
+    eventHub.$emit('remove.key', key);
+
+    expect(vm.disableKey).toHaveBeenCalledWith(key);
+  });
+});
diff --git a/spec/javascripts/fixtures/deploy_keys.rb b/spec/javascripts/fixtures/deploy_keys.rb
new file mode 100644
index 0000000000000..16e598a4b290c
--- /dev/null
+++ b/spec/javascripts/fixtures/deploy_keys.rb
@@ -0,0 +1,36 @@
+require 'spec_helper'
+
+describe Projects::DeployKeysController, '(JavaScript fixtures)', type: :controller do
+  include JavaScriptFixturesHelpers
+
+  let(:admin) { create(:admin) }
+  let(:namespace) { create(:namespace, name: 'frontend-fixtures' )}
+  let(:project) { create(:project_empty_repo, namespace: namespace, path: 'todos-project') }
+  let(:project2) { create(:empty_project, :internal)}
+
+  before(:all) do
+    clean_frontend_fixtures('deploy_keys/')
+  end
+
+  before(:each) do
+    sign_in(admin)
+  end
+
+  render_views
+
+  it 'deploy_keys/keys.json' do |example|
+    create(:deploy_key, public: true)
+    project_key = create(:deploy_key, key: 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQCdMHEHyhRjbhEZVddFn6lTWdgEy5Q6Bz4nwGB76xWZI5YT/1WJOMEW+sL5zYd31kk7sd3FJ5L9ft8zWMWrr/iWXQikC2cqZK24H1xy+ZUmrRuJD4qGAaIVoyyzBL+avL+lF8J5lg6YSw8gwJY/lX64/vnJHUlWw2n5BF8IFOWhiw== dummy@gitlab.com')
+    internal_key = create(:deploy_key, key: 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQDNd/UJWhPrpb+b/G5oL109y57yKuCxE+WUGJGYaj7WQKsYRJmLYh1mgjrl+KVyfsWpq4ylOxIfFSnN9xBBFN8mlb0Fma5DC7YsSsibJr3MZ19ZNBprwNcdogET7aW9I0In7Wu5f2KqI6e5W/spJHCy4JVxzVMUvk6Myab0LnJ2iQ== dummy@gitlab.com')
+    create(:deploy_keys_project, project: project, deploy_key: project_key)
+    create(:deploy_keys_project, project: project2, deploy_key: internal_key)
+
+    get :index,
+      namespace_id: project.namespace.to_param,
+      project_id: project,
+      format: :json
+
+    expect(response).to be_success
+    store_frontend_fixture(response, example.description)
+  end
+end
-- 
GitLab


From 02fc1846832593b2e803a83446690183715f5df1 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Tue, 2 May 2017 10:29:09 +0200
Subject: [PATCH 131/363] Improve specs for jobs API regarding manual actions

---
 spec/requests/api/jobs_spec.rb | 67 +++++++++++++++++++++++++---------
 1 file changed, 50 insertions(+), 17 deletions(-)

diff --git a/spec/requests/api/jobs_spec.rb b/spec/requests/api/jobs_spec.rb
index d8a56c02a638a..e5e5872dc1ff6 100644
--- a/spec/requests/api/jobs_spec.rb
+++ b/spec/requests/api/jobs_spec.rb
@@ -1,16 +1,26 @@
 require 'spec_helper'
 
-describe API::Jobs, api: true do
-  include ApiHelpers
+describe API::Jobs, :api do
+  let!(:project) do
+    create(:project, :repository, public_builds: false)
+  end
+
+  let!(:pipeline) do
+    create(:ci_empty_pipeline, project: project,
+                               sha: project.commit.id,
+                               ref: project.default_branch)
+  end
+
+  let!(:build) { create(:ci_build, pipeline: pipeline) }
 
   let(:user) { create(:user) }
   let(:api_user) { user }
-  let!(:project) { create(:project, :repository, creator: user, public_builds: false) }
-  let!(:developer) { create(:project_member, :developer, user: user, project: project) }
-  let(:reporter) { create(:project_member, :reporter, project: project) }
-  let(:guest) { create(:project_member, :guest, project: project) }
-  let!(:pipeline) { create(:ci_empty_pipeline, project: project, sha: project.commit.id, ref: project.default_branch) }
-  let!(:build) { create(:ci_build, pipeline: pipeline) }
+  let(:reporter) { create(:project_member, :reporter, project: project).user }
+  let(:guest) { create(:project_member, :guest, project: project).user }
+
+  before do
+    project.add_developer(user)
+  end
 
   describe 'GET /projects/:id/jobs' do
     let(:query) { Hash.new }
@@ -213,7 +223,7 @@
   end
 
   describe 'GET /projects/:id/artifacts/:ref_name/download?job=name' do
-    let(:api_user) { reporter.user }
+    let(:api_user) { reporter }
     let(:build) { create(:ci_build, :artifacts, pipeline: pipeline) }
 
     before do
@@ -237,7 +247,7 @@ def get_for_ref(ref = pipeline.ref, job = build.name)
     end
 
     context 'when logging as guest' do
-      let(:api_user) { guest.user }
+      let(:api_user) { guest }
 
       before do
         get_for_ref
@@ -347,7 +357,7 @@ def get_for_ref(ref = pipeline.ref, job = build.name)
       end
 
       context 'user without :update_build permission' do
-        let(:api_user) { reporter.user }
+        let(:api_user) { reporter }
 
         it 'does not cancel job' do
           expect(response).to have_http_status(403)
@@ -381,7 +391,7 @@ def get_for_ref(ref = pipeline.ref, job = build.name)
       end
 
       context 'user without :update_build permission' do
-        let(:api_user) { reporter.user }
+        let(:api_user) { reporter }
 
         it 'does not retry job' do
           expect(response).to have_http_status(403)
@@ -457,16 +467,39 @@ def get_for_ref(ref = pipeline.ref, job = build.name)
 
   describe 'POST /projects/:id/jobs/:job_id/play' do
     before do
-      post api("/projects/#{project.id}/jobs/#{build.id}/play", user)
+      post api("/projects/#{project.id}/jobs/#{build.id}/play", api_user)
     end
 
     context 'on an playable job' do
       let(:build) { create(:ci_build, :manual, project: project, pipeline: pipeline) }
 
-      it 'plays the job' do
-        expect(response).to have_http_status(200)
-        expect(json_response['user']['id']).to eq(user.id)
-        expect(json_response['id']).to eq(build.id)
+      context 'when user is authorized to trigger a manual action' do
+        it 'plays the job' do
+          expect(response).to have_http_status(200)
+          expect(json_response['user']['id']).to eq(user.id)
+          expect(json_response['id']).to eq(build.id)
+          expect(build.reload).to be_pending
+        end
+      end
+
+      context 'when user is not authorized to trigger a manual action' do
+        context 'when user does not have access to the project' do
+          let(:api_user) { create(:user) }
+
+          it 'does not trigger a manual action' do
+            expect(build.reload).to be_manual
+            expect(response).to have_http_status(404)
+          end
+        end
+
+        context 'when user is not allowed to trigger the manual action' do
+          let(:api_user) { reporter }
+
+          it 'does not trigger a manual action' do
+            expect(build.reload).to be_manual
+            expect(response).to have_http_status(403)
+          end
+        end
       end
     end
 
-- 
GitLab


From f5441c93203cd5c8ccd208c00bee59432facae7f Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Tue, 2 May 2017 10:35:51 +0200
Subject: [PATCH 132/363] Document protected manual actions feature

---
 doc/ci/yaml/README.md | 6 +++++-
 1 file changed, 5 insertions(+), 1 deletion(-)

diff --git a/doc/ci/yaml/README.md b/doc/ci/yaml/README.md
index ad3ebd144dfe4..0cab3270d94dc 100644
--- a/doc/ci/yaml/README.md
+++ b/doc/ci/yaml/README.md
@@ -553,6 +553,8 @@ The above script will:
 #### Manual actions
 
 > Introduced in GitLab 8.10.
+> Blocking manual actions were introduced in GitLab 9.0
+> Protected actions were introduced in GitLab 9.2
 
 Manual actions are a special type of job that are not executed automatically;
 they need to be explicitly started by a user. Manual actions can be started
@@ -578,7 +580,9 @@ Optional manual actions have `allow_failure: true` set by default.
 
 **Statuses of optional actions do not contribute to overall pipeline status.**
 
-> Blocking manual actions were introduced in GitLab 9.0
+**Manual actions do inherit permissions of protected branches. In other words,
+in order to trigger a manual action assigned to a branch that the pipeline is
+running for, user needs to have ability to push to this branch.**
 
 ### environment
 
-- 
GitLab


From c9def85844531ffdd2984707a1bc8cbca18f6742 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Tue, 2 May 2017 10:37:50 +0200
Subject: [PATCH 133/363] Add Changelog entry for protected manual actions

---
 ...ature-gb-manual-actions-protected-branches-permissions.yml | 4 ++++
 1 file changed, 4 insertions(+)
 create mode 100644 changelogs/unreleased/feature-gb-manual-actions-protected-branches-permissions.yml

diff --git a/changelogs/unreleased/feature-gb-manual-actions-protected-branches-permissions.yml b/changelogs/unreleased/feature-gb-manual-actions-protected-branches-permissions.yml
new file mode 100644
index 0000000000000..6f8e80e7d648a
--- /dev/null
+++ b/changelogs/unreleased/feature-gb-manual-actions-protected-branches-permissions.yml
@@ -0,0 +1,4 @@
+---
+title: Implement protected manual actions
+merge_request: 10494
+author:
-- 
GitLab


From 4ddce55315167b362056616151244f137c81945f Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Tue, 2 May 2017 11:27:10 +0200
Subject: [PATCH 134/363] Fix environment policy class name in specs

---
 spec/policies/environment_policy_spec.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/policies/environment_policy_spec.rb b/spec/policies/environment_policy_spec.rb
index f43caffa94633..0e15beaa5e8c1 100644
--- a/spec/policies/environment_policy_spec.rb
+++ b/spec/policies/environment_policy_spec.rb
@@ -1,6 +1,6 @@
 require 'spec_helper'
 
-describe Ci::EnvironmentPolicy do
+describe EnvironmentPolicy do
   let(:user) { create(:user) }
   let(:project) { create(:project) }
 
-- 
GitLab


From ccb8e6936c9e85743e4cb42c3580b297efa889ee Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=A9my=20Coutable?= <remy@rymai.me>
Date: Thu, 27 Apr 2017 18:55:15 +0200
Subject: [PATCH 135/363] Document labels in CONTRIBUTING.md
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Signed-off-by: Rémy Coutable <remy@rymai.me>
---
 CONTRIBUTING.md | 180 +++++++++++++++++++++++++++++++++++++++---------
 PROCESS.md      |  67 ++++++++++--------
 2 files changed, 183 insertions(+), 64 deletions(-)

diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index 73c8a77364b67..c8b4d0250ff77 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -13,27 +13,30 @@ _This notice should stay as the first item in the CONTRIBUTING.MD file._
 <!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
 **Table of Contents**  *generated with [DocToc](https://github.com/thlorenz/doctoc)*
 
-- [Contributor license agreement](#contributor-license-agreement)
 - [Contribute to GitLab](#contribute-to-gitlab)
 - [Security vulnerability disclosure](#security-vulnerability-disclosure)
 - [Closing policy for issues and merge requests](#closing-policy-for-issues-and-merge-requests)
 - [Helping others](#helping-others)
 - [I want to contribute!](#i-want-to-contribute)
-- [Implement design & UI elements](#implement-design-ui-elements)
-- [Release retrospective and kickoff](#release-retrospective-and-kickoff)
-    - [Retrospective](#retrospective)
-    - [Kickoff](#kickoff)
+- [Workflow labels](#workflow-labels)
+  - [Type labels (~"feature proposal", ~bug, ~customer, etc.)](#type-labels-feature-proposal-bug-customer-etc)
+  - [Subject labels (~wiki, ~"container registry", ~ldap, ~api, etc.)](#subject-labels-wiki-container-registry-ldap-api-etc)
+  - [Team labels (~CI, ~Discussion, ~Edge, ~Frontend, ~Platform, etc.)](#team-labels-ci-discussion-edge-frontend-platform-etc)
+  - [Priority labels (`Deliverable` and `Stretch`)](#priority-labels-deliverable-and-stretch)
+  - [Label for community contributors (~"Accepting Merge Requests")](#label-for-community-contributors-accepting-merge-requests)
+- [Implement design & UI elements](#implement-design-&-ui-elements)
 - [Issue tracker](#issue-tracker)
-    - [Feature proposals](#feature-proposals)
-    - [Issue tracker guidelines](#issue-tracker-guidelines)
-    - [Issue weight](#issue-weight)
-    - [Regression issues](#regression-issues)
-    - [Technical debt](#technical-debt)
-    - [Stewardship](#stewardship)
+  - [Issue triaging](#issue-triaging)
+  - [Feature proposals](#feature-proposals)
+  - [Issue tracker guidelines](#issue-tracker-guidelines)
+  - [Issue weight](#issue-weight)
+  - [Regression issues](#regression-issues)
+  - [Technical debt](#technical-debt)
+  - [Stewardship](#stewardship)
 - [Merge requests](#merge-requests)
-    - [Merge request guidelines](#merge-request-guidelines)
-    - [Contribution acceptance criteria](#contribution-acceptance-criteria)
-- [Changes for Stable Releases](#changes-for-stable-releases)
+  - [Merge request guidelines](#merge-request-guidelines)
+  - [Getting your merge request reviewed, approved, and merged](#getting-your-merge-request-reviewed-approved-and-merged)
+  - [Contribution acceptance criteria](#contribution-acceptance-criteria)
 - [Definition of done](#definition-of-done)
 - [Style guides](#style-guides)
 - [Code of conduct](#code-of-conduct)
@@ -103,34 +106,125 @@ contributing to GitLab.
 
 ## Workflow labels
 
-Labelling issues is described in the [GitLab Inc engineering workflow].
+To allow for asynchronous issue handling, we use [milestones][milestones-page]
+and [labels][labels-page]. Leads and product managers handle most of the
+scheduling into milestones. Labelling is a task for everyone.
 
-## Implement design & UI elements
+Most issues will have labels for at least one of the following:
 
-Please see the [UX Guide for GitLab].
+- Type: ~"feature proposal", ~bug, ~customer, etc.
+- Subject: ~wiki, ~"container registry", ~ldap, ~api, etc.
+- Team: ~CI, ~Discussion, ~Edge, ~Frontend, ~Platform, etc.
+- Priority: ~Deliverable, ~Stretch
+
+All labels, their meaning and priority are defined on the
+[labels page][labels-page].
+
+If you come across an issue that has none of these, and you're allowed to set
+labels, you can _always_ add the team and type, and often also the subject.
+
+[milestones-page]: https://gitlab.com/gitlab-org/gitlab-ce/milestones
+[labels-page]: https://gitlab.com/gitlab-org/gitlab-ce/labels
+
+### Type labels (~"feature proposal", ~bug, ~customer, etc.)
+
+Type labels are very important. They define what kind of issue this is. Every
+issue should have one or more.
+
+Examples of type labels are ~"feature proposal", ~bug, ~customer, ~security,
+and ~"direction".
+
+A number of type labels have a priority assigned to them, which automatically
+makes them float to the top, depending on their importance.
+
+Type labels are always lowercase, but can have any color, besides blue (which is
+already reserved for subject labels).
+
+The descriptions on the [labels page][labels-page] explain what falls under each type label.
+
+### Subject labels (~wiki, ~"container registry", ~ldap, ~api, etc.)
+
+Subject labels are labels that define what area or feature of GitLab this issue
+hits. They are not always necessary, but very convenient.
+
+If you are an expert in a particular area, it makes it easier to find issues to
+work on. You can also subscribe to those labels to receive an email each time an
+issue is labelled with a subject label corresponding to your expertise.
+
+Examples of subject labels are ~wiki, ~"container registry", ~ldap, ~api,
+~issues, ~"merge requests", ~labels, and ~"container registry".
+
+Subject labels are always colored blue and all-lowercase.
+
+### Team labels (~CI, ~Discussion, ~Edge, ~Frontend, ~Platform, etc.)
+
+Team labels specify what team is responsible for this issue.
+Assigning a team label makes sure issues get the attention of the appropriate
+people.
 
-## Release retrospective and kickoff
+The current team labels are ~Build, ~CI, ~Discussion, ~Documentation, ~Edge,
+~Frontend, ~Gitaly, ~Platform, ~Prometheus, ~Release, and ~"UX".
 
-### Retrospective
+The descriptions on the [labels page][labels-page] explain what falls under the
+responsibility of each team.
 
-After each release, we have a retrospective call where we discuss what went well,
-what went wrong, and what we can improve for the next release. The
-[retrospective notes] are public and you are invited to comment on them.
-If you're interested, you can even join the
-[retrospective call][retro-kickoff-call], on the first working day after the
-22nd at 6pm CET / 9am PST.
+Team labels are always colored aqua, and are capitalized so that they show up as
+the first label for any issue.
 
-### Kickoff
+### Priority labels (~Deliverable and ~Stretch)
 
-Before working on the next release, we have a
-kickoff call to explain what we expect to ship in the next release. The
-[kickoff notes] are public and you are invited to comment on them.
-If you're interested, you can even join the [kickoff call][retro-kickoff-call],
-on the first working day after the 7th at 6pm CET / 9am PST..
+Priority labels help us clearly communicate expectations of the work for the
+release. There are two levels of priority labels:
 
-[retrospective notes]: https://docs.google.com/document/d/1nEkM_7Dj4bT21GJy0Ut3By76FZqCfLBmFQNVThmW2TY/edit?usp=sharing
-[kickoff notes]: https://docs.google.com/document/d/1ElPkZ90A8ey_iOkTvUs_ByMlwKK6NAB2VOK5835wYK0/edit?usp=sharing
-[retro-kickoff-call]: https://gitlab.zoom.us/j/918821206
+- ~Deliverable: Issues that are expected to be delivered in the current
+  milestone.
+- ~Stretch: Issues that are a stretch goal for delivering in the current
+  milestone. If these issues are not done in the current release, they will
+  strongly be considered for the next release.
+
+### Label for community contributors (~"Accepting Merge Requests")
+
+Issues that are beneficial to our users, 'nice to haves', that we currently do
+not have the capacity for or want to give the priority to, are labeled as
+~"Accepting Merge Requests", so the community can make a contribution.
+
+Community contributors can submit merge requests for any issue they want, but
+the ~"Accepting Merge Requests" label has a special meaning. It points to
+changes that:
+
+1. We already agreed on,
+1. Are well-defined,
+1. Are likely to get accepted by a maintainer.
+
+We want to avoid a situation when a contributor picks an
+~"Accepting Merge Requests" issue and then their merge request gets closed,
+because we realize that it does not fit our vision, or we want to solve it in a
+different way.
+
+We add the ~"Accepting Merge Requests" label to:
+
+- Low priority ~bug issues (i.e. we do not add it to the bugs that we want to
+solve in the ~"Next Patch Release")
+- Small ~"feature proposal" that do not need ~UX / ~"Product work", or for which
+the ~UX / ~"Product work" is already done
+- Small ~"technical debt" issues
+
+After adding the ~"Accepting Merge Requests" label, we try to estimate the
+[weight](#issue-weight) of the issue. We use issue weight to let contributors
+know how difficult the issue is. Additionally:
+
+- We advertise [~"Accepting Merge Requests" issues with weight < 5][up-for-grabs]
+  as suitable for people that have never contributed to GitLab before on the
+  [Up For Grabs campaign](http://up-for-grabs.net)
+- We encourage people that have never contributed to any open source project to
+  look for [~"Accepting Merge Requests" issues with a weight of 1][firt-timers]
+
+[up-for-grabs]: https://gitlab.com/gitlab-org/gitlab-ce/issues?label_name=Accepting+Merge+Requests&scope=all&sort=weight_asc&state=opened
+[firt-timers]: https://gitlab.com/gitlab-org/gitlab-ce/issues?label_name%5B%5D=Accepting+Merge+Requests&scope=all&sort=upvotes_desc&state=opened&weight=1
+
+## Implement design & UI elements
+
+Please see the [UX Guide for GitLab].
 
 ## Issue tracker
 
@@ -154,6 +248,24 @@ If it happens that you know the solution to an existing bug, please first
 open the issue in order to keep track of it and then open the relevant merge
 request that potentially fixes it.
 
+### Issue triaging
+
+Our issue triage policies are [described in our handbook]. You are very welcome
+to help the GitLab team triage issues. We also organize [issue bash events] once
+every quarter.
+
+The most important thing is making sure valid issues receive feedback from the
+development team. Therefore the priority is mentioning developers that can help
+on those issues. Please select someone with relevant experience from
+[GitLab team][team]. If there is nobody mentioned with that expertise
+look in the commit history for the affected files to find someone. Avoid
+mentioning the lead developer, this is the person that is least likely to give a
+timely response. If the involvement of the lead developer is needed the other
+core team members will mention this person.
+
+[described in our handbook]: https://about.gitlab.com/handbook/engineering/issues/issue-triage-policies/
+[issue bash events]: https://gitlab.com/gitlab-org/gitlab-ce/issues/17815
+
 ### Feature proposals
 
 To create a feature proposal for CE, open an issue on the
diff --git a/PROCESS.md b/PROCESS.md
index fac3c22e09fc8..735120a369ebe 100644
--- a/PROCESS.md
+++ b/PROCESS.md
@@ -1,35 +1,48 @@
-# GitLab Contributing Process
+<!-- START doctoc generated TOC please keep comment here to allow auto update -->
+<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
+**Table of Contents**  *generated with [DocToc](https://github.com/thlorenz/doctoc)*
+
+- [GitLab Contributing Process](#gitlab-contributing-process)
+  - [Purpose of describing the contributing process](#purpose-of-describing-the-contributing-process)
+  - [Common actions](#common-actions)
+    - [Merge request coaching](#merge-request-coaching)
+  - [Workflow labels](#workflow-labels)
+  - [Assigning issues](#assigning-issues)
+  - [Be kind](#be-kind)
+  - [Feature freeze on the 7th for the release on the 22nd](#feature-freeze-on-the-7th-for-the-release-on-the-22nd)
+    - [Between the 1st and the 7th](#between-the-1st-and-the-7th)
+    - [On the 7th](#on-the-7th)
+    - [After the 7th](#after-the-7th)
+  - [Copy & paste responses](#copy-&-paste-responses)
+    - [Improperly formatted issue](#improperly-formatted-issue)
+    - [Issue report for old version](#issue-report-for-old-version)
+    - [Support requests and configuration questions](#support-requests-and-configuration-questions)
+    - [Code format](#code-format)
+    - [Issue fixed in newer version](#issue-fixed-in-newer-version)
+    - [Improperly formatted merge request](#improperly-formatted-merge-request)
+    - [Inactivity close of an issue](#inactivity-close-of-an-issue)
+    - [Inactivity close of a merge request](#inactivity-close-of-a-merge-request)
+    - [Accepting merge requests](#accepting-merge-requests)
+    - [Only accepting merge requests with green tests](#only-accepting-merge-requests-with-green-tests)
+    - [Closing down the issue tracker on GitHub](#closing-down-the-issue-tracker-on-github)
+
+<!-- END doctoc generated TOC please keep comment here to allow auto update -->
+
+# GitLab Core Team Contributing Process
 
 ## Purpose of describing the contributing process
 
-Below we describe the contributing process to GitLab for two reasons. So that
-contributors know what to expect from maintainers (possible responses, friendly
-treatment, etc.). And so that maintainers know what to expect from contributors
-(use the latest version, ensure that the issue is addressed, friendly treatment,
-etc.).
+Below we describe the contributing process to GitLab for two reasons:
+
+1. Contributors know what to expect from maintainers (possible responses, friendly
+  treatment, etc.)
+1. Maintainers know what to expect from contributors (use the latest version,
+  ensure that the issue is addressed, friendly treatment, etc.).
 
 - [GitLab Inc engineers should refer to the engineering workflow document](https://about.gitlab.com/handbook/engineering/workflow/)
 
 ## Common actions
 
-### Issue triaging
-
-Our issue triage policies are [described in our handbook]. You are very welcome
-to help the GitLab team triage issues. We also organize [issue bash events] once
-every quarter.
-
-The most important thing is making sure valid issues receive feedback from the
-development team. Therefore the priority is mentioning developers that can help
-on those issues. Please select someone with relevant experience from
-[GitLab team][team]. If there is nobody mentioned with that expertise
-look in the commit history for the affected files to find someone. Avoid
-mentioning the lead developer, this is the person that is least likely to give a
-timely response. If the involvement of the lead developer is needed the other
-core team members will mention this person.
-
-[described in our handbook]: https://about.gitlab.com/handbook/engineering/issues/issue-triage-policies/
-[issue bash events]: https://gitlab.com/gitlab-org/gitlab-ce/issues/17815
-
 ### Merge request coaching
 
 Several people from the [GitLab team][team] are helping community members to get
@@ -37,12 +50,6 @@ their contributions accepted by meeting our [Definition of done][done].
 
 What you can expect from them is described at https://about.gitlab.com/jobs/merge-request-coach/.
 
-## Workflow labels
-
-Labelling issues is described in the [GitLab Inc engineering workflow].
-
-[GitLab Inc engineering workflow]: https://about.gitlab.com/handbook/engineering/workflow/#labelling-issues
-
 ## Assigning issues
 
 If an issue is complex and needs the attention of a specific person, assignment is a good option but assigning issues might discourage other people from contributing to that issue. We need all the contributions we can get so this should never be discouraged. Also, an assigned person might not have time for a few weeks, so others should feel free to takeover.
-- 
GitLab


From eb45582f55cae42acc41fa228f4797b4067c01dc Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Tue, 2 May 2017 12:25:30 +0200
Subject: [PATCH 136/363] Fix builds controller spec related to protected
 actions

---
 spec/controllers/projects/builds_controller_spec.rb | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/spec/controllers/projects/builds_controller_spec.rb b/spec/controllers/projects/builds_controller_spec.rb
index 22193eac6722f..3bf03c88ff55a 100644
--- a/spec/controllers/projects/builds_controller_spec.rb
+++ b/spec/controllers/projects/builds_controller_spec.rb
@@ -260,6 +260,8 @@ def post_retry
   end
 
   describe 'POST play' do
+    let(:project) { create(:project, :public) }
+
     before do
       project.add_developer(user)
       sign_in(user)
-- 
GitLab


From 4134d700623404948f163349882caf4a6d940cf3 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Tue, 2 May 2017 12:03:58 +0100
Subject: [PATCH 137/363] Component specs

---
 .../deploy_keys/components/key.vue            |  5 -
 .../deploy_keys/components/keys_panel.vue     |  2 +-
 .../deploy_keys/components/action_btn_spec.js | 70 ++++++++++++++
 .../deploy_keys/components/app_spec.js        |  4 +
 .../deploy_keys/components/key_spec.js        | 92 +++++++++++++++++++
 .../deploy_keys/components/keys_panel_spec.js | 70 ++++++++++++++
 6 files changed, 237 insertions(+), 6 deletions(-)
 create mode 100644 spec/javascripts/deploy_keys/components/action_btn_spec.js
 create mode 100644 spec/javascripts/deploy_keys/components/key_spec.js
 create mode 100644 spec/javascripts/deploy_keys/components/keys_panel_spec.js

diff --git a/app/assets/javascripts/deploy_keys/components/key.vue b/app/assets/javascripts/deploy_keys/components/key.vue
index 1b47cceba78a1..0a06a481b9686 100644
--- a/app/assets/javascripts/deploy_keys/components/key.vue
+++ b/app/assets/javascripts/deploy_keys/components/key.vue
@@ -7,11 +7,6 @@
         type: Object,
         required: true,
       },
-      enabled: {
-        type: Boolean,
-        required: false,
-        default: true,
-      },
       store: {
         type: Object,
         required: true,
diff --git a/app/assets/javascripts/deploy_keys/components/keys_panel.vue b/app/assets/javascripts/deploy_keys/components/keys_panel.vue
index e5113fe5245ce..eccc470578b6e 100644
--- a/app/assets/javascripts/deploy_keys/components/keys_panel.vue
+++ b/app/assets/javascripts/deploy_keys/components/keys_panel.vue
@@ -37,7 +37,7 @@
       v-if="keys.length">
       <li
         v-for="deployKey in keys"
-        key="deployKey.id">
+        :key="deployKey.id">
         <key
           :deploy-key="deployKey"
           :store="store" />
diff --git a/spec/javascripts/deploy_keys/components/action_btn_spec.js b/spec/javascripts/deploy_keys/components/action_btn_spec.js
new file mode 100644
index 0000000000000..5b93fbc5575d7
--- /dev/null
+++ b/spec/javascripts/deploy_keys/components/action_btn_spec.js
@@ -0,0 +1,70 @@
+import Vue from 'vue';
+import eventHub from '~/deploy_keys/eventhub';
+import actionBtn from '~/deploy_keys/components/action_btn.vue';
+
+describe('Deploy keys action btn', () => {
+  const data = getJSONFixture('deploy_keys/keys.json');
+  const deployKey = data.enabled_keys[0];
+  let vm;
+
+  beforeEach((done) => {
+    const ActionBtnComponent = Vue.extend(actionBtn);
+
+    vm = new ActionBtnComponent({
+      propsData: {
+        deployKey,
+        type: 'enable',
+      },
+    }).$mount();
+
+    setTimeout(done);
+  });
+
+  it('renders the type as uppercase', () => {
+    expect(
+      vm.$el.textContent.trim(),
+    ).toBe('Enable');
+  });
+
+  it('sends eventHub event with btn type', (done) => {
+    spyOn(eventHub, '$emit');
+
+    vm.$el.click();
+
+    setTimeout(() => {
+      expect(
+        eventHub.$emit,
+      ).toHaveBeenCalledWith('enable.key', deployKey);
+
+      done();
+    });
+  });
+
+  it('shows loading spinner after click', (done) => {
+    vm.$el.click();
+
+    setTimeout(() => {
+      expect(
+        vm.$el.querySelector('.fa'),
+      ).toBeDefined();
+
+      done();
+    });
+  });
+
+  it('disables button after click', (done) => {
+    vm.$el.click();
+
+    setTimeout(() => {
+      expect(
+        vm.$el.classList.contains('disabled'),
+      ).toBeTruthy();
+
+      expect(
+        vm.$el.getAttribute('disabled'),
+      ).toBe('disabled');
+
+      done();
+    });
+  });
+});
diff --git a/spec/javascripts/deploy_keys/components/app_spec.js b/spec/javascripts/deploy_keys/components/app_spec.js
index e061992b2bda8..43b8f71850883 100644
--- a/spec/javascripts/deploy_keys/components/app_spec.js
+++ b/spec/javascripts/deploy_keys/components/app_spec.js
@@ -126,4 +126,8 @@ describe('Deploy keys app component', () => {
 
     expect(vm.disableKey).toHaveBeenCalledWith(key);
   });
+
+  it('hasKeys returns true when there are keys', () => {
+    expect(vm.hasKeys).toEqual(3);
+  });
 });
diff --git a/spec/javascripts/deploy_keys/components/key_spec.js b/spec/javascripts/deploy_keys/components/key_spec.js
new file mode 100644
index 0000000000000..793ab8c451d85
--- /dev/null
+++ b/spec/javascripts/deploy_keys/components/key_spec.js
@@ -0,0 +1,92 @@
+import Vue from 'vue';
+import DeployKeysStore from '~/deploy_keys/store';
+import key from '~/deploy_keys/components/key.vue';
+
+describe('Deploy keys key', () => {
+  let vm;
+  const KeyComponent = Vue.extend(key);
+  const data = getJSONFixture('deploy_keys/keys.json');
+  const createComponent = (deployKey) => {
+    const store = new DeployKeysStore();
+    store.keys = data;
+
+    vm = new KeyComponent({
+      propsData: {
+        deployKey,
+        store,
+      },
+    }).$mount();
+  };
+
+  describe('enabled key', () => {
+    const deployKey = data.enabled_keys[0];
+
+    beforeEach((done) => {
+      createComponent(deployKey);
+
+      setTimeout(done);
+    });
+
+    it('renders the keys title', () => {
+      expect(
+        vm.$el.querySelector('.title').textContent.trim(),
+      ).toContain('My title');
+    });
+
+    it('renders human friendly formatted created date', () => {
+      expect(
+        vm.$el.querySelector('.key-created-at').textContent.trim(),
+      ).toBe(`created ${gl.utils.getTimeago().format(deployKey.created_at)}`);
+    });
+
+    it('shows remove button', () => {
+      expect(
+        vm.$el.querySelector('.btn').textContent.trim(),
+      ).toBe('Remove');
+    });
+
+    it('shows write access text when key has write access', (done) => {
+      vm.deployKey.can_push = true;
+
+      Vue.nextTick(() => {
+        expect(
+          vm.$el.querySelector('.write-access-allowed'),
+        ).not.toBeNull();
+
+        expect(
+          vm.$el.querySelector('.write-access-allowed').textContent.trim(),
+        ).toBe('Write access allowed');
+
+        done();
+      });
+    });
+  });
+
+  describe('public keys', () => {
+    const deployKey = data.public_keys[0];
+
+    beforeEach((done) => {
+      createComponent(deployKey);
+
+      setTimeout(done);
+    });
+
+    it('shows enable button', () => {
+      expect(
+        vm.$el.querySelector('.btn').textContent.trim(),
+      ).toBe('Enable');
+    });
+
+    it('shows disable button when key is enabled', (done) => {
+      vm.store.keys.enabled_keys.push(deployKey);
+
+      Vue.nextTick(() => {
+        expect(
+          vm.$el.querySelector('.btn').textContent.trim(),
+        ).toBe('Disable');
+
+        done();
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/deploy_keys/components/keys_panel_spec.js b/spec/javascripts/deploy_keys/components/keys_panel_spec.js
new file mode 100644
index 0000000000000..416b80525fc2d
--- /dev/null
+++ b/spec/javascripts/deploy_keys/components/keys_panel_spec.js
@@ -0,0 +1,70 @@
+import Vue from 'vue';
+import DeployKeysStore from '~/deploy_keys/store';
+import deployKeysPanel from '~/deploy_keys/components/keys_panel.vue';
+
+describe('Deploy keys panel', () => {
+  const data = getJSONFixture('deploy_keys/keys.json');
+  let vm;
+
+  beforeEach((done) => {
+    const DeployKeysPanelComponent = Vue.extend(deployKeysPanel);
+    const store = new DeployKeysStore();
+    store.keys = data;
+
+    vm = new DeployKeysPanelComponent({
+      propsData: {
+        title: 'test',
+        keys: data.enabled_keys,
+        showHelpBox: true,
+        store,
+      },
+    }).$mount();
+
+    setTimeout(done);
+  });
+
+  it('renders the title with keys count', () => {
+    expect(
+      vm.$el.querySelector('h5').textContent.trim(),
+    ).toContain('test');
+
+    expect(
+      vm.$el.querySelector('h5').textContent.trim(),
+    ).toContain('(1)');
+  });
+
+  it('renders list of keys', () => {
+    expect(
+      vm.$el.querySelectorAll('li').length,
+    ).toBe(1);
+  });
+
+  it('renders help box if keys are empty', (done) => {
+    vm.keys = [];
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelector('.settings-message'),
+      ).toBeDefined();
+
+      expect(
+        vm.$el.querySelector('.settings-message').textContent.trim(),
+      ).toBe('No deploy keys found. Create one with the form above.');
+
+      done();
+    });
+  });
+
+  it('does not render help box if keys are empty & showHelpBox is false', (done) => {
+    vm.keys = [];
+    vm.showHelpBox = false;
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelector('.settings-message'),
+      ).toBeNull();
+
+      done();
+    });
+  });
+});
-- 
GitLab


From 48bb8921240387ce7d8f9401c98e13a9b3d0100f Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Tue, 2 May 2017 13:21:06 +0200
Subject: [PATCH 138/363] Hide environment external URL button if not defined

---
 .../projects/environments/terminal.html.haml  |  5 +--
 .../environments/terminal.html.haml_spec.rb   | 32 +++++++++++++++++++
 2 files changed, 35 insertions(+), 2 deletions(-)
 create mode 100644 spec/views/projects/environments/terminal.html.haml_spec.rb

diff --git a/app/views/projects/environments/terminal.html.haml b/app/views/projects/environments/terminal.html.haml
index c8363087d6aa1..4c4aa0baff3fa 100644
--- a/app/views/projects/environments/terminal.html.haml
+++ b/app/views/projects/environments/terminal.html.haml
@@ -16,8 +16,9 @@
 
       .col-sm-6
         .nav-controls
-          = link_to @environment.external_url, class: 'btn btn-default', target: '_blank', rel: 'noopener noreferrer nofollow' do
-            = icon('external-link')
+          - if @environment.external_url.present?
+            = link_to @environment.external_url, class: 'btn btn-default', target: '_blank', rel: 'noopener noreferrer nofollow' do
+              = icon('external-link')
           = render 'projects/deployments/actions', deployment: @environment.last_deployment
 
 .terminal-container{ class: container_class }
diff --git a/spec/views/projects/environments/terminal.html.haml_spec.rb b/spec/views/projects/environments/terminal.html.haml_spec.rb
new file mode 100644
index 0000000000000..d2e4722522651
--- /dev/null
+++ b/spec/views/projects/environments/terminal.html.haml_spec.rb
@@ -0,0 +1,32 @@
+require 'spec_helper'
+
+describe 'projects/environments/terminal' do
+  let!(:environment) { create(:environment, :with_review_app) }
+
+  before do
+    assign(:environment, environment)
+    assign(:project, environment.project)
+
+    allow(view).to receive(:can?).and_return(true)
+  end
+
+  context 'when environment has external URL' do
+    it 'shows external URL button' do
+      environment.update_attribute(:external_url, 'https://gitlab.com')
+
+      render
+
+      expect(rendered).to have_link(nil, href: 'https://gitlab.com')
+    end
+  end
+
+  context 'when environment does not have external URL' do
+    it 'shows external URL button' do
+      environment.update_attribute(:external_url, nil)
+
+      render
+
+      expect(rendered).not_to have_link(nil, href: 'https://gitlab.com')
+    end
+  end
+end
-- 
GitLab


From 3aa92cb5cbe8456c845e16d14489591dd81dbcb3 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Tue, 2 May 2017 13:26:53 +0200
Subject: [PATCH 139/363] Add changelog entry for external env URL btn fix

---
 ...gb-hide-environment-external-url-btn-when-not-provided.yml | 4 ++++
 1 file changed, 4 insertions(+)
 create mode 100644 changelogs/unreleased/fix-gb-hide-environment-external-url-btn-when-not-provided.yml

diff --git a/changelogs/unreleased/fix-gb-hide-environment-external-url-btn-when-not-provided.yml b/changelogs/unreleased/fix-gb-hide-environment-external-url-btn-when-not-provided.yml
new file mode 100644
index 0000000000000..66158e337fdf0
--- /dev/null
+++ b/changelogs/unreleased/fix-gb-hide-environment-external-url-btn-when-not-provided.yml
@@ -0,0 +1,4 @@
+---
+title: Hide external environment URL button on terminal page if URL is not defined
+merge_request: 11029
+author:
-- 
GitLab


From 702b291f81b576c08f8e194d87fb779b4ce0fd6c Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Tue, 2 May 2017 14:55:32 +0200
Subject: [PATCH 140/363] remove gl_project_id for I/E version update

---
 app/models/commit_status.rb                | 6 ------
 lib/gitlab/import_export/import_export.yml | 1 -
 2 files changed, 7 deletions(-)

diff --git a/app/models/commit_status.rb b/app/models/commit_status.rb
index 2c4033146bff9..75d04fd2b08c3 100644
--- a/app/models/commit_status.rb
+++ b/app/models/commit_status.rb
@@ -142,12 +142,6 @@ def auto_canceled?
     canceled? && auto_canceled_by_id?
   end
 
-  # Added in 9.0 to keep backward compatibility for projects exported in 8.17
-  # and prior.
-  def gl_project_id
-    'dummy'
-  end
-
   def detailed_status(current_user)
     Gitlab::Ci::Status::Factory
       .new(self, current_user)
diff --git a/lib/gitlab/import_export/import_export.yml b/lib/gitlab/import_export/import_export.yml
index 899a656776812..70d2ef7103cf2 100644
--- a/lib/gitlab/import_export/import_export.yml
+++ b/lib/gitlab/import_export/import_export.yml
@@ -87,7 +87,6 @@ methods:
     - :type
   statuses:
     - :type
-    - :gl_project_id
   services:
     - :type
   merge_request_diff:
-- 
GitLab


From 5ed2465c1af700bd32242d0f1f04a1968ee65039 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 2 May 2017 17:00:57 +0100
Subject: [PATCH 141/363] Fixed tests

---
 spec/features/projects/merge_request_button_spec.rb | 12 ++++++------
 1 file changed, 6 insertions(+), 6 deletions(-)

diff --git a/spec/features/projects/merge_request_button_spec.rb b/spec/features/projects/merge_request_button_spec.rb
index 1a4d613ef9417..1370ab1c5217b 100644
--- a/spec/features/projects/merge_request_button_spec.rb
+++ b/spec/features/projects/merge_request_button_spec.rb
@@ -25,7 +25,7 @@
       it 'shows Create merge request button' do
         href = new_namespace_project_merge_request_path(project.namespace,
                                                         project,
-                                                        merge_request: { source_branch: "'test'",
+                                                        merge_request: { source_branch: 'feature',
                                                                          target_branch: 'master' })
 
         visit url
@@ -69,7 +69,7 @@
         it 'shows Create merge request button' do
           href = new_namespace_project_merge_request_path(forked_project.namespace,
                                                           forked_project,
-                                                          merge_request: { source_branch: "'test'",
+                                                          merge_request: { source_branch: 'feature',
                                                                            target_branch: 'master' })
 
           visit fork_url
@@ -93,16 +93,16 @@
   context 'on compare page' do
     it_behaves_like 'Merge request button only shown when allowed' do
       let(:label) { 'Create merge request' }
-      let(:url) { namespace_project_compare_path(project.namespace, project, from: 'master', to: "'test'") }
-      let(:fork_url) { namespace_project_compare_path(forked_project.namespace, forked_project, from: 'master', to: "'test'") }
+      let(:url) { namespace_project_compare_path(project.namespace, project, from: 'master', to: 'feature') }
+      let(:fork_url) { namespace_project_compare_path(forked_project.namespace, forked_project, from: 'master', to: 'feature') }
     end
   end
 
   context 'on commits page' do
     it_behaves_like 'Merge request button only shown when allowed' do
       let(:label) { 'Create merge request' }
-      let(:url) { namespace_project_commits_path(project.namespace, project, "'test'") }
-      let(:fork_url) { namespace_project_commits_path(forked_project.namespace, forked_project, "'test'") }
+      let(:url) { namespace_project_commits_path(project.namespace, project, 'feature') }
+      let(:fork_url) { namespace_project_commits_path(forked_project.namespace, forked_project, 'feature') }
     end
   end
 end
-- 
GitLab


From 3b82444eb7791e58e3e0ba2f08b8ccde48e3d4c6 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Tue, 2 May 2017 13:13:28 -0500
Subject: [PATCH 142/363] Fix Rubocop complains.

---
 .../initializers/gettext_rails_i18n_patch.rb  |  1 -
 lib/tasks/gettext.rake                        | 23 ++-----------------
 2 files changed, 2 insertions(+), 22 deletions(-)

diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
index 616f6c45b0add..69118f464caa0 100644
--- a/config/initializers/gettext_rails_i18n_patch.rb
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -25,7 +25,6 @@ def self.convert_to_code(text)
 module GettextI18nRailsJs
   module Parser
     module Javascript
-
       # This is required to tell the `rake gettext:find` script to use the Javascript
       # parser for *.vue files.
       #
diff --git a/lib/tasks/gettext.rake b/lib/tasks/gettext.rake
index cc18616d2a785..0aa21a4bd13ba 100644
--- a/lib/tasks/gettext.rake
+++ b/lib/tasks/gettext.rake
@@ -4,27 +4,8 @@ namespace :gettext do
   # Customize list of translatable files
   # See: https://github.com/grosser/gettext_i18n_rails#customizing-list-of-translatable-files
   def files_to_translate
-    folders = [
-      "app",
-      "lib",
-      "config",
-      locale_path
-    ].join(",")
-
-    exts = [
-      "rb",
-      "erb",
-      "haml",
-      "slim",
-      "rhtml",
-      "js",
-      "jsx",
-      "vue",
-      "coffee",
-      "handlebars",
-      "hbs",
-      "mustache"
-    ].join(",")
+    folders = %W(app lib config #{locale_path}).join(',')
+    exts = %w(rb erb haml slim rhtml js jsx vue coffee handlebars hbs mustache).join(',')
 
     Dir.glob(
       "{#{folders}}/**/*.{#{exts}}"
-- 
GitLab


From 290cba5b23118bd40d64e5c6187afc820a6a0e38 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 2 May 2017 12:53:03 -0600
Subject: [PATCH 143/363] remove curr and previous states - extract tasks logic
 to tasks action

---
 .../javascripts/issue_show/actions/tasks.js   | 27 ++++++++
 .../issue_show/issue_title_description.vue    | 68 ++++++-------------
 app/controllers/projects/issues_controller.rb |  1 +
 app/views/projects/issues/_issue.html.haml    |  4 --
 4 files changed, 48 insertions(+), 52 deletions(-)
 create mode 100644 app/assets/javascripts/issue_show/actions/tasks.js

diff --git a/app/assets/javascripts/issue_show/actions/tasks.js b/app/assets/javascripts/issue_show/actions/tasks.js
new file mode 100644
index 0000000000000..0f6e71ce7ac97
--- /dev/null
+++ b/app/assets/javascripts/issue_show/actions/tasks.js
@@ -0,0 +1,27 @@
+export default (apiData, tasks) => {
+  const $tasks = $('#task_status');
+  const $tasksShort = $('#task_status_short');
+  const $issueableHeader = $('.issuable-header');
+  const zeroData = { api: null, tasks: null };
+
+  if ($tasks.length === 0) {
+    if (!(apiData.task_status.indexOf('0 of 0') >= 0)) {
+      $issueableHeader.append(`<span id="task_status">${apiData.task_status}</span>`);
+    } else {
+      $issueableHeader.append('<span id="task_status"></span>');
+    }
+  } else {
+    zeroData.api = apiData.task_status.indexOf('0 of 0') >= 0;
+    zeroData.tasks = tasks.indexOf('0 of 0') >= 0;
+  }
+
+  if ($tasks && !zeroData.api) {
+    $tasks.text(apiData.task_status);
+    $tasksShort.text(apiData.task_status);
+  } else if (zeroData.tasks) {
+    $issueableHeader.append(`<span id="task_status">${apiData.task_status}</span>`);
+  } else if (zeroData.api) {
+    $tasks.remove();
+    $tasksShort.remove();
+  }
+};
diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 7862923c00442..f6c3308388cd5 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -2,6 +2,7 @@
 import Visibility from 'visibilityjs';
 import Poll from './../lib/utils/poll';
 import Service from './services/index';
+import tasks from './actions/tasks';
 
 export default {
   props: {
@@ -30,14 +31,13 @@ export default {
     return {
       poll,
       apiData: {},
-      current: true,
       timeoutId: null,
       title: '<span></span>',
       titleText: '',
       description: '<span></span>',
       descriptionText: '',
       descriptionChange: false,
-      previousDescription: null,
+      tasks: '0 of 0',
     };
   },
   methods: {
@@ -46,18 +46,7 @@ export default {
       this.triggerAnimation();
     },
     updateTaskHTML() {
-      const tasks = document.querySelector('#task_status_short');
-      const zeroTasks = this.apiData.task_status.indexOf('0 of 0') >= 0;
-
-      if (tasks && !zeroTasks) {
-        tasks.innerText = this.apiData.task_status;
-      } else if (!tasks && !zeroTasks) {
-        $('.issuable-header').append(`
-          <span id="task_status_short" class="hidden-md hidden-lg">${this.apiData.task_status}</span>
-        `);
-      } else if (zeroTasks) {
-        $('#task_status_short').remove();
-      }
+      tasks(this.apiData, this.tasks);
     },
     elementsToVisualize(noTitleChange, noDescriptionChange) {
       const elementStack = [];
@@ -71,11 +60,7 @@ export default {
         // only change to true when we need to bind TaskLists the html of description
         this.descriptionChange = true;
         this.updateTaskHTML();
-
-        if (this.description !== '<span></span>') {
-          this.previousDescription = this.description;
-        }
-
+        this.tasks = this.apiData.task_status;
         elementStack.push(this.$el.querySelector('.wiki'));
       }
 
@@ -129,24 +114,18 @@ export default {
 
       this.animate(title, description, elementsToVisualize);
     },
-    handleCurrentOrPrevious() {
-      this.descriptionChange = true;
-      this.current = !this.current;
+    updateEditedTimeAgo() {
+      const toolTipTime = gl.utils.formatDate(this.apiData.updated_at);
+      const $timeAgoNode = $('.issue_edited_ago');
+
+      $timeAgoNode.attr('datetime', this.apiData.updated_at);
+      $timeAgoNode.attr('data-original-title', toolTipTime);
     },
   },
   computed: {
     descriptionClass() {
       return `description ${this.candescription} is-task-list-enabled`;
     },
-    showDescription() {
-      return this.current ? this.description : this.previousDescription;
-    },
-    previousOrCurrentButtonText() {
-      return this.current ? '<< Show Previous Decription' : 'Show Current Description >>';
-    },
-    prevCurrBtnClass() {
-      return this.current ? 'btn btn-sm btn-default' : 'btn btn-sm btn-primary';
-    },
   },
   created() {
     if (!Visibility.hidden()) {
@@ -164,18 +143,19 @@ export default {
   updated() {
     // if new html is injected (description changed) - bind TaskList and call renderGFM
     if (this.descriptionChange) {
+      this.updateEditedTimeAgo();
+
       $(this.$refs['issue-content-container-gfm-entry']).renderGFM();
 
-      if (this.current) {
-        const tl = new gl.TaskList({
-          dataType: 'issue',
-          fieldName: 'description',
-          selector: '.detail-page-description',
-        });
+      const tl = new gl.TaskList({
+        dataType: 'issue',
+        fieldName: 'description',
+        selector: '.detail-page-description',
+      });
 
-        return tl;
-      }
+      return tl && null;
     }
+
     return null;
   },
 };
@@ -188,17 +168,9 @@ export default {
       :class="descriptionClass"
       v-if="description"
     >
-      <div v-if="previousDescription">
-        <button
-          :class="prevCurrBtnClass"
-          @click="handleCurrentOrPrevious"
-        >
-          {{ previousOrCurrentButtonText }}
-        </button>
-      </div><br>
       <div
         class="wiki issue-realtime-trigger-pulse"
-        v-html="showDescription"
+        v-html="description"
         ref="issue-content-container-gfm-entry"
       >
       </div>
diff --git a/app/controllers/projects/issues_controller.rb b/app/controllers/projects/issues_controller.rb
index 7932b92c00549..87370dec3ca79 100644
--- a/app/controllers/projects/issues_controller.rb
+++ b/app/controllers/projects/issues_controller.rb
@@ -206,6 +206,7 @@ def rendered_title
       description_text: @issue.description,
       task_status: @issue.task_status,
       issue_number: @issue.iid,
+      updated_at: @issue.updated_at,
     }
   end
 
diff --git a/app/views/projects/issues/_issue.html.haml b/app/views/projects/issues/_issue.html.haml
index 0e3902c066ad2..6199569a20a9e 100644
--- a/app/views/projects/issues/_issue.html.haml
+++ b/app/views/projects/issues/_issue.html.haml
@@ -37,10 +37,6 @@
           &nbsp;
           - issue.labels.each do |label|
             = link_to_label(label, subject: issue.project, css_class: 'label-link')
-        - if issue.tasks?
-          &nbsp;
-          %span.task-status
-            = issue.task_status
 
         .pull-right.issue-updated-at
           %span updated #{time_ago_with_tooltip(issue.updated_at, placement: 'bottom', html_class: 'issue_update_ago')}
-- 
GitLab


From 165bcec30506380efa912b3dba25815aeea9c92f Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Tue, 2 May 2017 20:17:45 +0100
Subject: [PATCH 144/363] [ci skip] added changelog entry

---
 changelogs/unreleased/balsalmiq-support.yml | 4 ++++
 1 file changed, 4 insertions(+)
 create mode 100644 changelogs/unreleased/balsalmiq-support.yml

diff --git a/changelogs/unreleased/balsalmiq-support.yml b/changelogs/unreleased/balsalmiq-support.yml
new file mode 100644
index 0000000000000..56a0b4c83fae6
--- /dev/null
+++ b/changelogs/unreleased/balsalmiq-support.yml
@@ -0,0 +1,4 @@
+---
+title: Added balsamiq file viewer
+merge_request: 10564
+author:
-- 
GitLab


From f6df135717a09b7598f274a1c9d877ffcb59be26 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 2 May 2017 15:01:23 -0600
Subject: [PATCH 145/363] test realtime changes and hit more branches

---
 .../issue_title_description_spec.js           | 41 ++++++++++---------
 spec/javascripts/issue_show/mock_data.js      | 22 +++++++---
 2 files changed, 37 insertions(+), 26 deletions(-)

diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index 4c7d4748693dd..e1d0f51f9cb12 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -7,9 +7,12 @@ import issueShowData from './mock_data';
 
 window.$ = $;
 
-const issueShowInterceptor = (request, next) => {
-  next(request.respondWith(JSON.stringify(issueShowData), {
+const issueShowInterceptor = data => (request, next) => {
+  next(request.respondWith(JSON.stringify(data), {
     status: 200,
+    headers: {
+      'POLL-INTERVAL': 1,
+    },
   }));
 };
 
@@ -22,16 +25,15 @@ describe('Issue Title', () => {
 
   beforeEach(() => {
     comps.IssueTitleComponent = Vue.extend(issueTitle);
-    Vue.http.interceptors.push(issueShowInterceptor);
   });
 
   afterEach(() => {
-    Vue.http.interceptors = _.without(
-      Vue.http.interceptors, issueShowInterceptor,
-    );
+    Vue.http.interceptors = _.without(Vue.http.interceptors, issueShowInterceptor);
   });
 
-  it('should render a title', (done) => {
+  it('should render a title/description and update title/description on update', (done) => {
+    Vue.http.interceptors.push(issueShowInterceptor(issueShowData.initialRequest));
+
     const issueShowComponent = new comps.IssueTitleComponent({
       propsData: {
         candescription: '.css-stuff',
@@ -41,22 +43,21 @@ describe('Issue Title', () => {
 
     // need setTimeout because actual setTimeout in code :P
     setTimeout(() => {
-      expect(document.querySelector('title').innerText)
-        .toContain('this is a title (#1)');
-
-      expect(issueShowComponent.$el.querySelector('.title').innerHTML)
-        .toContain('<p>this is a title</p>');
-
-      expect(issueShowComponent.$el.querySelector('.wiki').innerHTML)
-        .toContain('<p>this is a description!</p>');
+      expect(document.querySelector('title').innerText).toContain('this is a title (#1)');
+      expect(issueShowComponent.$el.querySelector('.title').innerHTML).toContain('<p>this is a title</p>');
+      expect(issueShowComponent.$el.querySelector('.wiki').innerHTML).toContain('<p>this is a description!</p>');
+      expect(issueShowComponent.$el.querySelector('.js-task-list-field').innerText).toContain('this is a description');
 
-      const hiddenText = issueShowComponent.$el
-        .querySelector('.js-task-list-field').innerText;
+      Vue.http.interceptors.push(issueShowInterceptor(issueShowData.secondRequest));
 
-      expect(hiddenText)
-        .toContain('this is a description');
+      setTimeout(() => {
+        expect(document.querySelector('title').innerText).toContain('2 (#1)');
+        expect(issueShowComponent.$el.querySelector('.title').innerHTML).toContain('<p>2</p>');
+        expect(issueShowComponent.$el.querySelector('.wiki').innerHTML).toContain('<p>42</p>');
+        expect(issueShowComponent.$el.querySelector('.js-task-list-field').innerText).toContain('42');
 
-      done();
+        done();
+      }, 10);
     }, 10);
     // 10ms is just long enough for the update hook to fire
   });
diff --git a/spec/javascripts/issue_show/mock_data.js b/spec/javascripts/issue_show/mock_data.js
index b7b01391a18a8..5d234ebf25425 100644
--- a/spec/javascripts/issue_show/mock_data.js
+++ b/spec/javascripts/issue_show/mock_data.js
@@ -1,8 +1,18 @@
 export default {
-  title: '<p>this is a title</p>',
-  title_text: 'this is a title',
-  description: '<p>this is a description!</p>',
-  description_text: 'this is a description',
-  issue_number: 1,
-  task_status: '2/4 completed',
+  initialRequest: {
+    title: '<p>this is a title</p>',
+    title_text: 'this is a title',
+    description: '<p>this is a description!</p>',
+    description_text: 'this is a description',
+    issue_number: 1,
+    task_status: '2 of 4 completed',
+  },
+  secondRequest: {
+    title: '<p>2</p>',
+    title_text: '2',
+    description: '<p>42</p>',
+    description_text: '42',
+    issue_number: 1,
+    task_status: '0 of 0 completed',
+  },
 };
-- 
GitLab


From c50c5e92e9619cbb2a0ae7cc49697af5df46988c Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 2 May 2017 20:16:39 -0600
Subject: [PATCH 146/363] issue_spec - getting closer

---
 .../issue_title_description_spec.js           |  6 +--
 spec/javascripts/issue_show/mock_data.js      |  8 +++
 spec/javascripts/issue_spec.js                | 52 +++++++++++++++++--
 3 files changed, 57 insertions(+), 9 deletions(-)

diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index e1d0f51f9cb12..3877ed2164b90 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -11,14 +11,12 @@ const issueShowInterceptor = data => (request, next) => {
   next(request.respondWith(JSON.stringify(data), {
     status: 200,
     headers: {
-      'POLL-INTERVAL': 1,
+      'POLL-INTERVAL': 10,
     },
   }));
 };
 
 describe('Issue Title', () => {
-  document.body.innerHTML = '<span id="task_status"></span>';
-
   const comps = {
     IssueTitleComponent: {},
   };
@@ -57,7 +55,7 @@ describe('Issue Title', () => {
         expect(issueShowComponent.$el.querySelector('.js-task-list-field').innerText).toContain('42');
 
         done();
-      }, 10);
+      }, 20);
     }, 10);
     // 10ms is just long enough for the update hook to fire
   });
diff --git a/spec/javascripts/issue_show/mock_data.js b/spec/javascripts/issue_show/mock_data.js
index 5d234ebf25425..ad5a7b63470ad 100644
--- a/spec/javascripts/issue_show/mock_data.js
+++ b/spec/javascripts/issue_show/mock_data.js
@@ -15,4 +15,12 @@ export default {
     issue_number: 1,
     task_status: '0 of 0 completed',
   },
+  issueSpecRequest: {
+    title: '<p>this is a title</p>',
+    title_text: 'this is a title',
+    description: '<li class="task-list-item enabled"><input type="checkbox" class="task-list-item-checkbox">Task List Item</li>',
+    description_text: '- [ ] Task List Item',
+    issue_number: 1,
+    task_status: '0 of 1 completed',
+  },
 };
diff --git a/spec/javascripts/issue_spec.js b/spec/javascripts/issue_spec.js
index 9a2570ef7e93c..e2e9cc01424d4 100644
--- a/spec/javascripts/issue_spec.js
+++ b/spec/javascripts/issue_spec.js
@@ -1,5 +1,10 @@
 /* eslint-disable space-before-function-paren, one-var, one-var-declaration-per-line, no-use-before-define, comma-dangle, max-len */
 import Issue from '~/issue';
+import Vue from 'vue';
+import '~/render_math';
+import '~/render_gfm';
+import IssueTitle from '~/issue_show/issue_title_description.vue';
+import issueShowData from './issue_show/mock_data';
 
 require('~/lib/utils/text_utility');
 
@@ -75,16 +80,53 @@ describe('Issue', function() {
     expect($btnReopen).toHaveText('Reopen issue');
   }
 
-  describe('task lists', function() {
+  fdescribe('task lists', function() {
+    const issueShowInterceptor = data => (request, next) => {
+      next(request.respondWith(JSON.stringify(data), {
+        status: 200,
+      }));
+    };
+
     beforeEach(function() {
       loadFixtures('issues/issue-with-task-list.html.raw');
       this.issue = new Issue();
+      Vue.http.interceptors.push(issueShowInterceptor(issueShowData.issueSpecRequest));
+    });
+
+    afterEach(function() {
+      Vue.http.interceptors = _.without(Vue.http.interceptors, issueShowInterceptor);
     });
 
-    it('modifies the Markdown field', function() {
-      spyOn(jQuery, 'ajax').and.stub();
-      $('input[type=checkbox]').attr('checked', true).trigger('change');
-      expect($('.js-task-list-field').val()).toBe('- [x] Task List Item');
+    it('modifies the Markdown field', function(done) {
+      // gotta actually render it for jquery to find elements
+      const vm = new Vue({
+        el: document.querySelector('.issue-title-entrypoint'),
+        components: {
+          IssueTitle,
+        },
+        render: createElement => createElement(IssueTitle, {
+          props: {
+            candescription: '.js-task-list-container',
+            endpoint: '/gitlab-org/gitlab-shell/issues/9/rendered_title',
+          },
+        }),
+      });
+
+      setTimeout(() => {
+        spyOn(jQuery, 'ajax').and.stub();
+
+        const description = '<li class="task-list-item enabled"><input type="checkbox" class="task-list-item-checkbox"> Task List Item</li>';
+
+        expect(document.querySelector('title').innerText).toContain('this is a title (#1)');
+        expect(vm.$el.querySelector('.title').innerHTML).toContain('<p>this is a title</p>');
+        expect(vm.$el.querySelector('.wiki').innerHTML).toContain(description);
+        expect(vm.$el.querySelector('.js-task-list-field').value).toContain('- [ ] Task List Item');
+
+        // somehow the dom does not have a closest `.js-task-list.field` to the `.task-list-item-checkbox`
+        $('input[type=checkbox]').attr('checked', true).trigger('change');
+        expect($('.js-task-list-field').val()).toBe('- [x] Task List Item');
+        done();
+      }, 10);
     });
 
     it('submits an ajax request on tasklist:changed', function() {
-- 
GitLab


From b29f91b06cc90f48c8322175ad92087579f96bdd Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Tue, 2 May 2017 20:17:10 -0600
Subject: [PATCH 147/363] remove fdescribe [ci skip]

---
 spec/javascripts/issue_spec.js | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/javascripts/issue_spec.js b/spec/javascripts/issue_spec.js
index e2e9cc01424d4..788d7ccc5e42c 100644
--- a/spec/javascripts/issue_spec.js
+++ b/spec/javascripts/issue_spec.js
@@ -80,7 +80,7 @@ describe('Issue', function() {
     expect($btnReopen).toHaveText('Reopen issue');
   }
 
-  fdescribe('task lists', function() {
+  describe('task lists', function() {
     const issueShowInterceptor = data => (request, next) => {
       next(request.respondWith(JSON.stringify(data), {
         status: 200,
-- 
GitLab


From 6d5364cfb0e39f49afac9b465f37bd19185c3755 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Tue, 2 May 2017 23:36:36 -0500
Subject: [PATCH 148/363] First round of updates from the code review.

---
 .../components/stage_code_component.js        |  2 +-
 .../components/stage_issue_component.js       |  2 +-
 .../components/stage_plan_component.js        |  2 +-
 .../components/stage_production_component.js  |  2 +-
 .../components/stage_review_component.js      |  2 +-
 .../components/stage_staging_component.js     |  2 +-
 .../components/total_time_component.js        |  6 +-
 app/assets/javascripts/locale/de/app.js       |  2 +-
 app/assets/javascripts/locale/en/app.js       |  2 +-
 app/assets/javascripts/locale/es/app.js       |  2 +-
 app/controllers/application_controller.rb     | 15 +++--
 app/controllers/profiles_controller.rb        |  1 -
 .../projects/cycle_analytics_controller.rb    |  1 -
 app/models/user.rb                            |  1 +
 app/views/profiles/show.html.haml             |  2 +-
 .../projects/cycle_analytics/show.html.haml   |  2 +-
 config/initializers/fast_gettext.rb           |  4 +-
 ...3035209_add_preferred_language_to_users.rb |  4 +-
 lib/gitlab/i18n.rb                            | 12 +++-
 locale/de/gitlab.po                           | 51 +++++++++--------
 locale/en/gitlab.po                           | 51 +++++++++--------
 locale/es/gitlab.po                           | 54 +++++++++---------
 locale/gitlab.pot                             | 55 +++++++++----------
 23 files changed, 143 insertions(+), 134 deletions(-)

diff --git a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
index 0bd15ee773d34..f22299898a5c4 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageCodeComponent = Vue.extend({
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              {{ __('by') }}
+              {{ __('Author|by') }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
           </div>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
index b6bf9a2572e8f..f400fd2fb142c 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageIssueComponent = Vue.extend({
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-              {{ __('by') }}
+              {{ __('Author|by') }}
               <a :href="issue.author.webUrl" class="issue-author-link">
                 {{ issue.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
index 116c118b82413..b85b8902fc7c5 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
@@ -31,7 +31,7 @@ global.cycleAnalytics.StagePlanComponent = Vue.extend({
               </a>
             </h5>
             <span>
-              {{ __('First') }}
+              {{ __('OfFirstTime|First') }}
               <span class="commit-icon">${iconCommit}</span>
               <a :href="commit.commitUrl" class="commit-hash-link monospace">{{ commit.shortSha }}</a>
               {{ __('pushed by') }}
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
index 18fb09d6d2f95..18c30f490fd76 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageProductionComponent = Vue.extend({
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-            {{ __('by') }}
+            {{ __('Author|by') }}
             <a :href="issue.author.webUrl" class="issue-author-link">
               {{ issue.author.name }}
             </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
index af5f1b42ea2b7..988067601284b 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageReviewComponent = Vue.extend({
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              {{ __('by') }}
+              {{ __('Author|by') }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
             <template v-if="mergeRequest.state === 'closed'">
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
index a0bd1bab96e55..f13294c7af707 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageStagingComponent = Vue.extend({
             </h5>
             <span>
               <a :href="build.url" class="build-date">{{ build.date }}</a>
-              {{ __('by') }}
+              {{ __('Author|by') }}
               <a :href="build.author.webUrl" class="issue-author-link">
                 {{ build.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/total_time_component.js b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
index 48cc0a5c58dbb..1eef7d9a8738d 100644
--- a/app/assets/javascripts/cycle_analytics/components/total_time_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
@@ -13,9 +13,9 @@ global.cycleAnalytics.TotalTimeComponent = Vue.extend({
     <span class="total-time">
       <template v-if="Object.keys(time).length">
         <template v-if="time.days">{{ time.days }} <span>{{ n__('day', 'days', time.days) }}</span></template>
-        <template v-if="time.hours">{{ time.hours }} <span>{{ __('hr') }}</span></template>
-        <template v-if="time.mins && !time.days">{{ time.mins }} <span>{{ n__('min', 'mins', time.mins) }}</span></template>
-        <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>s</span></template>
+        <template v-if="time.hours">{{ time.hours }} <span>{{ n__('Time|hr', 'Time|hrs', time.hours) }}</span></template>
+        <template v-if="time.mins && !time.days">{{ time.mins }} <span>{{ n__('Time|min', 'Time|mins', time.mins) }}</span></template>
+        <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>{{ __('Time|s') }}</span></template>
       </template>
       <template v-else>
         --
diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index 1b10c19c48e78..51c5ad512fc7a 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index 2754e177653b0..68beb0afda016 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Deutsch":[""],"English":[""],"First":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"Opened":[""],"Pipeline Health":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"Spanish":[""],"Stage":[""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"by":[""],"day":["",""],"hr":[""],"min":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index 5d1cfddf76abf..511c70b30a6ca 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-27 19:15-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Cycle|Code":["Código"],"Cycle|Issue":["Tema"],"Cycle|Plan":["Planificación"],"Cycle|Review":["Revisión"],"Cycle|Staging":["Etapa"],"Cycle|Test":["Pruebas"],"Deploys":["Despliegues"],"Deutsch":["Alemán"],"English":["Inglés"],"First":["Primer"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Limited to showing 50 events at most":["Limitado a mostrar máximo 50 eventos"],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"Spanish":["Español"],"Stage":["Etapa"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"by":["por"],"day":["día","días"],"hr":[""],"min":["",""],"pushed by":["enviado por"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-02 23:17-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":["por"],"Commits":["Cambios"],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Cycle|Code":["Código"],"Cycle|Issue":["Tema"],"Cycle|Plan":["Planificación"],"Cycle|Review":["Revisión"],"Cycle|Staging":["Etapa"],"Cycle|Test":["Pruebas"],"Deploys":["Despliegues"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Limited to showing 50 events at most":["Limitado a mostrar máximo 50 eventos"],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OfFirstTime|First":["Primer"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"],"pushed by":["enviado por"]}}};
\ No newline at end of file
diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index f2997d0b0c421..24017e8ea40ae 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -21,6 +21,8 @@ class ApplicationController < ActionController::Base
   before_action :configure_permitted_parameters, if: :devise_controller?
   before_action :require_email, unless: :devise_controller?
 
+  around_action :set_locale
+
   protect_from_forgery with: :exception
 
   helper_method :can?, :current_application_settings
@@ -271,9 +273,14 @@ def u2f_app_id
   end
 
   def set_locale
-    requested_locale = current_user&.preferred_language || request.env['HTTP_ACCEPT_LANGUAGE'] || I18n.default_locale
-    locale = FastGettext.set_locale(requested_locale)
-
-    I18n.locale = locale
+    begin
+      requested_locale = current_user&.preferred_language || I18n.default_locale
+      locale = FastGettext.set_locale(requested_locale)
+      I18n.locale = locale
+
+      yield
+    ensure
+      I18n.locale = I18n.default_locale
+    end
   end
 end
diff --git a/app/controllers/profiles_controller.rb b/app/controllers/profiles_controller.rb
index 0f01bf7e7067a..57e23cea00eb5 100644
--- a/app/controllers/profiles_controller.rb
+++ b/app/controllers/profiles_controller.rb
@@ -3,7 +3,6 @@ class ProfilesController < Profiles::ApplicationController
 
   before_action :user
   before_action :authorize_change_username!, only: :update_username
-  before_action :set_locale, only: :show
   skip_before_action :require_email, only: [:show, :update]
 
   def show
diff --git a/app/controllers/projects/cycle_analytics_controller.rb b/app/controllers/projects/cycle_analytics_controller.rb
index 7ef8872a90b5b..88ac3ad046b57 100644
--- a/app/controllers/projects/cycle_analytics_controller.rb
+++ b/app/controllers/projects/cycle_analytics_controller.rb
@@ -4,7 +4,6 @@ class Projects::CycleAnalyticsController < Projects::ApplicationController
   include CycleAnalyticsParams
 
   before_action :authorize_read_cycle_analytics!
-  before_action :set_locale, only: :show
 
   def show
     @cycle_analytics = ::CycleAnalytics.new(@project, options(cycle_analytics_params))
diff --git a/app/models/user.rb b/app/models/user.rb
index 2b7ebe6c1a71f..43c5fdc038d47 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -23,6 +23,7 @@ class User < ActiveRecord::Base
   default_value_for :hide_no_password, false
   default_value_for :project_view, :files
   default_value_for :notified_of_own_activity, false
+  default_value_for :preferred_language, I18n.default_locale
 
   attr_encrypted :otp_secret,
     key:       Gitlab::Application.secrets.otp_key_base,
diff --git a/app/views/profiles/show.html.haml b/app/views/profiles/show.html.haml
index 9846f01603fd2..6e14be62e32e3 100644
--- a/app/views/profiles/show.html.haml
+++ b/app/views/profiles/show.html.haml
@@ -74,7 +74,7 @@
         %span.help-block This email will be displayed on your public profile.
       .form-group
         = f.label :preferred_language, class: "label-light"
-        = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES.map { |value, label| [_(label), value] },
+        = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES.map { |value, label| [label, value] },
           {}, class: "select2"
       .form-group
         = f.label :skype, class: "label-light"
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index a1a227d2fe9d0..a8ce021e05f7a 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -52,7 +52,7 @@
             %ul
               %li.stage-header
                 %span.stage-name
-                  {{ __('Stage') }}
+                  {{ __('ProjectLifecycle|Stage') }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The phase of the development lifecycle."), "aria-hidden" => "true" }
               %li.median-header
                 %span.stage-name
diff --git a/config/initializers/fast_gettext.rb b/config/initializers/fast_gettext.rb
index 10a3ee02b8510..bea85b43b9547 100644
--- a/config/initializers/fast_gettext.rb
+++ b/config/initializers/fast_gettext.rb
@@ -1,3 +1,5 @@
 FastGettext.add_text_domain 'gitlab', path: 'locale', type: :po
 FastGettext.default_text_domain = 'gitlab'
-FastGettext.default_available_locales = Gitlab::I18n::AVAILABLE_LANGUAGES.keys
+FastGettext.default_available_locales = Gitlab::I18n.available_locales
+
+I18n.available_locales = Gitlab::I18n.available_locales
diff --git a/db/migrate/20170413035209_add_preferred_language_to_users.rb b/db/migrate/20170413035209_add_preferred_language_to_users.rb
index 6fe91656eeba5..92f1d6f243633 100644
--- a/db/migrate/20170413035209_add_preferred_language_to_users.rb
+++ b/db/migrate/20170413035209_add_preferred_language_to_users.rb
@@ -6,10 +6,8 @@ class AddPreferredLanguageToUsers < ActiveRecord::Migration
 
   DOWNTIME = false
 
-  disable_ddl_transaction!
-
   def up
-    add_column_with_default :users, :preferred_language, :string, default: 'en'
+    add_column :users, :preferred_language, :string
   end
 
   def down
diff --git a/lib/gitlab/i18n.rb b/lib/gitlab/i18n.rb
index 64a86b55c7f5f..a7addee0dcd15 100644
--- a/lib/gitlab/i18n.rb
+++ b/lib/gitlab/i18n.rb
@@ -1,9 +1,15 @@
 module Gitlab
   module I18n
+    extend self
+
     AVAILABLE_LANGUAGES = {
-      'en' => N_('English'),
-      'es' => N_('Spanish'),
-      'de' => N_('Deutsch')
+      en: 'English',
+      es: 'Español',
+      de: 'Deutsch'
     }.freeze
+
+    def available_locales
+      AVAILABLE_LANGUAGES.keys
+    end
   end
 end
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index 843648c25f1fe..beeca35b879dc 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -17,9 +17,15 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
+msgid "Author|by"
+msgstr ""
+
 msgid "Commits"
 msgstr ""
 
+msgid "Cycle"
+msgstr ""
+
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
@@ -44,15 +50,6 @@ msgstr ""
 msgid "Deploys"
 msgstr ""
 
-msgid "Deutsch"
-msgstr ""
-
-msgid "English"
-msgstr ""
-
-msgid "First"
-msgstr ""
-
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
@@ -77,12 +74,18 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
+msgid "OfFirstTime|First"
+msgstr ""
+
 msgid "Opened"
 msgstr ""
 
 msgid "Pipeline Health"
 msgstr ""
 
+msgid "ProjectLifecycle|Stage"
+msgstr ""
+
 msgid "Read more"
 msgstr ""
 
@@ -109,12 +112,6 @@ msgid_plural "Showing %d events"
 msgstr[0] ""
 msgstr[1] ""
 
-msgid "Spanish"
-msgstr ""
-
-msgid "Stage"
-msgstr ""
-
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
@@ -148,6 +145,19 @@ msgstr ""
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
 msgstr ""
 
+msgid "Time|hr"
+msgid_plural "Time|hrs"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|min"
+msgid_plural "Time|mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|s"
+msgstr ""
+
 msgid "Total Time"
 msgstr ""
 
@@ -160,21 +170,10 @@ msgstr ""
 msgid "You need permission."
 msgstr ""
 
-msgid "by"
-msgstr ""
-
 msgid "day"
 msgid_plural "days"
 msgstr[0] ""
 msgstr[1] ""
 
-msgid "hr"
-msgstr ""
-
-msgid "min"
-msgid_plural "mins"
-msgstr[0] ""
-msgstr[1] ""
-
 msgid "pushed by"
 msgstr ""
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index e98931685d753..9596531fdbd0c 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -17,9 +17,15 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
+msgid "Author|by"
+msgstr ""
+
 msgid "Commits"
 msgstr ""
 
+msgid "Cycle"
+msgstr ""
+
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
@@ -44,15 +50,6 @@ msgstr ""
 msgid "Deploys"
 msgstr ""
 
-msgid "Deutsch"
-msgstr ""
-
-msgid "English"
-msgstr ""
-
-msgid "First"
-msgstr ""
-
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
@@ -77,12 +74,18 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
+msgid "OfFirstTime|First"
+msgstr ""
+
 msgid "Opened"
 msgstr ""
 
 msgid "Pipeline Health"
 msgstr ""
 
+msgid "ProjectLifecycle|Stage"
+msgstr ""
+
 msgid "Read more"
 msgstr ""
 
@@ -109,12 +112,6 @@ msgid_plural "Showing %d events"
 msgstr[0] ""
 msgstr[1] ""
 
-msgid "Spanish"
-msgstr ""
-
-msgid "Stage"
-msgstr ""
-
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
@@ -148,6 +145,19 @@ msgstr ""
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
 msgstr ""
 
+msgid "Time|hr"
+msgid_plural "Time|hrs"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|min"
+msgid_plural "Time|mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|s"
+msgstr ""
+
 msgid "Total Time"
 msgstr ""
 
@@ -160,21 +170,10 @@ msgstr ""
 msgid "You need permission."
 msgstr ""
 
-msgid "by"
-msgstr ""
-
 msgid "day"
 msgid_plural "days"
 msgstr[0] ""
 msgstr[1] ""
 
-msgid "hr"
-msgstr ""
-
-msgid "min"
-msgid_plural "mins"
-msgstr[0] ""
-msgstr[1] ""
-
 msgid "pushed by"
 msgstr ""
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 644007f514928..56f889c7141b4 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-04-27 19:15-0500\n"
+"PO-Revision-Date: 2017-05-02 23:17-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -17,9 +17,15 @@ msgstr ""
 "Last-Translator: \n"
 "X-Generator: Poedit 2.0.1\n"
 
+msgid "Author|by"
+msgstr "por"
+
 msgid "Commits"
 msgstr "Cambios"
 
+msgid "Cycle"
+msgstr ""
+
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr "Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."
 
@@ -50,15 +56,6 @@ msgstr "Pruebas"
 msgid "Deploys"
 msgstr "Despliegues"
 
-msgid "Deutsch"
-msgstr "Alemán"
-
-msgid "English"
-msgstr "Inglés"
-
-msgid "First"
-msgstr "Primer"
-
 msgid "Introducing Cycle Analytics"
 msgstr "Introducción a Cycle Analytics"
 
@@ -83,12 +80,19 @@ msgstr "No disponible"
 msgid "Not enough data"
 msgstr "No hay suficientes datos"
 
+#, fuzzy
+msgid "OfFirstTime|First"
+msgstr "Primer"
+
 msgid "Opened"
 msgstr "Abiertos"
 
 msgid "Pipeline Health"
 msgstr "Estado del Pipeline"
 
+msgid "ProjectLifecycle|Stage"
+msgstr "Etapa"
+
 msgid "Read more"
 msgstr "Leer más"
 
@@ -117,12 +121,6 @@ msgid_plural "Showing %d events"
 msgstr[0] "Mostrando %d evento"
 msgstr[1] "Mostrando %d eventos"
 
-msgid "Spanish"
-msgstr "Español"
-
-msgid "Stage"
-msgstr "Etapa"
-
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr "La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."
 
@@ -156,6 +154,19 @@ msgstr "El tiempo utilizado por cada entrada de datos obtenido por esa etapa."
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
 msgstr "El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."
 
+msgid "Time|hr"
+msgid_plural "Time|hrs"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|min"
+msgid_plural "Time|mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|s"
+msgstr ""
+
 msgid "Total Time"
 msgstr "Tiempo Total"
 
@@ -168,21 +179,10 @@ msgstr "No hay suficientes datos para mostrar en esta etapa."
 msgid "You need permission."
 msgstr "Necesitas permisos."
 
-msgid "by"
-msgstr "por"
-
 msgid "day"
 msgid_plural "days"
 msgstr[0] "día"
 msgstr[1] "días"
 
-msgid "hr"
-msgstr ""
-
-msgid "min"
-msgid_plural "mins"
-msgstr[0] ""
-msgstr[1] ""
-
 msgid "pushed by"
 msgstr "enviado por"
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index 1b60f9dad7a57..956e05926e10b 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-27 19:13-0500\n"
-"PO-Revision-Date: 2017-04-27 19:13-0500\n"
+"POT-Creation-Date: 2017-05-02 23:33-0500\n"
+"PO-Revision-Date: 2017-05-02 23:33-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -18,9 +18,15 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
 
+msgid "Author|by"
+msgstr ""
+
 msgid "Commits"
 msgstr ""
 
+msgid "Cycle"
+msgstr ""
+
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
@@ -45,15 +51,6 @@ msgstr ""
 msgid "Deploys"
 msgstr ""
 
-msgid "Deutsch"
-msgstr ""
-
-msgid "English"
-msgstr ""
-
-msgid "First"
-msgstr ""
-
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
@@ -78,12 +75,18 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
+msgid "OfFirstTime|First"
+msgstr ""
+
 msgid "Opened"
 msgstr ""
 
 msgid "Pipeline Health"
 msgstr ""
 
+msgid "ProjectLifecycle|Stage"
+msgstr ""
+
 msgid "Read more"
 msgstr ""
 
@@ -110,12 +113,6 @@ msgid_plural "Showing %d events"
 msgstr[0] ""
 msgstr[1] ""
 
-msgid "Spanish"
-msgstr ""
-
-msgid "Stage"
-msgstr ""
-
 msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
 msgstr ""
 
@@ -149,6 +146,19 @@ msgstr ""
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
 msgstr ""
 
+msgid "Time|hr"
+msgid_plural "Time|hrs"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|min"
+msgid_plural "Time|mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|s"
+msgstr ""
+
 msgid "Total Time"
 msgstr ""
 
@@ -161,21 +171,10 @@ msgstr ""
 msgid "You need permission."
 msgstr ""
 
-msgid "by"
-msgstr ""
-
 msgid "day"
 msgid_plural "days"
 msgstr[0] ""
 msgstr[1] ""
 
-msgid "hr"
-msgstr ""
-
-msgid "min"
-msgid_plural "mins"
-msgstr[0] ""
-msgstr[1] ""
-
 msgid "pushed by"
 msgstr ""
-- 
GitLab


From 9ee274c03166b5773d433e3947b5d566fda02e53 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 3 May 2017 09:06:28 +0100
Subject: [PATCH 149/363] Fixed tags sort dropdown being empty

Closes #31618
---
 app/helpers/sorting_helper.rb               |  8 ++++++++
 app/views/projects/tags/index.html.haml     | 16 +++++++---------
 changelogs/unreleased/tags-sort-default.yml |  4 ++++
 spec/features/projects/tags/sort_spec.rb    | 15 +++++++++++++++
 4 files changed, 34 insertions(+), 9 deletions(-)
 create mode 100644 changelogs/unreleased/tags-sort-default.yml
 create mode 100644 spec/features/projects/tags/sort_spec.rb

diff --git a/app/helpers/sorting_helper.rb b/app/helpers/sorting_helper.rb
index 2fda98cae9040..4882d9b71d2b4 100644
--- a/app/helpers/sorting_helper.rb
+++ b/app/helpers/sorting_helper.rb
@@ -70,6 +70,14 @@ def branches_sort_options_hash
     }
   end
 
+  def tags_sort_options_hash
+    {
+      sort_value_name => sort_title_name,
+      sort_value_recently_updated => sort_title_recently_updated,
+      sort_value_oldest_updated => sort_title_oldest_updated
+    }
+  end
+
   def sort_title_priority
     'Priority'
   end
diff --git a/app/views/projects/tags/index.html.haml b/app/views/projects/tags/index.html.haml
index 7f9a44e565f61..c14bbf4f05f72 100644
--- a/app/views/projects/tags/index.html.haml
+++ b/app/views/projects/tags/index.html.haml
@@ -14,16 +14,14 @@
       .dropdown
         %button.dropdown-toggle{ type: 'button', data: { toggle: 'dropdown'} }
           %span.light
-            = projects_sort_options_hash[@sort]
+            = tags_sort_options_hash[@sort]
           = icon('chevron-down')
-        %ul.dropdown-menu.dropdown-menu-align-right
-          %li
-            = link_to filter_tags_path(sort: sort_value_name) do
-              = sort_title_name
-            = link_to filter_tags_path(sort: sort_value_recently_updated) do
-              = sort_title_recently_updated
-            = link_to filter_tags_path(sort: sort_value_oldest_updated) do
-              = sort_title_oldest_updated
+        %ul.dropdown-menu.dropdown-menu-align-right.dropdown-menu-selectable
+          %li.dropdown-header
+            Sort by
+          - tags_sort_options_hash.each do |value, title|
+            %li
+              = link_to title, filter_tags_path(sort: value), class: ("is-active" if @sort == value)
       - if can?(current_user, :push_code, @project)
         = link_to new_namespace_project_tag_path(@project.namespace, @project), class: 'btn btn-create new-tag-btn' do
           New tag
diff --git a/changelogs/unreleased/tags-sort-default.yml b/changelogs/unreleased/tags-sort-default.yml
new file mode 100644
index 0000000000000..265b765d5400c
--- /dev/null
+++ b/changelogs/unreleased/tags-sort-default.yml
@@ -0,0 +1,4 @@
+---
+title: Fixed tags sort from defaulting to empty
+merge_request:
+author:
diff --git a/spec/features/projects/tags/sort_spec.rb b/spec/features/projects/tags/sort_spec.rb
new file mode 100644
index 0000000000000..835cd1507fbfc
--- /dev/null
+++ b/spec/features/projects/tags/sort_spec.rb
@@ -0,0 +1,15 @@
+require 'spec_helper'
+
+feature 'Tags sort dropdown', :feature do
+  let(:project) { create(:project) }
+
+  before do
+    login_as(:admin)
+
+    visit namespace_project_tags_path(project.namespace, project)
+  end
+
+  it 'defaults sort dropdown to last updated' do
+    expect(page).to have_button('Last updated')
+  end
+end
-- 
GitLab


From 72307940bb04d29dc751adb5e76330d2ce45e9ce Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 3 May 2017 12:32:39 +0200
Subject: [PATCH 150/363] Improve code style related to protected actions

---
 app/serializers/build_action_entity.rb              | 2 +-
 app/serializers/build_entity.rb                     | 4 ++--
 app/serializers/pipeline_entity.rb                  | 6 +++---
 app/services/ci/play_build_service.rb               | 2 +-
 spec/controllers/projects/builds_controller_spec.rb | 4 +---
 5 files changed, 8 insertions(+), 10 deletions(-)

diff --git a/app/serializers/build_action_entity.rb b/app/serializers/build_action_entity.rb
index 0bb7e5610731f..3eff724ae9149 100644
--- a/app/serializers/build_action_entity.rb
+++ b/app/serializers/build_action_entity.rb
@@ -19,6 +19,6 @@ class BuildActionEntity < Grape::Entity
   alias_method :build, :object
 
   def playable?
-    can?(request.user, :play_build, build) && build.playable?
+    build.playable? && can?(request.user, :play_build, build)
   end
 end
diff --git a/app/serializers/build_entity.rb b/app/serializers/build_entity.rb
index f301900c43c43..10ba7c90c10c2 100644
--- a/app/serializers/build_entity.rb
+++ b/app/serializers/build_entity.rb
@@ -12,7 +12,7 @@ class BuildEntity < Grape::Entity
     path_to(:retry_namespace_project_build, build)
   end
 
-  expose :play_path, if: proc { playable? } do |build|
+  expose :play_path, if: -> (*) { playable? } do |build|
     path_to(:play_namespace_project_build, build)
   end
 
@@ -26,7 +26,7 @@ class BuildEntity < Grape::Entity
   alias_method :build, :object
 
   def playable?
-    can?(request.user, :play_build, build) && build.playable?
+    build.playable? && can?(request.user, :play_build, build)
   end
 
   def detailed_status
diff --git a/app/serializers/pipeline_entity.rb b/app/serializers/pipeline_entity.rb
index ad8b4d43e8f07..7eb7aac72eb5a 100644
--- a/app/serializers/pipeline_entity.rb
+++ b/app/serializers/pipeline_entity.rb
@@ -48,15 +48,15 @@ class PipelineEntity < Grape::Entity
   end
 
   expose :commit, using: CommitEntity
-  expose :yaml_errors, if: ->(pipeline, _) { pipeline.has_yaml_errors? }
+  expose :yaml_errors, if: -> (pipeline, _) { pipeline.has_yaml_errors? }
 
-  expose :retry_path, if: proc { can_retry? }  do |pipeline|
+  expose :retry_path, if: -> (*) { can_retry? }  do |pipeline|
     retry_namespace_project_pipeline_path(pipeline.project.namespace,
                                           pipeline.project,
                                           pipeline.id)
   end
 
-  expose :cancel_path, if: proc { can_cancel? } do |pipeline|
+  expose :cancel_path, if: -> (*) { can_cancel? } do |pipeline|
     cancel_namespace_project_pipeline_path(pipeline.project.namespace,
                                            pipeline.project,
                                            pipeline.id)
diff --git a/app/services/ci/play_build_service.rb b/app/services/ci/play_build_service.rb
index c9ed45408f223..5b3ff1ced3855 100644
--- a/app/services/ci/play_build_service.rb
+++ b/app/services/ci/play_build_service.rb
@@ -5,7 +5,7 @@ def execute(build)
         raise Gitlab::Access::AccessDeniedError
       end
 
-      # Try to enqueue thebuild, otherwise create a duplicate.
+      # Try to enqueue the build, otherwise create a duplicate.
       #
       if build.enqueue
         build.tap { |action| action.update(user: current_user) }
diff --git a/spec/controllers/projects/builds_controller_spec.rb b/spec/controllers/projects/builds_controller_spec.rb
index 3bf03c88ff55a..3ce23c17cdc68 100644
--- a/spec/controllers/projects/builds_controller_spec.rb
+++ b/spec/controllers/projects/builds_controller_spec.rb
@@ -260,10 +260,8 @@ def post_retry
   end
 
   describe 'POST play' do
-    let(:project) { create(:project, :public) }
-
     before do
-      project.add_developer(user)
+      project.add_master(user)
       sign_in(user)
 
       post_play
-- 
GitLab


From 88ace4a7ad0809d64f688dc1ee6fc8ea6cda9045 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 3 May 2017 12:35:13 +0200
Subject: [PATCH 151/363] Rephrase documentation for protected actions feature

---
 doc/ci/yaml/README.md | 7 ++++---
 1 file changed, 4 insertions(+), 3 deletions(-)

diff --git a/doc/ci/yaml/README.md b/doc/ci/yaml/README.md
index 0cab3270d94dc..16308a957cb31 100644
--- a/doc/ci/yaml/README.md
+++ b/doc/ci/yaml/README.md
@@ -580,9 +580,10 @@ Optional manual actions have `allow_failure: true` set by default.
 
 **Statuses of optional actions do not contribute to overall pipeline status.**
 
-**Manual actions do inherit permissions of protected branches. In other words,
-in order to trigger a manual action assigned to a branch that the pipeline is
-running for, user needs to have ability to push to this branch.**
+**Manual actions are considered to be write actions, so permissions for
+protected branches are used when user wants to trigger an action. In other
+words, in order to trigger a manual action assigned to a branch that the
+pipeline is running for, user needs to have ability to push to this branch.**
 
 ### environment
 
-- 
GitLab


From 93f27958d02389b3550d25470ccd29b304005e96 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 3 May 2017 12:02:26 +0100
Subject: [PATCH 152/363] Updated some JS translate methods to correctly take
 in context

---
 .../components/limit_warning_component.js     |  2 +-
 .../components/total_time_component.js        |  2 +-
 app/assets/javascripts/locale/de/app.js       |  2 +-
 app/assets/javascripts/locale/en/app.js       |  2 +-
 app/assets/javascripts/locale/es/app.js       |  2 +-
 app/assets/javascripts/locale/index.js        | 21 +++++++++++--------
 .../javascripts/vue_shared/translate.js       | 12 ++++++++---
 .../cycle_analytics/_no_access.html.haml      |  2 +-
 .../projects/cycle_analytics/show.html.haml   | 13 ++++++------
 locale/de/gitlab.po                           | 17 ++++++++-------
 locale/en/gitlab.po                           | 17 ++++++++-------
 locale/es/gitlab.po                           | 18 +++++++++-------
 locale/gitlab.pot                             | 21 ++++++++++---------
 13 files changed, 72 insertions(+), 59 deletions(-)

diff --git a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
index c43a45067553b..8d3d34f836f5e 100644
--- a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
@@ -9,7 +9,7 @@ export default {
     <span v-if="count === 50" class="events-info pull-right">
       <i class="fa fa-warning has-tooltip"
           aria-hidden="true"
-          :title="__('Limited to showing 50 events at most')"
+          :title="n__('Limited to showing %d event at most', 'Limited to showing %d events at most', 50)"
           data-placement="top"></i>
       {{ n__('Showing %d event', 'Showing %d events', 50) }}
     </span>
diff --git a/app/assets/javascripts/cycle_analytics/components/total_time_component.js b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
index 1eef7d9a8738d..d5e6167b2a82f 100644
--- a/app/assets/javascripts/cycle_analytics/components/total_time_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
@@ -15,7 +15,7 @@ global.cycleAnalytics.TotalTimeComponent = Vue.extend({
         <template v-if="time.days">{{ time.days }} <span>{{ n__('day', 'days', time.days) }}</span></template>
         <template v-if="time.hours">{{ time.hours }} <span>{{ n__('Time|hr', 'Time|hrs', time.hours) }}</span></template>
         <template v-if="time.mins && !time.days">{{ time.mins }} <span>{{ n__('Time|min', 'Time|mins', time.mins) }}</span></template>
-        <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>{{ __('Time|s') }}</span></template>
+        <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>{{ s__('Time|s') }}</span></template>
       </template>
       <template v-else>
         --
diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index 51c5ad512fc7a..a7922e911bdec 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index 68beb0afda016..3ed43559e74ee 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last 30 days":[""],"Last 90 days":[""],"Limited to showing 50 events at most":[""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index 511c70b30a6ca..b91dcbd1ff36c 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-02 23:17-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":["por"],"Commits":["Cambios"],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Cycle|Code":["Código"],"Cycle|Issue":["Tema"],"Cycle|Plan":["Planificación"],"Cycle|Review":["Revisión"],"Cycle|Staging":["Etapa"],"Cycle|Test":["Pruebas"],"Deploys":["Despliegues"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last 30 days":["Últimos 30 días"],"Last 90 days":["Últimos 90 días"],"Limited to showing 50 events at most":["Limitado a mostrar máximo 50 eventos"],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OfFirstTime|First":["Primer"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"],"pushed by":["enviado por"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-02 23:17-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":["por"],"Commits":["Cambios"],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Cycle|Code":["Código"],"Cycle|Issue":["Tema"],"Cycle|Plan":["Planificación"],"Cycle|Review":["Revisión"],"Cycle|Staging":["Etapa"],"Cycle|Test":["Pruebas"],"Deploys":["Despliegues"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OfFirstTime|First":["Primer"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"],"pushed by":["enviado por"]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/index.js b/app/assets/javascripts/locale/index.js
index ca3ed69fbb3ba..3b23e3a69d491 100644
--- a/app/assets/javascripts/locale/index.js
+++ b/app/assets/javascripts/locale/index.js
@@ -1,15 +1,18 @@
 import Jed from 'jed';
-import { de } from './de/app';
-import { es } from './es/app';
-import { en } from './en/app';
 
-const locales = {
-  de,
-  es,
-  en,
-};
+function requireAll(requireContext) { return requireContext.keys().map(requireContext); }
+
+const allLocales = requireAll(require.context('./', true, /^(?!.*(?:index.js$)).*\.js$/));
+const locales = allLocales.reduce((d, obj) => {
+  const data = d;
+  const localeKey = Object.keys(obj)[0];
+
+  data[localeKey] = obj[localeKey];
+
+  return data;
+}, {});
 
-const lang = document.querySelector('html').getAttribute('lang') || 'en';
+const lang = document.querySelector('html').getAttribute('lang').replace(/-/g, '_') || 'en';
 const locale = new Jed(locales[lang]);
 const gettext = locale.gettext.bind(locale);
 const ngettext = locale.ngettext.bind(locale);
diff --git a/app/assets/javascripts/vue_shared/translate.js b/app/assets/javascripts/vue_shared/translate.js
index c2c20ea0853d3..07ef00c10d06a 100644
--- a/app/assets/javascripts/vue_shared/translate.js
+++ b/app/assets/javascripts/vue_shared/translate.js
@@ -1,7 +1,6 @@
 import {
   __,
   n__,
-  s__,
 } from '../locale';
 
 export default (Vue) => {
@@ -9,9 +8,16 @@ export default (Vue) => {
     methods: {
       __(text) { return __(text); },
       n__(text, pluralText, count) {
-        return n__(text, pluralText, count).replace(/%d/g, count);
+        const translated = n__(text, pluralText, count).replace(/%d/g, count).split('|');
+        return translated[translated.length - 1];
+      },
+      s__(keyOrContext, key) {
+        const normalizedKey = key ? `${keyOrContext}|${key}` : keyOrContext;
+        // eslint-disable-next-line no-underscore-dangle
+        const translated = this.__(normalizedKey).split('|');
+
+        return translated[translated.length - 1];
       },
-      s__(context, key) { return s__(context, key); },
     },
   });
 };
diff --git a/app/views/projects/cycle_analytics/_no_access.html.haml b/app/views/projects/cycle_analytics/_no_access.html.haml
index dcfd1b7afc09d..c3eda39823422 100644
--- a/app/views/projects/cycle_analytics/_no_access.html.haml
+++ b/app/views/projects/cycle_analytics/_no_access.html.haml
@@ -4,4 +4,4 @@
       = custom_icon ('icon_lock')
     %h4 {{ __('You need permission.') }}
     %p
-      {{ __('Want to see the data? Please ask administrator for access.') }}
+      {{ __('Want to see the data? Please ask an administrator for access.') }}
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index a8ce021e05f7a..376b1369405c5 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -20,8 +20,7 @@
           %p
             {{ __('Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.') }}
 
-          = link_to help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn' do
-            {{ __('Read more') }}
+          = link_to _('Read more'), help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn'
   = icon("spinner spin", "v-show" => "isLoading")
   .wrapper{ "v-show" => "!isLoading && !hasError" }
     .panel.panel-default
@@ -32,19 +31,19 @@
           .row
             .col-sm-3.col-xs-12.column{ "v-for" => "item in state.summary" }
               %h3.header {{ item.value }}
-              %p.text {{ __(item.title) }}
+              %p.text {{ s__('Cycle', item.title) }}
             .col-sm-3.col-xs-12.column
               .dropdown.inline.js-ca-dropdown
                 %button.dropdown-menu-toggle{ "data-toggle" => "dropdown", :type => "button" }
-                  %span.dropdown-label {{ __('Last 30 days') }}
+                  %span.dropdown-label {{ n__('Last %d day', 'Last %d days', 30) }}
                   %i.fa.fa-chevron-down
                 %ul.dropdown-menu.dropdown-menu-align-right
                   %li
                     %a{ "href" => "#", "data-value" => "30" }
-                      {{ __('Last 30 days') }}
+                      {{ n__('Last %d day', 'Last %d days', 30) }}
                   %li
                     %a{ "href" => "#", "data-value" => "90" }
-                      {{ __('Last 90 days') }}
+                      {{ n__('Last %d day', 'Last %d days', 90) }}
     .stage-panel-container
       .panel.panel-default.stage-panel
         .panel-heading
@@ -52,7 +51,7 @@
             %ul
               %li.stage-header
                 %span.stage-name
-                  {{ __('ProjectLifecycle|Stage') }}
+                  {{ s__('ProjectLifecycle|Stage') }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The phase of the development lifecycle."), "aria-hidden" => "true" }
               %li.median-header
                 %span.stage-name
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index beeca35b879dc..d2ea50e5a7835 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -53,14 +53,15 @@ msgstr ""
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
-msgid "Last 30 days"
-msgstr ""
-
-msgid "Last 90 days"
-msgstr ""
+msgid "Last %d day"
+msgid_plural "Last %d days"
+msgstr[0] ""
+msgstr[1] ""
 
-msgid "Limited to showing 50 events at most"
-msgstr ""
+msgid "Limited to showing %d event at most"
+msgid_plural "Limited to showing %d events at most"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Median"
 msgstr ""
@@ -161,7 +162,7 @@ msgstr ""
 msgid "Total Time"
 msgstr ""
 
-msgid "Want to see the data? Please ask administrator for access."
+msgid "Want to see the data? Please ask an administrator for access."
 msgstr ""
 
 msgid "We don't have enough data to show this stage."
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 9596531fdbd0c..f473ce2e0dcd4 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -53,14 +53,15 @@ msgstr ""
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
-msgid "Last 30 days"
-msgstr ""
-
-msgid "Last 90 days"
-msgstr ""
+msgid "Last %d day"
+msgid_plural "Last %d days"
+msgstr[0] ""
+msgstr[1] ""
 
-msgid "Limited to showing 50 events at most"
-msgstr ""
+msgid "Limited to showing %d event at most"
+msgid_plural "Limited to showing %d events at most"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Median"
 msgstr ""
@@ -161,7 +162,7 @@ msgstr ""
 msgid "Total Time"
 msgstr ""
 
-msgid "Want to see the data? Please ask administrator for access."
+msgid "Want to see the data? Please ask an administrator for access."
 msgstr ""
 
 msgid "We don't have enough data to show this stage."
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 56f889c7141b4..91135a983f653 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -59,14 +59,15 @@ msgstr "Despliegues"
 msgid "Introducing Cycle Analytics"
 msgstr "Introducción a Cycle Analytics"
 
-msgid "Last 30 days"
-msgstr "Últimos 30 días"
-
-msgid "Last 90 days"
-msgstr "Últimos 90 días"
+#, fuzzy
+msgid "Last %d day"
+msgid_plural "Last %d days"
+msgstr[0] "Últimos %d días"
 
-msgid "Limited to showing 50 events at most"
-msgstr "Limitado a mostrar máximo 50 eventos"
+#, fuzzy
+msgid "Limited to showing %d event at most"
+msgid_plural "Limited to showing %d events at most"
+msgstr[0] "Limitado a mostrar máximo %d eventos"
 
 msgid "Median"
 msgstr "Mediana"
@@ -170,7 +171,8 @@ msgstr ""
 msgid "Total Time"
 msgstr "Tiempo Total"
 
-msgid "Want to see the data? Please ask administrator for access."
+#, fuzzy
+msgid "Want to see the data? Please ask an administrator for access."
 msgstr "¿Quieres ver los datos? Por favor pedir acceso al administrador."
 
 msgid "We don't have enough data to show this stage."
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index 956e05926e10b..bc16b7a0968c4 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-05-02 23:33-0500\n"
-"PO-Revision-Date: 2017-05-02 23:33-0500\n"
+"POT-Creation-Date: 2017-05-03 12:00+0100\n"
+"PO-Revision-Date: 2017-05-03 12:00+0100\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -54,14 +54,15 @@ msgstr ""
 msgid "Introducing Cycle Analytics"
 msgstr ""
 
-msgid "Last 30 days"
-msgstr ""
-
-msgid "Last 90 days"
-msgstr ""
+msgid "Last %d day"
+msgid_plural "Last %d days"
+msgstr[0] ""
+msgstr[1] ""
 
-msgid "Limited to showing 50 events at most"
-msgstr ""
+msgid "Limited to showing %d event at most"
+msgid_plural "Limited to showing %d events at most"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Median"
 msgstr ""
@@ -162,7 +163,7 @@ msgstr ""
 msgid "Total Time"
 msgstr ""
 
-msgid "Want to see the data? Please ask administrator for access."
+msgid "Want to see the data? Please ask an administrator for access."
 msgstr ""
 
 msgid "We don't have enough data to show this stage."
-- 
GitLab


From 9c78b17a113c593d5d1d2370688ac814a4d0f3a2 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 3 May 2017 13:33:12 +0100
Subject: [PATCH 153/363] Fixed karma failure

---
 app/assets/javascripts/locale/index.js | 4 +++-
 1 file changed, 3 insertions(+), 1 deletion(-)

diff --git a/app/assets/javascripts/locale/index.js b/app/assets/javascripts/locale/index.js
index 3b23e3a69d491..c9df5f3723af7 100644
--- a/app/assets/javascripts/locale/index.js
+++ b/app/assets/javascripts/locale/index.js
@@ -12,7 +12,9 @@ const locales = allLocales.reduce((d, obj) => {
   return data;
 }, {});
 
-const lang = document.querySelector('html').getAttribute('lang').replace(/-/g, '_') || 'en';
+let lang = document.querySelector('html').getAttribute('lang') || 'en';
+lang = lang.replace(/-/g, '_');
+
 const locale = new Jed(locales[lang]);
 const gettext = locale.gettext.bind(locale);
 const ngettext = locale.ngettext.bind(locale);
-- 
GitLab


From 7ac89d05dbf6c932e9cb6e87cc64c46e015a22d5 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Wed, 3 May 2017 14:53:54 +0200
Subject: [PATCH 154/363] add service spec

---
 spec/models/service_spec.rb | 26 ++++++++++++++++++++++++++
 1 file changed, 26 insertions(+)

diff --git a/spec/models/service_spec.rb b/spec/models/service_spec.rb
index 134882648b9b4..e9acbb81ad99e 100644
--- a/spec/models/service_spec.rb
+++ b/spec/models/service_spec.rb
@@ -254,4 +254,30 @@
       end
     end
   end
+
+  describe "#update_and_propagate" do
+    let!(:service) do
+      RedmineService.new(
+        project: project,
+        active: false,
+        properties: {
+          project_url: 'http://redmine/projects/project_name_in_redmine',
+          issues_url: "http://redmine/#{project.id}/project_name_in_redmine/:id",
+          new_issue_url: 'http://redmine/projects/project_name_in_redmine/issues/new'
+        }
+      )
+    end
+
+    it 'updates the service params successfully and calls the propagation worker' do
+      expect(PropagateProjectServiceWorker).to receve(:perform_async)
+
+      expect(service.update_and_propagate(active: true)).to be true
+    end
+
+    it 'updates the service params successfully' do
+      expect(PropagateProjectServiceWorker).not_to receve(:perform_asyncs)
+
+      expect(service.update_and_propagate(properties: {})).to be true
+    end
+  end
 end
-- 
GitLab


From 36c6447f9de74039f374a49f9c29ce370fb45ee2 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 3 May 2017 14:28:42 +0100
Subject: [PATCH 155/363] [ci skip] remove accidental debug print

---
 app/views/layouts/_head.html.haml | 2 --
 1 file changed, 2 deletions(-)

diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index fa4d760a20eb1..afcc2b6e4f31f 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -35,8 +35,6 @@
   = webpack_bundle_tag "main"
   = webpack_bundle_tag "raven" if current_application_settings.clientside_sentry_enabled
 
-  = current_application_settings.clientside_sentry_enabled
-
   - if content_for?(:page_specific_javascripts)
     = yield :page_specific_javascripts
 
-- 
GitLab


From 83154f21542ec04076d20ce9c4a8997d55fc5f43 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 3 May 2017 15:42:29 +0200
Subject: [PATCH 156/363] Improve environment policy class

---
 app/policies/environment_policy.rb | 12 +++++++-----
 1 file changed, 7 insertions(+), 5 deletions(-)

diff --git a/app/policies/environment_policy.rb b/app/policies/environment_policy.rb
index c0c9bbf2d9e9d..cc94d4a7e058e 100644
--- a/app/policies/environment_policy.rb
+++ b/app/policies/environment_policy.rb
@@ -4,12 +4,14 @@ class EnvironmentPolicy < BasePolicy
   def rules
     delegate! environment.project
 
-    if environment.stop_action?
-      delegate! environment.stop_action
+    if can?(:create_deployment) && environment.stop_action?
+      can! :stop_environment if can_play_stop_action?
     end
+  end
 
-    if can?(:create_deployment) && can?(:play_build)
-      can! :stop_environment
-    end
+  private
+
+  def can_play_stop_action?
+    Ability.allowed?(user, :play_build, environment.stop_action)
   end
 end
-- 
GitLab


From e81ea165ba738fedab07d5e20423856e004e2594 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Wed, 3 May 2017 15:43:26 +0200
Subject: [PATCH 157/363] added worker spec

---
 .../propagate_project_service_worker_spec.rb  | 36 +++++++++++++++++++
 1 file changed, 36 insertions(+)
 create mode 100644 spec/workers/propagate_project_service_worker_spec.rb

diff --git a/spec/workers/propagate_project_service_worker_spec.rb b/spec/workers/propagate_project_service_worker_spec.rb
new file mode 100644
index 0000000000000..ce01a663a8fa6
--- /dev/null
+++ b/spec/workers/propagate_project_service_worker_spec.rb
@@ -0,0 +1,36 @@
+require 'spec_helper'
+
+describe PruneOldEventsWorker do
+  describe '#perform' do
+    let!(:service_template) do
+      PushoverService.create(
+        template: true,
+        properties: {
+          device: 'MyDevice',
+          sound: 'mic',
+          priority: 4,
+          api_key: '123456789'
+        })
+    end
+
+    let!(:project) { create(:empty_project) }
+
+    it 'creates services for projects' do
+      expect { subject.perform }.to change { Service.count }.by(1)
+    end
+
+    it 'does not create the service if it exists already' do
+      Service.build_from_template(project.id, service_template).save!
+
+      expect { subject.perform }.not_to change { Service.count }
+    end
+
+    it 'creates the service containing the template attributes' do
+      subject.perform
+
+      service = Service.find_by(service_template.merge(project_id: project.id, template: false))
+
+      expect(service).not_to be_nil
+    end
+  end
+end
-- 
GitLab


From b0dfc741c499ec72c5cad53fc1bb1ed65ee38e70 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 3 May 2017 14:47:21 +0100
Subject: [PATCH 158/363] [ci skip] Removed substring

---
 app/assets/javascripts/raven/raven_config.js | 2 +-
 spec/javascripts/raven/raven_config_spec.js  | 6 +++---
 2 files changed, 4 insertions(+), 4 deletions(-)

diff --git a/app/assets/javascripts/raven/raven_config.js b/app/assets/javascripts/raven/raven_config.js
index 43e55bc541512..c7fe1cacf49ea 100644
--- a/app/assets/javascripts/raven/raven_config.js
+++ b/app/assets/javascripts/raven/raven_config.js
@@ -85,7 +85,7 @@ const RavenConfig = {
         url: config.url,
         data: config.data,
         status: req.status,
-        response: responseText.substring(0, 100),
+        response: responseText,
         error,
         event,
       },
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index b4c35420f9094..7b3bef3162d39 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -1,7 +1,7 @@
 import Raven from 'raven-js';
 import RavenConfig from '~/raven/raven_config';
 
-describe('RavenConfig', () => {
+fdescribe('RavenConfig', () => {
   describe('IGNORE_ERRORS', () => {
     it('should be an array of strings', () => {
       const areStrings = RavenConfig.IGNORE_ERRORS.every(error => typeof error === 'string');
@@ -190,7 +190,7 @@ describe('RavenConfig', () => {
           url: config.url,
           data: config.data,
           status: req.status,
-          response: req.responseText.substring(0, 100),
+          response: req.responseText,
           error: err,
           event,
         },
@@ -211,7 +211,7 @@ describe('RavenConfig', () => {
             url: config.url,
             data: config.data,
             status: req.status,
-            response: req.responseText.substring(0, 100),
+            response: req.responseText,
             error: req.statusText,
             event,
           },
-- 
GitLab


From 4fed7036aaf25dcf4fb4ff9660215e397654fe4c Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 3 May 2017 14:48:10 +0100
Subject: [PATCH 159/363] [ci skip] Moved the required restart message to only
 backend sentry field

---
 app/views/admin/application_settings/_form.html.haml | 3 +--
 1 file changed, 1 insertion(+), 2 deletions(-)

diff --git a/app/views/admin/application_settings/_form.html.haml b/app/views/admin/application_settings/_form.html.haml
index 6030c8b1dfa25..4b6628169ef55 100644
--- a/app/views/admin/application_settings/_form.html.haml
+++ b/app/views/admin/application_settings/_form.html.haml
@@ -394,8 +394,6 @@
 
   %fieldset
     %legend Error Reporting and Logging
-    %p
-      These settings require a restart to take effect.
     .form-group
       .col-sm-offset-2.col-sm-10
         .checkbox
@@ -403,6 +401,7 @@
             = f.check_box :sentry_enabled
             Enable Sentry
           .help-block
+            %p This setting requires a restart to take effect.
             Sentry is an error reporting and logging tool which is currently not shipped with GitLab, get it here:
             %a{ href: 'https://getsentry.com', target: '_blank', rel: 'noopener noreferrer' } https://getsentry.com
 
-- 
GitLab


From f3732e167b3d1ab58cbd49261ef7624959387cab Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 3 May 2017 14:51:53 +0100
Subject: [PATCH 160/363] [ci skip] Update migration to be reversible and use
 add_column_with_default

---
 ...ientside_sentry_to_application_settings.rb | 37 ++++++++++++++-----
 1 file changed, 28 insertions(+), 9 deletions(-)

diff --git a/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb b/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
index 380060b18b391..f3e08aed81988 100644
--- a/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
+++ b/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
@@ -1,12 +1,31 @@
-# rubocop:disable all
 class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
-  DOWNTIME = true
-  DOWNTIME_REASON = 'This migration requires downtime because we must add 2 new columns, 1 of which has a default value.'
-
-  def change
-    change_table :application_settings do |t|
-      t.boolean :clientside_sentry_enabled, default: false
-      t.string :clientside_sentry_dsn
-    end
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using the methods "add_concurrent_index" or "add_column_with_default"
+  # you must disable the use of transactions as these methods can not run in an
+  # existing transaction. When using "add_concurrent_index" make sure that this
+  # method is the _only_ method called in the migration, any other changes
+  # should go in a separate migration. This ensures that upon failure _only_ the
+  # index creation fails and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  disable_ddl_transaction!
+
+  def up
+    add_column_with_default :application_settings, :clientside_sentry_enabled, :boolean, default: false
+    add_column :application_settings, :clientside_sentry_dsn, :string
+  end
+
+  def down
+    remove_columns :application_settings, :clientside_sentry_enabled, :clientside_sentry_dsn
   end
 end
-- 
GitLab


From 555a3cf5bbfb78d697c70339bc65047d90e26d95 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 3 May 2017 14:53:46 +0100
Subject: [PATCH 161/363] Updated schema.rb with patch mode

---
 db/schema.rb | 13 ++++++-------
 1 file changed, 6 insertions(+), 7 deletions(-)

diff --git a/db/schema.rb b/db/schema.rb
index c3fbb74e878c9..ea303678ac02e 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -63,7 +63,6 @@
     t.boolean "shared_runners_enabled", default: true, null: false
     t.integer "max_artifacts_size", default: 100, null: false
     t.string "runners_registration_token"
-    t.integer "max_pages_size", default: 100, null: false
     t.boolean "require_two_factor_authentication", default: false
     t.integer "two_factor_grace_period", default: 48
     t.boolean "metrics_enabled", default: false
@@ -112,16 +111,17 @@
     t.boolean "html_emails_enabled", default: true
     t.string "plantuml_url"
     t.boolean "plantuml_enabled"
+    t.integer "max_pages_size", default: 100, null: false
     t.integer "terminal_max_session_time", default: 0, null: false
     t.string "default_artifacts_expire_in", default: "0", null: false
     t.integer "unique_ips_limit_per_user"
     t.integer "unique_ips_limit_time_window"
     t.boolean "unique_ips_limit_enabled", default: false, null: false
     t.decimal "polling_interval_multiplier", default: 1.0, null: false
-    t.integer "cached_markdown_version"
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
-    t.boolean "clientside_sentry_enabled", default: false
+    t.integer "cached_markdown_version"
+    t.boolean "clientside_sentry_enabled", default: false, null: false
     t.string "clientside_sentry_dsn"
   end
 
@@ -735,8 +735,8 @@
     t.integer "visibility_level", default: 20, null: false
     t.boolean "request_access_enabled", default: false, null: false
     t.datetime "deleted_at"
-    t.text "description_html"
     t.boolean "lfs_enabled"
+    t.text "description_html"
     t.integer "parent_id"
     t.boolean "require_two_factor_authentication", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
@@ -969,9 +969,9 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.integer "cached_markdown_version"
   end
 
@@ -1074,7 +1074,6 @@
     t.string "line_code"
     t.string "note_type"
     t.text "position"
-    t.string "in_reply_to_discussion_id"
   end
 
   add_index "sent_notifications", ["reply_key"], name: "index_sent_notifications_on_reply_key", unique: true, using: :btree
@@ -1324,10 +1323,10 @@
     t.string "organization"
     t.boolean "authorized_projects_populated"
     t.boolean "ghost"
-    t.date "last_activity_on"
     t.boolean "notified_of_own_activity"
     t.boolean "require_two_factor_authentication_from_group", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
+    t.date "last_activity_on"
   end
 
   add_index "users", ["admin"], name: "index_users_on_admin", using: :btree
-- 
GitLab


From 264bf229277caf1c1eaca4e83921ca1b305d5401 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Wed, 3 May 2017 17:20:12 +0200
Subject: [PATCH 162/363] add propagate service worker and updated spec and
 controller

---
 app/controllers/admin/services_controller.rb  |  2 +-
 app/models/service.rb                         | 10 ++++
 .../propagate_project_service_worker.rb       | 49 +++++++++++++++++++
 .../propagate_project_service_worker_spec.rb  | 19 ++++---
 4 files changed, 73 insertions(+), 7 deletions(-)
 create mode 100644 app/workers/propagate_project_service_worker.rb

diff --git a/app/controllers/admin/services_controller.rb b/app/controllers/admin/services_controller.rb
index 37a1a23178eb7..2b6f335cb2b1c 100644
--- a/app/controllers/admin/services_controller.rb
+++ b/app/controllers/admin/services_controller.rb
@@ -15,7 +15,7 @@ def edit
   end
 
   def update
-    if service.update_attributes(service_params[:service])
+    if service.update_and_propagate(service_params[:service])
       redirect_to admin_application_settings_services_path,
         notice: 'Application settings saved successfully'
     else
diff --git a/app/models/service.rb b/app/models/service.rb
index c71a7d169eca7..dea22fd96a7aa 100644
--- a/app/models/service.rb
+++ b/app/models/service.rb
@@ -254,6 +254,16 @@ def self.build_from_template(project_id, template)
     service
   end
 
+  def update_and_propagate(service_params)
+    return false unless update_attributes(service_params)
+
+    if service_params[:active] == 1
+      PropagateProjectServiceWorker.perform_async(service_params[:id])
+    end
+
+    true
+  end
+
   private
 
   def cache_project_has_external_issue_tracker
diff --git a/app/workers/propagate_project_service_worker.rb b/app/workers/propagate_project_service_worker.rb
new file mode 100644
index 0000000000000..535517709682f
--- /dev/null
+++ b/app/workers/propagate_project_service_worker.rb
@@ -0,0 +1,49 @@
+# Worker for updating any project specific caches.
+class PropagateProjectServiceWorker
+  include Sidekiq::Worker
+  include DedicatedSidekiqQueue
+
+  LEASE_TIMEOUT = 30.minutes.to_i
+
+  def perform(template_id)
+    template = Service.find_by(id: template_id)
+
+    return unless template&.active
+    return unless try_obtain_lease_for(template.id)
+
+    Rails.logger.info("Propagating services for template #{template.id}")
+
+    project_ids_for_template(template) do |project_id|
+      Service.build_from_template(project_id, template).save!
+    end
+  end
+
+  private
+
+  def project_ids_for_template(template)
+    limit = 100
+    offset = 0
+
+    loop do
+      batch = project_ids_batch(limit, offset, template.type)
+
+      batch.each { |project_id| yield(project_id) }
+
+      break if batch.count < limit
+
+      offset += limit
+    end
+  end
+
+  def project_ids_batch(limit, offset, template_type)
+    Project.joins('LEFT JOIN services ON services.project_id = projects.id').
+      where('services.type != ? OR services.id IS NULL', template_type).
+      limit(limit).offset(offset).pluck(:id)
+  end
+
+  def try_obtain_lease_for(template_id)
+    Gitlab::ExclusiveLease.
+      new("propagate_project_service_worker:#{template_id}", timeout: LEASE_TIMEOUT).
+      try_obtain
+  end
+end
diff --git a/spec/workers/propagate_project_service_worker_spec.rb b/spec/workers/propagate_project_service_worker_spec.rb
index ce01a663a8fa6..d525a8b4a236b 100644
--- a/spec/workers/propagate_project_service_worker_spec.rb
+++ b/spec/workers/propagate_project_service_worker_spec.rb
@@ -1,36 +1,43 @@
 require 'spec_helper'
 
-describe PruneOldEventsWorker do
+describe PropagateProjectServiceWorker do
   describe '#perform' do
     let!(:service_template) do
       PushoverService.create(
         template: true,
+        active: true,
         properties: {
           device: 'MyDevice',
           sound: 'mic',
           priority: 4,
+          user_key: 'asdf',
           api_key: '123456789'
         })
     end
 
     let!(:project) { create(:empty_project) }
 
+    before do
+      allow_any_instance_of(Gitlab::ExclusiveLease).to receive(:try_obtain).
+        and_return(true)
+    end
+
     it 'creates services for projects' do
-      expect { subject.perform }.to change { Service.count }.by(1)
+      expect { subject.perform(service_template.id) }.to change { Service.count }.by(1)
     end
 
     it 'does not create the service if it exists already' do
       Service.build_from_template(project.id, service_template).save!
 
-      expect { subject.perform }.not_to change { Service.count }
+      expect { subject.perform(service_template.id) }.not_to change { Service.count }
     end
 
     it 'creates the service containing the template attributes' do
-      subject.perform
+      subject.perform(service_template.id)
 
-      service = Service.find_by(service_template.merge(project_id: project.id, template: false))
+      service = Service.find_by(type: service_template.type, template: false)
 
-      expect(service).not_to be_nil
+      expect(service.properties).to eq(service_template.properties)
     end
   end
 end
-- 
GitLab


From 1faf398c5ddae576ef6b1aa63b7fed063c17d4f4 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 3 May 2017 16:46:18 +0100
Subject: [PATCH 163/363] Fixed schema diffs

---
 db/schema.rb | 11 ++++++-----
 1 file changed, 6 insertions(+), 5 deletions(-)

diff --git a/db/schema.rb b/db/schema.rb
index ea303678ac02e..97282e8f97088 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -63,6 +63,7 @@
     t.boolean "shared_runners_enabled", default: true, null: false
     t.integer "max_artifacts_size", default: 100, null: false
     t.string "runners_registration_token"
+    t.integer "max_pages_size", default: 100, null: false
     t.boolean "require_two_factor_authentication", default: false
     t.integer "two_factor_grace_period", default: 48
     t.boolean "metrics_enabled", default: false
@@ -111,16 +112,15 @@
     t.boolean "html_emails_enabled", default: true
     t.string "plantuml_url"
     t.boolean "plantuml_enabled"
-    t.integer "max_pages_size", default: 100, null: false
     t.integer "terminal_max_session_time", default: 0, null: false
     t.string "default_artifacts_expire_in", default: "0", null: false
     t.integer "unique_ips_limit_per_user"
     t.integer "unique_ips_limit_time_window"
     t.boolean "unique_ips_limit_enabled", default: false, null: false
     t.decimal "polling_interval_multiplier", default: 1.0, null: false
+    t.integer "cached_markdown_version"
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
-    t.integer "cached_markdown_version"
     t.boolean "clientside_sentry_enabled", default: false, null: false
     t.string "clientside_sentry_dsn"
   end
@@ -735,8 +735,8 @@
     t.integer "visibility_level", default: 20, null: false
     t.boolean "request_access_enabled", default: false, null: false
     t.datetime "deleted_at"
-    t.boolean "lfs_enabled"
     t.text "description_html"
+    t.boolean "lfs_enabled"
     t.integer "parent_id"
     t.boolean "require_two_factor_authentication", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
@@ -969,9 +969,9 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.integer "cached_markdown_version"
   end
 
@@ -1074,6 +1074,7 @@
     t.string "line_code"
     t.string "note_type"
     t.text "position"
+    t.string "in_reply_to_discussion_id"
   end
 
   add_index "sent_notifications", ["reply_key"], name: "index_sent_notifications_on_reply_key", unique: true, using: :btree
@@ -1323,10 +1324,10 @@
     t.string "organization"
     t.boolean "authorized_projects_populated"
     t.boolean "ghost"
+    t.date "last_activity_on"
     t.boolean "notified_of_own_activity"
     t.boolean "require_two_factor_authentication_from_group", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
-    t.date "last_activity_on"
   end
 
   add_index "users", ["admin"], name: "index_users_on_admin", using: :btree
-- 
GitLab


From e6829ae88eccfdb87b004762dd35f26ab9d212cb Mon Sep 17 00:00:00 2001
From: Eric Eastwood <contact@ericeastwood.com>
Date: Wed, 3 May 2017 10:47:59 -0500
Subject: [PATCH 164/363] Add enableMap to gl.GfmAutoComplete for partial
 re-use

EE backport of
https://gitlab.com/gitlab-org/gitlab-ee/merge_requests/1800
---
 app/assets/javascripts/gfm_auto_complete.js | 169 ++++++++++++--------
 1 file changed, 102 insertions(+), 67 deletions(-)

diff --git a/app/assets/javascripts/gfm_auto_complete.js b/app/assets/javascripts/gfm_auto_complete.js
index 687a462a0d41a..f1b99023c723c 100644
--- a/app/assets/javascripts/gfm_auto_complete.js
+++ b/app/assets/javascripts/gfm_auto_complete.js
@@ -101,9 +101,17 @@ window.gl.GfmAutoComplete = {
       }
     }
   },
-  setup: function(input) {
+  setup: function(input, enableMap = {
+    emojis: true,
+    members: true,
+    issues: true,
+    milestones: true,
+    mergeRequests: true,
+    labels: true
+  }) {
     // Add GFM auto-completion to all input fields, that accept GFM input.
     this.input = input || $('.js-gfm-input');
+    this.enableMap = enableMap;
     this.setupLifecycle();
   },
   setupLifecycle() {
@@ -115,7 +123,84 @@ window.gl.GfmAutoComplete = {
       $input.on('inserted-commands.atwho', $input.trigger.bind($input, 'keyup'));
     });
   },
+
   setupAtWho: function($input) {
+    if (this.enableMap.emojis) this.setupEmoji($input);
+    if (this.enableMap.members) this.setupMembers($input);
+    if (this.enableMap.issues) this.setupIssues($input);
+    if (this.enableMap.milestones) this.setupMilestones($input);
+    if (this.enableMap.mergeRequests) this.setupMergeRequests($input);
+    if (this.enableMap.labels) this.setupLabels($input);
+
+    // We don't instantiate the slash commands autocomplete for note and issue/MR edit forms
+    $input.filter('[data-supports-slash-commands="true"]').atwho({
+      at: '/',
+      alias: 'commands',
+      searchKey: 'search',
+      skipSpecialCharacterTest: true,
+      data: this.defaultLoadingData,
+      displayTpl: function(value) {
+        if (this.isLoading(value)) return this.Loading.template;
+        var tpl = '<li>/${name}';
+        if (value.aliases.length > 0) {
+          tpl += ' <small>(or /<%- aliases.join(", /") %>)</small>';
+        }
+        if (value.params.length > 0) {
+          tpl += ' <small><%- params.join(" ") %></small>';
+        }
+        if (value.description !== '') {
+          tpl += '<small class="description"><i><%- description %></i></small>';
+        }
+        tpl += '</li>';
+        return _.template(tpl)(value);
+      }.bind(this),
+      insertTpl: function(value) {
+        var tpl = "/${name} ";
+        var reference_prefix = null;
+        if (value.params.length > 0) {
+          reference_prefix = value.params[0][0];
+          if (/^[@%~]/.test(reference_prefix)) {
+            tpl += '<%- reference_prefix %>';
+          }
+        }
+        return _.template(tpl)({ reference_prefix: reference_prefix });
+      },
+      suffix: '',
+      callbacks: {
+        sorter: this.DefaultOptions.sorter,
+        filter: this.DefaultOptions.filter,
+        beforeInsert: this.DefaultOptions.beforeInsert,
+        beforeSave: function(commands) {
+          if (gl.GfmAutoComplete.isLoading(commands)) return commands;
+          return $.map(commands, function(c) {
+            var search = c.name;
+            if (c.aliases.length > 0) {
+              search = search + " " + c.aliases.join(" ");
+            }
+            return {
+              name: c.name,
+              aliases: c.aliases,
+              params: c.params,
+              description: c.description,
+              search: search
+            };
+          });
+        },
+        matcher: function(flag, subtext, should_startWithSpace, acceptSpaceBar) {
+          var regexp = /(?:^|\n)\/([A-Za-z_]*)$/gi;
+          var match = regexp.exec(subtext);
+          if (match) {
+            return match[1];
+          } else {
+            return null;
+          }
+        }
+      }
+    });
+    return;
+  },
+
+  setupEmoji($input) {
     // Emoji
     $input.atwho({
       at: ':',
@@ -139,6 +224,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupMembers($input) {
     // Team Members
     $input.atwho({
       at: '@',
@@ -180,6 +268,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupIssues($input) {
     $input.atwho({
       at: '#',
       alias: 'issues',
@@ -208,6 +299,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupMilestones($input) {
     $input.atwho({
       at: '%',
       alias: 'milestones',
@@ -236,6 +330,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupMergeRequests($input) {
     $input.atwho({
       at: '!',
       alias: 'mergerequests',
@@ -264,6 +361,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupLabels($input) {
     $input.atwho({
       at: '~',
       alias: 'labels',
@@ -298,73 +398,8 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
-    // We don't instantiate the slash commands autocomplete for note and issue/MR edit forms
-    $input.filter('[data-supports-slash-commands="true"]').atwho({
-      at: '/',
-      alias: 'commands',
-      searchKey: 'search',
-      skipSpecialCharacterTest: true,
-      data: this.defaultLoadingData,
-      displayTpl: function(value) {
-        if (this.isLoading(value)) return this.Loading.template;
-        var tpl = '<li>/${name}';
-        if (value.aliases.length > 0) {
-          tpl += ' <small>(or /<%- aliases.join(", /") %>)</small>';
-        }
-        if (value.params.length > 0) {
-          tpl += ' <small><%- params.join(" ") %></small>';
-        }
-        if (value.description !== '') {
-          tpl += '<small class="description"><i><%- description %></i></small>';
-        }
-        tpl += '</li>';
-        return _.template(tpl)(value);
-      }.bind(this),
-      insertTpl: function(value) {
-        var tpl = "/${name} ";
-        var reference_prefix = null;
-        if (value.params.length > 0) {
-          reference_prefix = value.params[0][0];
-          if (/^[@%~]/.test(reference_prefix)) {
-            tpl += '<%- reference_prefix %>';
-          }
-        }
-        return _.template(tpl)({ reference_prefix: reference_prefix });
-      },
-      suffix: '',
-      callbacks: {
-        sorter: this.DefaultOptions.sorter,
-        filter: this.DefaultOptions.filter,
-        beforeInsert: this.DefaultOptions.beforeInsert,
-        beforeSave: function(commands) {
-          if (gl.GfmAutoComplete.isLoading(commands)) return commands;
-          return $.map(commands, function(c) {
-            var search = c.name;
-            if (c.aliases.length > 0) {
-              search = search + " " + c.aliases.join(" ");
-            }
-            return {
-              name: c.name,
-              aliases: c.aliases,
-              params: c.params,
-              description: c.description,
-              search: search
-            };
-          });
-        },
-        matcher: function(flag, subtext, should_startWithSpace, acceptSpaceBar) {
-          var regexp = /(?:^|\n)\/([A-Za-z_]*)$/gi;
-          var match = regexp.exec(subtext);
-          if (match) {
-            return match[1];
-          } else {
-            return null;
-          }
-        }
-      }
-    });
-    return;
   },
+
   fetchData: function($input, at) {
     if (this.isLoadingData[at]) return;
     this.isLoadingData[at] = true;
-- 
GitLab


From 5c91113c5b6edc4fa1d63bc161b791c7e84e644d Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 3 May 2017 17:13:02 +0100
Subject: [PATCH 165/363] Fixed karma specs

---
 spec/javascripts/deploy_keys/components/keys_panel_spec.js | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/spec/javascripts/deploy_keys/components/keys_panel_spec.js b/spec/javascripts/deploy_keys/components/keys_panel_spec.js
index 416b80525fc2d..a69b39c35c4f1 100644
--- a/spec/javascripts/deploy_keys/components/keys_panel_spec.js
+++ b/spec/javascripts/deploy_keys/components/keys_panel_spec.js
@@ -30,13 +30,13 @@ describe('Deploy keys panel', () => {
 
     expect(
       vm.$el.querySelector('h5').textContent.trim(),
-    ).toContain('(1)');
+    ).toContain(`(${vm.keys.length})`);
   });
 
   it('renders list of keys', () => {
     expect(
       vm.$el.querySelectorAll('li').length,
-    ).toBe(1);
+    ).toBe(vm.keys.length);
   });
 
   it('renders help box if keys are empty', (done) => {
-- 
GitLab


From 11661d1d65b3b5b9ea71d91e56a7c263a3a4eec1 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=A9my=20Coutable?= <remy@rymai.me>
Date: Thu, 27 Apr 2017 18:55:31 +0200
Subject: [PATCH 166/363] Move retro/kickoff doc to PROCESS.md
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Signed-off-by: Rémy Coutable <remy@rymai.me>
---
 CONTRIBUTING.md | 13 -------------
 PROCESS.md      | 35 ++++++++++++++++++++++++++++++++---
 2 files changed, 32 insertions(+), 16 deletions(-)

diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index c8b4d0250ff77..8cdf314ca4a5c 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -540,19 +540,6 @@ There are a few rules to get your merge request accepted:
    See the instructions in that document for help if your MR fails the
    "license-finder" test with a "Dependencies that need approval" error.
 
-## Changes for Stable Releases
-
-Sometimes certain changes have to be added to an existing stable release.
-Two examples are bug fixes and performance improvements. In these cases the
-corresponding merge request should be updated to have the following:
-
-1. A milestone indicating what release the merge request should be merged into.
-1. The label "Pick into Stable"
-
-This makes it easier for release managers to keep track of what still has to be
-merged and where changes have to be merged into.
-Like all merge requests the target should be master so all bugfixes are in master.
-
 ## Definition of done
 
 If you contribute to GitLab please know that changes involve more than just
diff --git a/PROCESS.md b/PROCESS.md
index 735120a369ebe..e5f6ab2992776 100644
--- a/PROCESS.md
+++ b/PROCESS.md
@@ -1,18 +1,24 @@
+## GitLab Core Team & GitLab Inc. Team Contributing Process
+
+---
+
 <!-- START doctoc generated TOC please keep comment here to allow auto update -->
 <!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
 **Table of Contents**  *generated with [DocToc](https://github.com/thlorenz/doctoc)*
 
-- [GitLab Contributing Process](#gitlab-contributing-process)
+- [GitLab Core Team Contributing Process](#gitlab-core-team-contributing-process)
   - [Purpose of describing the contributing process](#purpose-of-describing-the-contributing-process)
   - [Common actions](#common-actions)
     - [Merge request coaching](#merge-request-coaching)
-  - [Workflow labels](#workflow-labels)
   - [Assigning issues](#assigning-issues)
   - [Be kind](#be-kind)
   - [Feature freeze on the 7th for the release on the 22nd](#feature-freeze-on-the-7th-for-the-release-on-the-22nd)
     - [Between the 1st and the 7th](#between-the-1st-and-the-7th)
     - [On the 7th](#on-the-7th)
     - [After the 7th](#after-the-7th)
+  - [Release retrospective and kickoff](#release-retrospective-and-kickoff)
+    - [Retrospective](#retrospective)
+    - [Kickoff](#kickoff)
   - [Copy & paste responses](#copy-&-paste-responses)
     - [Improperly formatted issue](#improperly-formatted-issue)
     - [Issue report for old version](#issue-report-for-old-version)
@@ -28,7 +34,7 @@
 
 <!-- END doctoc generated TOC please keep comment here to allow auto update -->
 
-# GitLab Core Team Contributing Process
+---
 
 ## Purpose of describing the contributing process
 
@@ -153,6 +159,29 @@ release should have the correct milestone assigned _and_ have the label
 Merge requests without a milestone and this label will
 not be merged into any stable branches.
 
+## Release retrospective and kickoff
+
+### Retrospective
+
+After each release, we have a retrospective call where we discuss what went well,
+what went wrong, and what we can improve for the next release. The
+[retrospective notes] are public and you are invited to comment on them.
+If you're interested, you can even join the
+[retrospective call][retro-kickoff-call], on the first working day after the
+22nd at 6pm CET / 9am PST.
+
+### Kickoff
+
+Before working on the next release, we have a
+kickoff call to explain what we expect to ship in the next release. The
+[kickoff notes] are public and you are invited to comment on them.
+If you're interested, you can even join the [kickoff call][retro-kickoff-call],
+on the first working day after the 7th at 6pm CET / 9am PST..
+
+[retrospective notes]: https://docs.google.com/document/d/1nEkM_7Dj4bT21GJy0Ut3By76FZqCfLBmFQNVThmW2TY/edit?usp=sharing
+[kickoff notes]: https://docs.google.com/document/d/1ElPkZ90A8ey_iOkTvUs_ByMlwKK6NAB2VOK5835wYK0/edit?usp=sharing
+[retro-kickoff-call]: https://gitlab.zoom.us/j/918821206
+
 ## Copy & paste responses
 
 ### Improperly formatted issue
-- 
GitLab


From 0374e22a1d5e8a9555106e60804ca6bb72843443 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=A9my=20Coutable?= <remy@rymai.me>
Date: Fri, 28 Apr 2017 20:03:18 +0200
Subject: [PATCH 167/363] Improve the merge request guidelines and DoD in
 CONTRIBUTING.md
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Signed-off-by: Rémy Coutable <remy@rymai.me>
---
 CONTRIBUTING.md | 26 +++++++++++++++-----------
 1 file changed, 15 insertions(+), 11 deletions(-)

diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index 8cdf314ca4a5c..bd1e0d7f2f512 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -439,13 +439,17 @@ request is as follows:
      "Description" field.
   1. If you are contributing documentation, choose `Documentation` from the
      "Choose a template" menu and fill in the template.
+  1. Mention the issue(s) your merge request solves, using the `Solves #XXX` or
+    `Closes #XXX` syntax to auto-close the issue(s) once the merge request will
+    be merged.
+1. If you're allowed to, set a relevant milestone and labels
 1. If the MR changes the UI it should include *Before* and *After* screenshots
 1. If the MR changes CSS classes please include the list of affected pages,
    `grep css-class ./app -R`
-1. Link any relevant [issues][ce-tracker] in the merge request description and
-   leave a comment on them with a link back to the MR
 1. Be prepared to answer questions and incorporate feedback even if requests
    for this arrive weeks or months after your MR submission
+  1. If a discussion has been addressed, select the "Resolve discussion" button
+    beneath it to mark it resolved.
 1. If your MR touches code that executes shell commands, reads or opens files or
    handles paths to files on disk, make sure it adheres to the
    [shell command guidelines](doc/development/shell_commands.md)
@@ -528,8 +532,7 @@ There are a few rules to get your merge request accepted:
 1. If you need polling to support real-time features, please use
    [polling with ETag caching][polling-etag].
 1. Changes after submitting the merge request should be in separate commits
-   (no squashing). If necessary, you will be asked to squash when the review is
-   over, before merging.
+   (no squashing).
 1. It conforms to the [style guides](#style-guides) and the following:
     - If your change touches a line that does not follow the style, modify the
       entire line to follow it. This prevents linting tools from generating warnings.
@@ -548,16 +551,16 @@ the feature you contribute through all of these steps.
 
 1. Description explaining the relevancy (see following item)
 1. Working and clean code that is commented where needed
-1. Unit and integration tests that pass on the CI server
+1. [Unit and system tests][testing] that pass on the CI server
 1. Performance/scalability implications have been considered, addressed, and tested
-1. [Documented][doc-styleguide] in the /doc directory
-1. Changelog entry added
+1. [Documented][doc-styleguide] in the `/doc` directory
+1. [Changelog entry added][changelog]
 1. Reviewed and any concerns are addressed
-1. Merged by the project lead
-1. Added to the release blog article
+1. Merged by a project maintainer
+1. Added to the release blog article if relevant
 1. Added to [the website](https://gitlab.com/gitlab-com/www-gitlab-com/) if relevant
 1. Community questions answered
-1. Answers to questions radiated (in docs/wiki/etc.)
+1. Answers to questions radiated (in docs/wiki/support etc.)
 
 If you add a dependency in GitLab (such as an operating system package) please
 consider updating the following and note the applicability of each in your
@@ -580,7 +583,7 @@ merge request:
     - string literal quoting style **Option A**: single quoted by default
 1.  [Rails](https://github.com/bbatsov/rails-style-guide)
 1.  [Newlines styleguide][newlines-styleguide]
-1.  [Testing](doc/development/testing.md)
+1.  [Testing][testing]
 1.  [JavaScript styleguide][js-styleguide]
 1.  [SCSS styleguide][scss-styleguide]
 1.  [Shell commands](doc/development/shell_commands.md) created by GitLab
@@ -657,6 +660,7 @@ available at [http://contributor-covenant.org/version/1/1/0/](http://contributor
 [license-finder-doc]: doc/development/licensing.md
 [GitLab Inc engineering workflow]: https://about.gitlab.com/handbook/engineering/workflow/#labelling-issues
 [polling-etag]: https://docs.gitlab.com/ce/development/polling.html
+[testing]: doc/development/testing.md
 
 [^1]: Please note that specs other than JavaScript specs are considered backend
       code.
-- 
GitLab


From f644b8c80a51a97e39d2f9422930efaa2b8bc334 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=A9my=20Coutable?= <remy@rymai.me>
Date: Fri, 28 Apr 2017 20:03:47 +0200
Subject: [PATCH 168/363] Improve the Code review guidelines documentation
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Signed-off-by: Rémy Coutable <remy@rymai.me>
---
 CONTRIBUTING.md                | 18 --------------
 doc/development/code_review.md | 44 ++++++++++++++++++++++++++++++----
 2 files changed, 39 insertions(+), 23 deletions(-)

diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index bd1e0d7f2f512..b8df8047a3269 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -485,24 +485,6 @@ Please ensure that your merge request meets the contribution acceptance criteria
 When having your code reviewed and when reviewing merge requests please take the
 [code review guidelines](doc/development/code_review.md) into account.
 
-### Getting your merge request reviewed, approved, and merged
-
-There are a few rules to get your merge request accepted:
-
-1. Your merge request should only be **merged by a [maintainer][team]**.
-  1. If your merge request includes only backend changes [^1], it must be
-    **approved by a [backend maintainer][team]**.
-  1. If your merge request includes only frontend changes [^1], it must be
-    **approved by a [frontend maintainer][team]**.
-  1. If your merge request includes frontend and backend changes [^1], it must
-    be **approved by a [frontend and a backend maintainer][team]**.
-1. To lower the amount of merge requests maintainers need to review, you can
-  ask or assign any [reviewers][team] for a first review.
-  1. If you need some guidance (e.g. it's your first merge request), feel free
-    to ask one of the [Merge request coaches][team].
-  1. The reviewer will assign the merge request to a maintainer once the
-    reviewer is satisfied with the state of the merge request.
-
 ### Contribution acceptance criteria
 
 1. The change is as small as possible
diff --git a/doc/development/code_review.md b/doc/development/code_review.md
index 819578404b6fb..138817f5440f4 100644
--- a/doc/development/code_review.md
+++ b/doc/development/code_review.md
@@ -1,5 +1,25 @@
 # Code Review Guidelines
 
+## Getting your merge request reviewed, approved, and merged
+
+There are a few rules to get your merge request accepted:
+
+1. Your merge request should only be **merged by a [maintainer][team]**.
+  1. If your merge request includes only backend changes [^1], it must be
+    **approved by a [backend maintainer][team]**.
+  1. If your merge request includes only frontend changes [^1], it must be
+    **approved by a [frontend maintainer][team]**.
+  1. If your merge request includes frontend and backend changes [^1], it must
+    be **approved by a [frontend and a backend maintainer][team]**.
+1. To lower the amount of merge requests maintainers need to review, you can
+  ask or assign any [reviewers][team] for a first review.
+  1. If you need some guidance (e.g. it's your first merge request), feel free
+    to ask one of the [Merge request coaches][team].
+  1. The reviewer will assign the merge request to a maintainer once the
+    reviewer is satisfied with the state of the merge request.
+
+## Best practices
+
 This guide contains advice and best practices for performing code review, and
 having your code reviewed.
 
@@ -12,7 +32,7 @@ of colleagues and contributors. However, the final decision to accept a merge
 request is up to one the project's maintainers, denoted on the
 [team page](https://about.gitlab.com/team).
 
-## Everyone
+### Everyone
 
 - Accept that many programming decisions are opinions. Discuss tradeoffs, which
   you prefer, and reach a resolution quickly.
@@ -31,8 +51,11 @@ request is up to one the project's maintainers, denoted on the
 - Consider one-on-one chats or video calls if there are too many "I didn't
   understand" or "Alternative solution:" comments. Post a follow-up comment
   summarizing one-on-one discussion.
+- If you ask a question to a specific person, always start the comment by
+  mentioning them; this will ensure they see it if their notification level is
+  set to "mentioned" and other people will understand they don't have to respond.
 
-## Having your code reviewed
+### Having your code reviewed
 
 Please keep in mind that code review is a process that can take multiple
 iterations, and reviewers may spot things later that they may not have seen the
@@ -50,11 +73,12 @@ first time.
 - Extract unrelated changes and refactorings into future merge requests/issues.
 - Seek to understand the reviewer's perspective.
 - Try to respond to every comment.
+- Let the reviewer select the "Resolve discussion" buttons.
 - Push commits based on earlier rounds of feedback as isolated commits to the
   branch. Do not squash until the branch is ready to merge. Reviewers should be
   able to read individual updates based on their earlier feedback.
 
-## Reviewing code
+### Reviewing code
 
 Understand why the change is necessary (fixes a bug, improves the user
 experience, refactors the existing code). Then:
@@ -69,12 +93,22 @@ experience, refactors the existing code). Then:
   someone else would be confused by it as well.
 - After a round of line notes, it can be helpful to post a summary note such as
   "LGTM :thumbsup:", or "Just a couple things to address."
+- Assign the merge request to the author if changes are required following your
+  review.
+- You should try to resolve merge conflicts yourself, using the [merge conflict
+  resolution][conflict-resolution] tool.
+- Set the milestone before merging a merge request.
 - Avoid accepting a merge request before the job succeeds. Of course, "Merge
   When Pipeline Succeeds" (MWPS) is fine.
 - If you set the MR to "Merge When Pipeline Succeeds", you should take over
   subsequent revisions for anything that would be spotted after that.
+- Consider using the [Squash and
+  merge][squash-and-merge] feature when the merge request has a lot of commits.
+
+[conflict-resolution]: https://docs.gitlab.com/ce/user/project/merge_requests/resolve_conflicts.html#merge-conflict-resolution
+[squash-and-merge]: https://docs.gitlab.com/ee/user/project/merge_requests/squash_and_merge.html#squash-and-merge
 
-## The right balance
+### The right balance
 
 One of the most difficult things during code review is finding the right
 balance in how deep the reviewer can interfere with the code created by a
@@ -100,7 +134,7 @@ reviewee.
   tomorrow. When you are not able to find the right balance, ask other people
   about their opinion.
 
-## Credits
+### Credits
 
 Largely based on the [thoughtbot code review guide].
 
-- 
GitLab


From cb87903c6eec14e42fc23e0488f6c0769ba627d9 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=A9my=20Coutable?= <remy@rymai.me>
Date: Wed, 3 May 2017 17:59:59 +0200
Subject: [PATCH 169/363] Update ToC of CONTRIBUTING.md and PROCESS.md
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Signed-off-by: Rémy Coutable <remy@rymai.me>
---
 CONTRIBUTING.md |  5 ++---
 PROCESS.md      | 49 ++++++++++++++++++++++++-------------------------
 2 files changed, 26 insertions(+), 28 deletions(-)

diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index b8df8047a3269..f27efb0ae85b2 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -22,9 +22,9 @@ _This notice should stay as the first item in the CONTRIBUTING.MD file._
   - [Type labels (~"feature proposal", ~bug, ~customer, etc.)](#type-labels-feature-proposal-bug-customer-etc)
   - [Subject labels (~wiki, ~"container registry", ~ldap, ~api, etc.)](#subject-labels-wiki-container-registry-ldap-api-etc)
   - [Team labels (~CI, ~Discussion, ~Edge, ~Frontend, ~Platform, etc.)](#team-labels-ci-discussion-edge-frontend-platform-etc)
-  - [Priority labels (`Deliverable` and `Stretch`)](#priority-labels-deliverable-and-stretch)
+  - [Priority labels (~Deliverable and ~Stretch)](#priority-labels-deliverable-and-stretch)
   - [Label for community contributors (~"Accepting Merge Requests")](#label-for-community-contributors-accepting-merge-requests)
-- [Implement design & UI elements](#implement-design-&-ui-elements)
+- [Implement design & UI elements](#implement-design--ui-elements)
 - [Issue tracker](#issue-tracker)
   - [Issue triaging](#issue-triaging)
   - [Feature proposals](#feature-proposals)
@@ -35,7 +35,6 @@ _This notice should stay as the first item in the CONTRIBUTING.MD file._
   - [Stewardship](#stewardship)
 - [Merge requests](#merge-requests)
   - [Merge request guidelines](#merge-request-guidelines)
-  - [Getting your merge request reviewed, approved, and merged](#getting-your-merge-request-reviewed-approved-and-merged)
   - [Contribution acceptance criteria](#contribution-acceptance-criteria)
 - [Definition of done](#definition-of-done)
 - [Style guides](#style-guides)
diff --git a/PROCESS.md b/PROCESS.md
index e5f6ab2992776..6d7d155ca6cf1 100644
--- a/PROCESS.md
+++ b/PROCESS.md
@@ -6,31 +6,30 @@
 <!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
 **Table of Contents**  *generated with [DocToc](https://github.com/thlorenz/doctoc)*
 
-- [GitLab Core Team Contributing Process](#gitlab-core-team-contributing-process)
-  - [Purpose of describing the contributing process](#purpose-of-describing-the-contributing-process)
-  - [Common actions](#common-actions)
-    - [Merge request coaching](#merge-request-coaching)
-  - [Assigning issues](#assigning-issues)
-  - [Be kind](#be-kind)
-  - [Feature freeze on the 7th for the release on the 22nd](#feature-freeze-on-the-7th-for-the-release-on-the-22nd)
-    - [Between the 1st and the 7th](#between-the-1st-and-the-7th)
-    - [On the 7th](#on-the-7th)
-    - [After the 7th](#after-the-7th)
-  - [Release retrospective and kickoff](#release-retrospective-and-kickoff)
-    - [Retrospective](#retrospective)
-    - [Kickoff](#kickoff)
-  - [Copy & paste responses](#copy-&-paste-responses)
-    - [Improperly formatted issue](#improperly-formatted-issue)
-    - [Issue report for old version](#issue-report-for-old-version)
-    - [Support requests and configuration questions](#support-requests-and-configuration-questions)
-    - [Code format](#code-format)
-    - [Issue fixed in newer version](#issue-fixed-in-newer-version)
-    - [Improperly formatted merge request](#improperly-formatted-merge-request)
-    - [Inactivity close of an issue](#inactivity-close-of-an-issue)
-    - [Inactivity close of a merge request](#inactivity-close-of-a-merge-request)
-    - [Accepting merge requests](#accepting-merge-requests)
-    - [Only accepting merge requests with green tests](#only-accepting-merge-requests-with-green-tests)
-    - [Closing down the issue tracker on GitHub](#closing-down-the-issue-tracker-on-github)
+- [Purpose of describing the contributing process](#purpose-of-describing-the-contributing-process)
+- [Common actions](#common-actions)
+  - [Merge request coaching](#merge-request-coaching)
+- [Assigning issues](#assigning-issues)
+- [Be kind](#be-kind)
+- [Feature freeze on the 7th for the release on the 22nd](#feature-freeze-on-the-7th-for-the-release-on-the-22nd)
+  - [Between the 1st and the 7th](#between-the-1st-and-the-7th)
+  - [On the 7th](#on-the-7th)
+  - [After the 7th](#after-the-7th)
+- [Release retrospective and kickoff](#release-retrospective-and-kickoff)
+  - [Retrospective](#retrospective)
+  - [Kickoff](#kickoff)
+- [Copy & paste responses](#copy--paste-responses)
+  - [Improperly formatted issue](#improperly-formatted-issue)
+  - [Issue report for old version](#issue-report-for-old-version)
+  - [Support requests and configuration questions](#support-requests-and-configuration-questions)
+  - [Code format](#code-format)
+  - [Issue fixed in newer version](#issue-fixed-in-newer-version)
+  - [Improperly formatted merge request](#improperly-formatted-merge-request)
+  - [Inactivity close of an issue](#inactivity-close-of-an-issue)
+  - [Inactivity close of a merge request](#inactivity-close-of-a-merge-request)
+  - [Accepting merge requests](#accepting-merge-requests)
+  - [Only accepting merge requests with green tests](#only-accepting-merge-requests-with-green-tests)
+  - [Closing down the issue tracker on GitHub](#closing-down-the-issue-tracker-on-github)
 
 <!-- END doctoc generated TOC please keep comment here to allow auto update -->
 
-- 
GitLab


From b6a1658e535ca0c97e9a06a355b5bff90b01bb18 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Wed, 3 May 2017 17:37:52 +0000
Subject: [PATCH 170/363] Remove focused kamra test

---
 spec/javascripts/raven/raven_config_spec.js | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index 7b3bef3162d39..2a2b91fe9480c 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -1,7 +1,7 @@
 import Raven from 'raven-js';
 import RavenConfig from '~/raven/raven_config';
 
-fdescribe('RavenConfig', () => {
+describe('RavenConfig', () => {
   describe('IGNORE_ERRORS', () => {
     it('should be an array of strings', () => {
       const areStrings = RavenConfig.IGNORE_ERRORS.every(error => typeof error === 'string');
-- 
GitLab


From ee65de48d8285ec86b83ad6bb0bf95f23d53c6ce Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 3 May 2017 12:40:21 -0500
Subject: [PATCH 171/363] More translations updates.

---
 app/assets/javascripts/locale/de/app.js       |  2 +-
 app/assets/javascripts/locale/en/app.js       |  2 +-
 app/assets/javascripts/locale/es/app.js       |  2 +-
 .../projects/cycle_analytics/show.html.haml   |  4 +-
 lib/gitlab/cycle_analytics/code_stage.rb      |  2 +-
 lib/gitlab/cycle_analytics/issue_stage.rb     |  2 +-
 lib/gitlab/cycle_analytics/plan_stage.rb      |  2 +-
 .../cycle_analytics/production_stage.rb       |  2 +-
 lib/gitlab/cycle_analytics/review_stage.rb    |  2 +-
 lib/gitlab/cycle_analytics/staging_stage.rb   |  2 +-
 lib/gitlab/cycle_analytics/test_stage.rb      |  2 +-
 locale/de/gitlab.po                           | 19 ++++---
 locale/en/gitlab.po                           | 19 ++++---
 locale/es/gitlab.po                           | 56 +++++++++----------
 locale/gitlab.pot                             | 23 ++++----
 locale/unfound_translations.rb                | 13 +++--
 16 files changed, 79 insertions(+), 75 deletions(-)

diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index a7922e911bdec..e1e4a2b7138f9 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index 3ed43559e74ee..c14e7b41ad137 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"Cycle|Code":[""],"Cycle|Issue":[""],"Cycle|Plan":[""],"Cycle|Review":[""],"Cycle|Staging":[""],"Cycle|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index b91dcbd1ff36c..9e09a28c0e55c 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-02 23:17-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":["por"],"Commits":["Cambios"],"Cycle":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"Cycle|Code":["Código"],"Cycle|Issue":["Tema"],"Cycle|Plan":["Planificación"],"Cycle|Review":["Revisión"],"Cycle|Staging":["Etapa"],"Cycle|Test":["Pruebas"],"Deploys":["Despliegues"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issues":["Nuevos temas"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OfFirstTime|First":["Primer"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Temas Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionadas Originadas por Cambios"],"Relative Deployed Builds":["Builds Desplegadas Relacionadas"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"],"pushed by":["enviado por"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-03 12:28-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":["por"],"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"CycleAnalyticsStage":[""],"CycleAnalyticsStage|Code":["Código"],"CycleAnalyticsStage|Issue":["Incidencia"],"CycleAnalyticsStage|Plan":["Planificación"],"CycleAnalyticsStage|Production":["Producción"],"CycleAnalyticsStage|Review":["Revisión"],"CycleAnalyticsStage|Staging":["Puesta en escena"],"CycleAnalyticsStage|Test":["Pruebas"],"Deploys":["Despliegues"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Último %d día","Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d evento","Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issues":["Nuevas incidencias"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OfFirstTime|First":["Primer"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Incidencias Relacionadas"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionados Originados por Cambios"],"Relative Deployed Builds":["Builds Desplegados Relacionados"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"],"pushed by":["enviado por"]}}};
\ No newline at end of file
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index 376b1369405c5..5c1426e49953b 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -31,7 +31,7 @@
           .row
             .col-sm-3.col-xs-12.column{ "v-for" => "item in state.summary" }
               %h3.header {{ item.value }}
-              %p.text {{ s__('Cycle', item.title) }}
+              %p.text {{ __(item.title) }}
             .col-sm-3.col-xs-12.column
               .dropdown.inline.js-ca-dropdown
                 %button.dropdown-menu-toggle{ "data-toggle" => "dropdown", :type => "button" }
@@ -70,7 +70,7 @@
             %ul
               %li.stage-nav-item{ ':class' => '{ active: stage.active }', '@click' => 'selectStage(stage)', "v-for" => "stage in state.stages" }
                 .stage-nav-item-cell.stage-name
-                  {{ s__('Cycle', stage.title) }}
+                  {{ s__('CycleAnalyticsStage', stage.title) }}
                 .stage-nav-item-cell.stage-median
                   %template{ "v-if" => "stage.isUserAllowed" }
                     %span{ "v-if" => "stage.value" }
diff --git a/lib/gitlab/cycle_analytics/code_stage.rb b/lib/gitlab/cycle_analytics/code_stage.rb
index afb8196484775..182d4c43fbda9 100644
--- a/lib/gitlab/cycle_analytics/code_stage.rb
+++ b/lib/gitlab/cycle_analytics/code_stage.rb
@@ -14,7 +14,7 @@ def name
       end
 
       def legend
-        N_("Related Merge Requests")
+        _("Related Merge Requests")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/issue_stage.rb b/lib/gitlab/cycle_analytics/issue_stage.rb
index d36967d9497f8..08e249116a44b 100644
--- a/lib/gitlab/cycle_analytics/issue_stage.rb
+++ b/lib/gitlab/cycle_analytics/issue_stage.rb
@@ -15,7 +15,7 @@ def name
       end
 
       def legend
-        N_("Related Issues")
+        _("Related Issues")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/plan_stage.rb b/lib/gitlab/cycle_analytics/plan_stage.rb
index 7231c45378874..c5b2809209ced 100644
--- a/lib/gitlab/cycle_analytics/plan_stage.rb
+++ b/lib/gitlab/cycle_analytics/plan_stage.rb
@@ -15,7 +15,7 @@ def name
       end
 
       def legend
-        N_("Related Commits")
+        _("Related Commits")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/production_stage.rb b/lib/gitlab/cycle_analytics/production_stage.rb
index 451720c81214f..8f05c840db195 100644
--- a/lib/gitlab/cycle_analytics/production_stage.rb
+++ b/lib/gitlab/cycle_analytics/production_stage.rb
@@ -16,7 +16,7 @@ def name
       end
 
       def legend
-        N_("Related Issues")
+        _("Related Issues")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/review_stage.rb b/lib/gitlab/cycle_analytics/review_stage.rb
index 08e895acad628..4f4c94f4020d4 100644
--- a/lib/gitlab/cycle_analytics/review_stage.rb
+++ b/lib/gitlab/cycle_analytics/review_stage.rb
@@ -14,7 +14,7 @@ def name
       end
 
       def legend
-        N_("Relative Merged Requests")
+        _("Relative Merged Requests")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/staging_stage.rb b/lib/gitlab/cycle_analytics/staging_stage.rb
index 706a5d14d3f35..da8eb8c55de74 100644
--- a/lib/gitlab/cycle_analytics/staging_stage.rb
+++ b/lib/gitlab/cycle_analytics/staging_stage.rb
@@ -15,7 +15,7 @@ def name
       end
 
       def legend
-        N_("Relative Deployed Builds")
+        _("Relative Deployed Builds")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/test_stage.rb b/lib/gitlab/cycle_analytics/test_stage.rb
index 887cf3e09416f..3ef7132bcc398 100644
--- a/lib/gitlab/cycle_analytics/test_stage.rb
+++ b/lib/gitlab/cycle_analytics/test_stage.rb
@@ -14,7 +14,7 @@ def name
       end
 
       def legend
-        N_("Relative Builds Trigger by Commits")
+        _("Relative Builds Trigger by Commits")
       end
 
       def description
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index d2ea50e5a7835..ea19349bac10e 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -23,28 +23,31 @@ msgstr ""
 msgid "Commits"
 msgstr ""
 
-msgid "Cycle"
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
-msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgid "CycleAnalyticsStage"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Code"
 msgstr ""
 
-msgid "Cycle|Code"
+msgid "CycleAnalyticsStage|Issue"
 msgstr ""
 
-msgid "Cycle|Issue"
+msgid "CycleAnalyticsStage|Plan"
 msgstr ""
 
-msgid "Cycle|Plan"
+msgid "CycleAnalyticsStage|Production"
 msgstr ""
 
-msgid "Cycle|Review"
+msgid "CycleAnalyticsStage|Review"
 msgstr ""
 
-msgid "Cycle|Staging"
+msgid "CycleAnalyticsStage|Staging"
 msgstr ""
 
-msgid "Cycle|Test"
+msgid "CycleAnalyticsStage|Test"
 msgstr ""
 
 msgid "Deploys"
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index f473ce2e0dcd4..2fed2aa1f57e9 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -23,28 +23,31 @@ msgstr ""
 msgid "Commits"
 msgstr ""
 
-msgid "Cycle"
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
-msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgid "CycleAnalyticsStage"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Code"
 msgstr ""
 
-msgid "Cycle|Code"
+msgid "CycleAnalyticsStage|Issue"
 msgstr ""
 
-msgid "Cycle|Issue"
+msgid "CycleAnalyticsStage|Plan"
 msgstr ""
 
-msgid "Cycle|Plan"
+msgid "CycleAnalyticsStage|Production"
 msgstr ""
 
-msgid "Cycle|Review"
+msgid "CycleAnalyticsStage|Review"
 msgstr ""
 
-msgid "Cycle|Staging"
+msgid "CycleAnalyticsStage|Staging"
 msgstr ""
 
-msgid "Cycle|Test"
+msgid "CycleAnalyticsStage|Test"
 msgstr ""
 
 msgid "Deploys"
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 91135a983f653..254b7ad18020d 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-05-02 23:17-0500\n"
+"PO-Revision-Date: 2017-05-03 12:28-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -23,34 +23,32 @@ msgstr "por"
 msgid "Commits"
 msgstr "Cambios"
 
-msgid "Cycle"
-msgstr ""
-
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr "Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."
 
-#, fuzzy
-msgid "Cycle|Code"
+msgid "CycleAnalyticsStage"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Code"
 msgstr "Código"
 
-#, fuzzy
-msgid "Cycle|Issue"
-msgstr "Tema"
+msgid "CycleAnalyticsStage|Issue"
+msgstr "Incidencia"
 
-#, fuzzy
-msgid "Cycle|Plan"
+msgid "CycleAnalyticsStage|Plan"
 msgstr "Planificación"
 
-#, fuzzy
-msgid "Cycle|Review"
+msgid "CycleAnalyticsStage|Production"
+msgstr "Producción"
+
+msgid "CycleAnalyticsStage|Review"
 msgstr "Revisión"
 
 #, fuzzy
-msgid "Cycle|Staging"
-msgstr "Etapa"
+msgid "CycleAnalyticsStage|Staging"
+msgstr "Puesta en escena"
 
-#, fuzzy
-msgid "Cycle|Test"
+msgid "CycleAnalyticsStage|Test"
 msgstr "Pruebas"
 
 msgid "Deploys"
@@ -59,21 +57,21 @@ msgstr "Despliegues"
 msgid "Introducing Cycle Analytics"
 msgstr "Introducción a Cycle Analytics"
 
-#, fuzzy
 msgid "Last %d day"
 msgid_plural "Last %d days"
-msgstr[0] "Últimos %d días"
+msgstr[0] "Último %d día"
+msgstr[1] "Últimos %d días"
 
-#, fuzzy
 msgid "Limited to showing %d event at most"
 msgid_plural "Limited to showing %d events at most"
-msgstr[0] "Limitado a mostrar máximo %d eventos"
+msgstr[0] "Limitado a mostrar máximo %d evento"
+msgstr[1] "Limitado a mostrar máximo %d eventos"
 
 msgid "Median"
 msgstr "Mediana"
 
 msgid "New Issues"
-msgstr "Nuevos temas"
+msgstr "Nuevas incidencias"
 
 msgid "Not available"
 msgstr "No disponible"
@@ -81,7 +79,6 @@ msgstr "No disponible"
 msgid "Not enough data"
 msgstr "No hay suficientes datos"
 
-#, fuzzy
 msgid "OfFirstTime|First"
 msgstr "Primer"
 
@@ -101,18 +98,16 @@ msgid "Related Commits"
 msgstr "Cambios Relacionados"
 
 msgid "Related Issues"
-msgstr "Temas Relacionados"
+msgstr "Incidencias Relacionadas"
 
 msgid "Related Merge Requests"
 msgstr "Solicitudes de fusión Relacionadas"
 
-#, fuzzy
 msgid "Relative Builds Trigger by Commits"
-msgstr "Builds Relacionadas Originadas por Cambios"
+msgstr "Builds Relacionados Originados por Cambios"
 
-#, fuzzy
 msgid "Relative Deployed Builds"
-msgstr "Builds Desplegadas Relacionadas"
+msgstr "Builds Desplegados Relacionados"
 
 msgid "Relative Merged Requests"
 msgstr "Solicitudes de fusión Relacionadas"
@@ -129,7 +124,7 @@ msgid "The collection of events added to the data gathered for that stage."
 msgstr "La colección de eventos agregados a los datos recopilados para esa etapa."
 
 msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
-msgstr "La etapa de temas muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."
+msgstr "La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."
 
 msgid "The phase of the development lifecycle."
 msgstr "La etapa del ciclo de vida del desarrollo."
@@ -138,7 +133,7 @@ msgid "The planning stage shows the time from the previous step to pushing your
 msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."
 
 msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
-msgstr "La etapa de producción muestra el tiempo total que tarda entre la creación de un tema y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."
+msgstr "La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."
 
 msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
 msgstr "La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."
@@ -171,7 +166,6 @@ msgstr ""
 msgid "Total Time"
 msgstr "Tiempo Total"
 
-#, fuzzy
 msgid "Want to see the data? Please ask an administrator for access."
 msgstr "¿Quieres ver los datos? Por favor pedir acceso al administrador."
 
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index bc16b7a0968c4..8f4dc30b8c861 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-05-03 12:00+0100\n"
-"PO-Revision-Date: 2017-05-03 12:00+0100\n"
+"POT-Creation-Date: 2017-05-03 12:32-0500\n"
+"PO-Revision-Date: 2017-05-03 12:32-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -24,28 +24,31 @@ msgstr ""
 msgid "Commits"
 msgstr ""
 
-msgid "Cycle"
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
-msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgid "CycleAnalyticsStage"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Code"
 msgstr ""
 
-msgid "Cycle|Code"
+msgid "CycleAnalyticsStage|Issue"
 msgstr ""
 
-msgid "Cycle|Issue"
+msgid "CycleAnalyticsStage|Plan"
 msgstr ""
 
-msgid "Cycle|Plan"
+msgid "CycleAnalyticsStage|Production"
 msgstr ""
 
-msgid "Cycle|Review"
+msgid "CycleAnalyticsStage|Review"
 msgstr ""
 
-msgid "Cycle|Staging"
+msgid "CycleAnalyticsStage|Staging"
 msgstr ""
 
-msgid "Cycle|Test"
+msgid "CycleAnalyticsStage|Test"
 msgstr ""
 
 msgid "Deploys"
diff --git a/locale/unfound_translations.rb b/locale/unfound_translations.rb
index e38b49b48a17c..e66211c161923 100644
--- a/locale/unfound_translations.rb
+++ b/locale/unfound_translations.rb
@@ -1,9 +1,10 @@
 N_('Commits')
-N_('Cycle|Code')
-N_('Cycle|Issue')
-N_('Cycle|Plan')
-N_('Cycle|Review')
-N_('Cycle|Staging')
-N_('Cycle|Test')
+N_('CycleAnalyticsStage|Code')
+N_('CycleAnalyticsStage|Issue')
+N_('CycleAnalyticsStage|Plan')
+N_('CycleAnalyticsStage|Production')
+N_('CycleAnalyticsStage|Review')
+N_('CycleAnalyticsStage|Staging')
+N_('CycleAnalyticsStage|Test')
 N_('Deploys')
 N_('New Issues')
-- 
GitLab


From b7f01f2b18940eb8dc0fa30e8ebbf1784c864304 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Wed, 3 May 2017 20:52:18 +0100
Subject: [PATCH 172/363] Moved all the text translation manipulation into the
 locale index file Commented the translation methods

---
 app/assets/javascripts/locale/index.js        | 48 +++++++++++++++++--
 .../javascripts/vue_shared/translate.js       | 39 +++++++++++----
 2 files changed, 73 insertions(+), 14 deletions(-)

diff --git a/app/assets/javascripts/locale/index.js b/app/assets/javascripts/locale/index.js
index c9df5f3723af7..7ba676d6d20aa 100644
--- a/app/assets/javascripts/locale/index.js
+++ b/app/assets/javascripts/locale/index.js
@@ -1,5 +1,9 @@
 import Jed from 'jed';
 
+/**
+  This is required to require all the translation folders in the current directory
+  this saves us having to do this manually & keep up to date with new languages
+**/
 function requireAll(requireContext) { return requireContext.keys().map(requireContext); }
 
 const allLocales = requireAll(require.context('./', true, /^(?!.*(?:index.js$)).*\.js$/));
@@ -16,11 +20,47 @@ let lang = document.querySelector('html').getAttribute('lang') || 'en';
 lang = lang.replace(/-/g, '_');
 
 const locale = new Jed(locales[lang]);
+
+/**
+  Translates `text`
+
+  @param text The text to be translated
+  @returns {String} The translated text
+**/
 const gettext = locale.gettext.bind(locale);
-const ngettext = locale.ngettext.bind(locale);
-const pgettext = (context, key) => {
-  const joinedKey = [context, key].join('|');
-  return gettext(joinedKey).split('|').pop();
+
+/**
+  Translate the text with a number
+  if the number is more than 1 it will use the `pluralText` translation.
+  This method allows for contexts, see below re. contexts
+
+  @param text Singular text to translate (eg. '%d day')
+  @param pluralText Plural text to translate (eg. '%d days')
+  @param count Number to decide which translation to use (eg. 2)
+  @returns {String} Translated text with the number replaced (eg. '2 days')
+**/
+const ngettext = (text, pluralText, count) => {
+  const translated = locale.ngettext(text, pluralText, count).replace(/%d/g, count).split('|');
+
+  return translated[translated.length - 1];
+};
+
+/**
+  Translate context based text
+  Either pass in the context translation like `Context|Text to translate`
+  or allow for dynamic text by doing passing in the context first & then the text to translate
+
+  @param keyOrContext Can be either the key to translate including the context
+                      (eg. 'Context|Text') or just the context for the translation
+                      (eg. 'Context')
+  @param key Is the dynamic variable you want to be translated
+  @returns {String} Translated context based text
+**/
+const pgettext = (keyOrContext, key) => {
+  const normalizedKey = key ? `${keyOrContext}|${key}` : keyOrContext;
+  const translated = gettext(normalizedKey).split('|');
+
+  return translated[translated.length - 1];
 };
 
 export { lang };
diff --git a/app/assets/javascripts/vue_shared/translate.js b/app/assets/javascripts/vue_shared/translate.js
index 07ef00c10d06a..ba9656d4e71e8 100644
--- a/app/assets/javascripts/vue_shared/translate.js
+++ b/app/assets/javascripts/vue_shared/translate.js
@@ -1,23 +1,42 @@
 import {
   __,
   n__,
+  s__,
 } from '../locale';
 
 export default (Vue) => {
   Vue.mixin({
     methods: {
+      /**
+        Translates `text`
+
+        @param text The text to be translated
+        @returns {String} The translated text
+      **/
       __(text) { return __(text); },
-      n__(text, pluralText, count) {
-        const translated = n__(text, pluralText, count).replace(/%d/g, count).split('|');
-        return translated[translated.length - 1];
-      },
-      s__(keyOrContext, key) {
-        const normalizedKey = key ? `${keyOrContext}|${key}` : keyOrContext;
-        // eslint-disable-next-line no-underscore-dangle
-        const translated = this.__(normalizedKey).split('|');
+      /**
+        Translate the text with a number
+        if the number is more than 1 it will use the `pluralText` translation.
+        This method allows for contexts, see below re. contexts
+
+        @param text Singular text to translate (eg. '%d day')
+        @param pluralText Plural text to translate (eg. '%d days')
+        @param count Number to decide which translation to use (eg. 2)
+        @returns {String} Translated text with the number replaced (eg. '2 days')
+      **/
+      n__(text, pluralText, count) { return n__(text, pluralText, count); },
+      /**
+        Translate context based text
+        Either pass in the context translation like `Context|Text to translate`
+        or allow for dynamic text by doing passing in the context first & then the text to translate
 
-        return translated[translated.length - 1];
-      },
+        @param keyOrContext Can be either the key to translate including the context
+                            (eg. 'Context|Text') or just the context for the translation
+                            (eg. 'Context')
+        @param key Is the dynamic variable you want to be translated
+        @returns {String} Translated context based text
+      **/
+      s__(keyOrContext, key) { return s__(keyOrContext, key); },
     },
   });
 };
-- 
GitLab


From 93a20ea87261c05e34db708f33441c36d2dc68bd Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Wed, 3 May 2017 14:35:54 -0600
Subject: [PATCH 173/363] fix tasks list spec in rspec

---
 app/views/projects/issues/_issue.html.haml |  8 +++++++
 spec/features/task_lists_spec.rb           | 27 +++++++++++-----------
 2 files changed, 22 insertions(+), 13 deletions(-)

diff --git a/app/views/projects/issues/_issue.html.haml b/app/views/projects/issues/_issue.html.haml
index 6199569a20a9e..064a7bd1f2606 100644
--- a/app/views/projects/issues/_issue.html.haml
+++ b/app/views/projects/issues/_issue.html.haml
@@ -16,6 +16,10 @@
           - if issue.assignee
             %li
               = link_to_member(@project, issue.assignee, name: false, title: "Assigned to :name")
+          - if issue.tasks?
+            &nbsp;
+            %span.task-status
+              = issue.task_status
 
           = render 'shared/issuable_meta_data', issuable: issue
 
@@ -37,6 +41,10 @@
           &nbsp;
           - issue.labels.each do |label|
             = link_to_label(label, subject: issue.project, css_class: 'label-link')
+        - if issue.tasks?
+          &nbsp;
+          %span.task-status
+            = issue.task_status
 
         .pull-right.issue-updated-at
           %span updated #{time_ago_with_tooltip(issue.updated_at, placement: 'bottom', html_class: 'issue_update_ago')}
diff --git a/spec/features/task_lists_spec.rb b/spec/features/task_lists_spec.rb
index f55c5aaf646be..8bd13caf2b0be 100644
--- a/spec/features/task_lists_spec.rb
+++ b/spec/features/task_lists_spec.rb
@@ -65,13 +65,12 @@ def visit_issue(project, issue)
   describe 'for Issues', feature: true do
     describe 'multiple tasks', js: true do
       include WaitForVueResource
-      
-      before { wait_for_vue_resource }
 
       let!(:issue) { create(:issue, description: markdown, author: user, project: project) }
 
       it 'renders' do
         visit_issue(project, issue)
+        wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 6)
@@ -80,25 +79,24 @@ def visit_issue(project, issue)
 
       it 'contains the required selectors' do
         visit_issue(project, issue)
+        wait_for_vue_resource
 
-        container = '.detail-page-description .description.js-task-list-container'
-
-        expect(page).to have_selector(container)
-        expect(page).to have_selector("#{container} .wiki .task-list .task-list-item .task-list-item-checkbox")
-        expect(page).to have_selector("#{container} .js-task-list-field")
-        expect(page).to have_selector('form.js-issuable-update')
+        expect(page).to have_selector(".wiki .task-list .task-list-item .task-list-item-checkbox")
         expect(page).to have_selector('a.btn-close')
       end
 
       it 'is only editable by author' do
         visit_issue(project, issue)
+        wait_for_vue_resource
 
-        expect(page).to have_selector('.js-task-list-container')
+        expect(page).to have_selector(".wiki .task-list .task-list-item .task-list-item-checkbox")
 
         logout(:user)
         login_as(user2)
         visit current_path
-        expect(page).not_to have_selector('.js-task-list-container')
+        wait_for_vue_resource
+
+        expect(page).to have_selector(".wiki .task-list .task-list-item .task-list-item-checkbox")
       end
 
       it 'provides a summary on Issues#index' do
@@ -112,10 +110,9 @@ def visit_issue(project, issue)
 
       let!(:issue) { create(:issue, description: singleIncompleteMarkdown, author: user, project: project) }
 
-      before { wait_for_vue_resource }
-
       it 'renders' do
         visit_issue(project, issue)
+        wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 1)
@@ -124,15 +121,18 @@ def visit_issue(project, issue)
 
       it 'provides a summary on Issues#index' do
         visit namespace_project_issues_path(project.namespace, project)
+
         expect(page).to have_content("0 of 1 task completed")
       end
     end
 
-    describe 'single complete task' do
+    describe 'single complete task', js: true do
+      include WaitForVueResource
       let!(:issue) { create(:issue, description: singleCompleteMarkdown, author: user, project: project) }
 
       it 'renders' do
         visit_issue(project, issue)
+        wait_for_vue_resource
 
         expect(page).to have_selector('ul.task-list',      count: 1)
         expect(page).to have_selector('li.task-list-item', count: 1)
@@ -141,6 +141,7 @@ def visit_issue(project, issue)
 
       it 'provides a summary on Issues#index' do
         visit namespace_project_issues_path(project.namespace, project)
+
         expect(page).to have_content("1 of 1 task completed")
       end
     end
-- 
GitLab


From c45341c816d78d51aee84a6068d778b9cbc502c8 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Alejandro=20Rodr=C3=ADguez?= <alejorro70@gmail.com>
Date: Fri, 28 Apr 2017 13:52:09 -0300
Subject: [PATCH 174/363] Generate and handle a gl_repository param to pass
 around components

This new param allows us to share project information between components
that don't share or don't have access to the same filesystem
mountpoints, for example between Gitaly and Rails or between Rails and
Gitlab-Shell hooks. The previous parameters are still supported, but if
found, gl_repository is prefered. The old parameters should be deprecated
once all components support the new format.
---
 ...onger-send-absolute-paths-to-gitlab-ce.yml |  4 +
 lib/api/helpers/internal_helpers.rb           | 52 ++++---------
 lib/api/internal.rb                           | 10 ++-
 lib/gitlab/gl_repository.rb                   | 16 ++++
 lib/gitlab/repo_path.rb                       | 29 ++++---
 spec/lib/gitlab/gl_repository_spec.rb         | 19 +++++
 spec/lib/gitlab/repo_path_spec.rb             | 24 ++++++
 .../api/helpers/internal_helpers_spec.rb      | 32 --------
 spec/requests/api/internal_spec.rb            | 77 ++++++++++++++++++-
 spec/support/matchers/gitlab_git_matchers.rb  |  6 ++
 10 files changed, 184 insertions(+), 85 deletions(-)
 create mode 100644 changelogs/unreleased/29925-gitlab-shell-hooks-can-no-longer-send-absolute-paths-to-gitlab-ce.yml
 create mode 100644 lib/gitlab/gl_repository.rb
 create mode 100644 spec/lib/gitlab/gl_repository_spec.rb
 delete mode 100644 spec/requests/api/helpers/internal_helpers_spec.rb
 create mode 100644 spec/support/matchers/gitlab_git_matchers.rb

diff --git a/changelogs/unreleased/29925-gitlab-shell-hooks-can-no-longer-send-absolute-paths-to-gitlab-ce.yml b/changelogs/unreleased/29925-gitlab-shell-hooks-can-no-longer-send-absolute-paths-to-gitlab-ce.yml
new file mode 100644
index 0000000000000..1df8f695ef1ac
--- /dev/null
+++ b/changelogs/unreleased/29925-gitlab-shell-hooks-can-no-longer-send-absolute-paths-to-gitlab-ce.yml
@@ -0,0 +1,4 @@
+---
+title: Generate and handle a gl_repository param to pass around components
+merge_request: 10992
+author:
diff --git a/lib/api/helpers/internal_helpers.rb b/lib/api/helpers/internal_helpers.rb
index 718f936a1fc5a..264df7271a3b0 100644
--- a/lib/api/helpers/internal_helpers.rb
+++ b/lib/api/helpers/internal_helpers.rb
@@ -1,48 +1,14 @@
 module API
   module Helpers
     module InternalHelpers
-      # Project paths may be any of the following:
-      #   * /repository/storage/path/namespace/project
-      #   * /namespace/project
-      #   * namespace/project
-      #
-      # In addition, they may have a '.git' extension and multiple namespaces
-      #
-      # Transform all these cases to 'namespace/project'
-      def clean_project_path(project_path, storages = Gitlab.config.repositories.storages.values)
-        project_path = project_path.sub(/\.git\z/, '')
-
-        storages.each do |storage|
-          storage_path = File.expand_path(storage['path'])
-
-          if project_path.start_with?(storage_path)
-            project_path = project_path.sub(storage_path, '')
-            break
-          end
-        end
-
-        project_path.sub(/\A\//, '')
-      end
-
-      def project_path
-        @project_path ||= clean_project_path(params[:project])
-      end
-
       def wiki?
-        @wiki ||= project_path.end_with?('.wiki') &&
-          !Project.find_by_full_path(project_path)
+        set_project unless defined?(@wiki)
+        @wiki
       end
 
       def project
-        @project ||= begin
-          # Check for *.wiki repositories.
-          # Strip out the .wiki from the pathname before finding the
-          # project. This applies the correct project permissions to
-          # the wiki repository as well.
-          project_path.chomp!('.wiki') if wiki?
-
-          Project.find_by_full_path(project_path)
-        end
+        set_project unless defined?(@project)
+        @project
       end
 
       def ssh_authentication_abilities
@@ -66,6 +32,16 @@ def log_user_activity(actor)
 
         ::Users::ActivityService.new(actor, 'Git SSH').execute if commands.include?(params[:action])
       end
+
+      private
+
+      def set_project
+        if params[:gl_repository]
+          @project, @wiki = Gitlab::GlRepository.parse(params[:gl_repository])
+        else
+          @project, @wiki = Gitlab::RepoPath.parse(params[:project])
+        end
+      end
     end
   end
 end
diff --git a/lib/api/internal.rb b/lib/api/internal.rb
index ebed26dd17866..ddb2047f68621 100644
--- a/lib/api/internal.rb
+++ b/lib/api/internal.rb
@@ -42,6 +42,10 @@ class Internal < Grape::API
         if access_status.status
           log_user_activity(actor)
 
+          # Project id to pass between components that don't share/don't have
+          # access to the same filesystem mounts
+          response[:gl_repository] = "#{wiki? ? 'wiki' : 'project'}-#{project.id}"
+
           # Return the repository full path so that gitlab-shell has it when
           # handling ssh commands
           response[:repository_path] =
@@ -134,11 +138,9 @@ class Internal < Grape::API
 
         return unless Gitlab::GitalyClient.enabled?
 
-        relative_path = Gitlab::RepoPath.strip_storage_path(params[:repo_path])
-        project = Project.find_by_full_path(relative_path.sub(/\.(git|wiki)\z/, ''))
-
         begin
-          Gitlab::GitalyClient::Notifications.new(project.repository).post_receive
+          repository = wiki? ? project.wiki.repository : project.repository
+          Gitlab::GitalyClient::Notifications.new(repository.raw_repository).post_receive
         rescue GRPC::Unavailable => e
           render_api_error!(e, 500)
         end
diff --git a/lib/gitlab/gl_repository.rb b/lib/gitlab/gl_repository.rb
new file mode 100644
index 0000000000000..6997546049ebb
--- /dev/null
+++ b/lib/gitlab/gl_repository.rb
@@ -0,0 +1,16 @@
+module Gitlab
+  module GlRepository
+    def self.parse(gl_repository)
+      match_data = /\A(project|wiki)-([1-9][0-9]*)\z/.match(gl_repository)
+      unless match_data
+        raise ArgumentError, "Invalid GL Repository \"#{gl_repository}\""
+      end
+
+      type, id = match_data.captures
+      project = Project.find_by(id: id)
+      wiki = type == 'wiki'
+
+      [project, wiki]
+    end
+  end
+end
diff --git a/lib/gitlab/repo_path.rb b/lib/gitlab/repo_path.rb
index 4b1d828c45ca3..878e03f61d70d 100644
--- a/lib/gitlab/repo_path.rb
+++ b/lib/gitlab/repo_path.rb
@@ -2,18 +2,29 @@ module Gitlab
   module RepoPath
     NotFoundError = Class.new(StandardError)
 
-    def self.strip_storage_path(repo_path)
-      result = nil
+    def self.parse(repo_path)
+      project_path = strip_storage_path(repo_path.sub(/\.git\z/, ''), fail_on_not_found: false)
+      project = Project.find_by_full_path(project_path)
+      if project_path.end_with?('.wiki') && !project
+        project = Project.find_by_full_path(project_path.chomp('.wiki'))
+        wiki = true
+      else
+        wiki = false
+      end
+
+      [project, wiki]
+    end
+
+    def self.strip_storage_path(repo_path, fail_on_not_found: true)
+      result = repo_path
 
-      Gitlab.config.repositories.storages.values.each do |params|
-        storage_path = params['path']
-        if repo_path.start_with?(storage_path)
-          result = repo_path.sub(storage_path, '')
-          break
-        end
+      storage = Gitlab.config.repositories.storages.values.find do |params|
+        repo_path.start_with?(params['path'])
       end
 
-      if result.nil?
+      if storage
+        result = result.sub(storage['path'], '')
+      elsif fail_on_not_found
         raise NotFoundError.new("No known storage path matches #{repo_path.inspect}")
       end
 
diff --git a/spec/lib/gitlab/gl_repository_spec.rb b/spec/lib/gitlab/gl_repository_spec.rb
new file mode 100644
index 0000000000000..ac3558ab38685
--- /dev/null
+++ b/spec/lib/gitlab/gl_repository_spec.rb
@@ -0,0 +1,19 @@
+require 'spec_helper'
+
+describe ::Gitlab::GlRepository do
+  describe '.parse' do
+    set(:project) { create(:project) }
+
+    it 'parses a project gl_repository' do
+      expect(described_class.parse("project-#{project.id}")).to eq([project, false])
+    end
+
+    it 'parses a wiki gl_repository' do
+      expect(described_class.parse("wiki-#{project.id}")).to eq([project, true])
+    end
+
+    it 'throws an argument error on an invalid gl_repository' do
+      expect { described_class.parse("badformat-#{project.id}") }.to raise_error(ArgumentError)
+    end
+  end
+end
diff --git a/spec/lib/gitlab/repo_path_spec.rb b/spec/lib/gitlab/repo_path_spec.rb
index 0fb5d7646f220..f94c9c2e3155f 100644
--- a/spec/lib/gitlab/repo_path_spec.rb
+++ b/spec/lib/gitlab/repo_path_spec.rb
@@ -1,6 +1,30 @@
 require 'spec_helper'
 
 describe ::Gitlab::RepoPath do
+  describe '.parse' do
+    set(:project) { create(:project) }
+
+    it 'parses a full repository path' do
+      expect(described_class.parse(project.repository.path)).to eq([project, false])
+    end
+
+    it 'parses a full wiki path' do
+      expect(described_class.parse(project.wiki.repository.path)).to eq([project, true])
+    end
+
+    it 'parses a relative repository path' do
+      expect(described_class.parse(project.full_path + '.git')).to eq([project, false])
+    end
+
+    it 'parses a relative wiki path' do
+      expect(described_class.parse(project.full_path + '.wiki.git')).to eq([project, true])
+    end
+
+    it 'parses a relative path starting with /' do
+      expect(described_class.parse('/' + project.full_path + '.git')).to eq([project, false])
+    end
+  end
+
   describe '.strip_storage_path' do
     before do
       allow(Gitlab.config.repositories).to receive(:storages).and_return({
diff --git a/spec/requests/api/helpers/internal_helpers_spec.rb b/spec/requests/api/helpers/internal_helpers_spec.rb
deleted file mode 100644
index db716b340f1c3..0000000000000
--- a/spec/requests/api/helpers/internal_helpers_spec.rb
+++ /dev/null
@@ -1,32 +0,0 @@
-require 'spec_helper'
-
-describe ::API::Helpers::InternalHelpers do
-  include described_class
-
-  describe '.clean_project_path' do
-    project = 'namespace/project'
-    namespaced = File.join('namespace2', project)
-
-    {
-      File.join(Dir.pwd, project)    => project,
-      File.join(Dir.pwd, namespaced) => namespaced,
-      project                        => project,
-      namespaced                     => namespaced,
-      project + '.git'               => project,
-      namespaced + '.git'            => namespaced,
-      "/" + project                  => project,
-      "/" + namespaced               => namespaced,
-    }.each do |project_path, expected|
-      context project_path do
-        # Relative and absolute storage paths, with and without trailing /
-        ['.', './', Dir.pwd, Dir.pwd + '/'].each do |storage_path|
-          context "storage path is #{storage_path}" do
-            subject { clean_project_path(project_path, [{ 'path' => storage_path }]) }
-
-            it { is_expected.to eq(expected) }
-          end
-        end
-      end
-    end
-  end
-end
diff --git a/spec/requests/api/internal_spec.rb b/spec/requests/api/internal_spec.rb
index 429f1a4e375c6..2ceb4648ece67 100644
--- a/spec/requests/api/internal_spec.rb
+++ b/spec/requests/api/internal_spec.rb
@@ -180,6 +180,7 @@
           expect(response).to have_http_status(200)
           expect(json_response["status"]).to be_truthy
           expect(json_response["repository_path"]).to eq(project.wiki.repository.path_to_repo)
+          expect(json_response["gl_repository"]).to eq("wiki-#{project.id}")
           expect(user).not_to have_an_activity_record
         end
       end
@@ -191,6 +192,7 @@
           expect(response).to have_http_status(200)
           expect(json_response["status"]).to be_truthy
           expect(json_response["repository_path"]).to eq(project.wiki.repository.path_to_repo)
+          expect(json_response["gl_repository"]).to eq("wiki-#{project.id}")
           expect(user).to have_an_activity_record
         end
       end
@@ -202,6 +204,7 @@
           expect(response).to have_http_status(200)
           expect(json_response["status"]).to be_truthy
           expect(json_response["repository_path"]).to eq(project.repository.path_to_repo)
+          expect(json_response["gl_repository"]).to eq("project-#{project.id}")
           expect(user).to have_an_activity_record
         end
       end
@@ -213,6 +216,7 @@
           expect(response).to have_http_status(200)
           expect(json_response["status"]).to be_truthy
           expect(json_response["repository_path"]).to eq(project.repository.path_to_repo)
+          expect(json_response["gl_repository"]).to eq("project-#{project.id}")
           expect(user).not_to have_an_activity_record
         end
 
@@ -223,6 +227,7 @@
             expect(response).to have_http_status(200)
             expect(json_response["status"]).to be_truthy
             expect(json_response["repository_path"]).to eq(project.repository.path_to_repo)
+            expect(json_response["gl_repository"]).to eq("project-#{project.id}")
           end
         end
 
@@ -233,6 +238,7 @@
             expect(response).to have_http_status(200)
             expect(json_response["status"]).to be_truthy
             expect(json_response["repository_path"]).to eq(project.repository.path_to_repo)
+            expect(json_response["gl_repository"]).to eq("project-#{project.id}")
           end
         end
       end
@@ -444,18 +450,39 @@
 
       expect(json_response).to eq([])
     end
+
+    context 'with a gl_repository parameter' do
+      let(:gl_repository) { "project-#{project.id}" }
+
+      it 'returns link to create new merge request' do
+        get api("/internal/merge_request_urls?gl_repository=#{gl_repository}&changes=#{changes}"), secret_token: secret_token
+
+        expect(json_response).to match [{
+          "branch_name" => "new_branch",
+          "url" => "http://#{Gitlab.config.gitlab.host}/#{project.namespace.name}/#{project.path}/merge_requests/new?merge_request%5Bsource_branch%5D=new_branch",
+          "new_merge_request" => true
+        }]
+      end
+    end
   end
 
   describe 'POST /notify_post_receive' do
     let(:valid_params) do
-      { repo_path: project.repository.path, secret_token: secret_token }
+      { project: project.repository.path, secret_token: secret_token }
+    end
+
+    let(:valid_wiki_params) do
+      { project: project.wiki.repository.path, secret_token: secret_token }
     end
 
     before do
       allow(Gitlab.config.gitaly).to receive(:enabled).and_return(true)
     end
 
-    it "calls the Gitaly client if it's enabled" do
+    it "calls the Gitaly client with the project's repository" do
+      expect(Gitlab::GitalyClient::Notifications).
+        to receive(:new).with(gitlab_git_repository_with(path: project.repository.path)).
+        and_call_original
       expect_any_instance_of(Gitlab::GitalyClient::Notifications).
         to receive(:post_receive)
 
@@ -464,6 +491,18 @@
       expect(response).to have_http_status(200)
     end
 
+    it "calls the Gitaly client with the wiki's repository if it's a wiki" do
+      expect(Gitlab::GitalyClient::Notifications).
+        to receive(:new).with(gitlab_git_repository_with(path: project.wiki.repository.path)).
+        and_call_original
+      expect_any_instance_of(Gitlab::GitalyClient::Notifications).
+        to receive(:post_receive)
+
+      post api("/internal/notify_post_receive"), valid_wiki_params
+
+      expect(response).to have_http_status(200)
+    end
+
     it "returns 500 if the gitaly call fails" do
       expect_any_instance_of(Gitlab::GitalyClient::Notifications).
         to receive(:post_receive).and_raise(GRPC::Unavailable)
@@ -472,6 +511,40 @@
 
       expect(response).to have_http_status(500)
     end
+
+    context 'with a gl_repository parameter' do
+      let(:valid_params) do
+        { gl_repository: "project-#{project.id}", secret_token: secret_token }
+      end
+
+      let(:valid_wiki_params) do
+        { gl_repository: "wiki-#{project.id}", secret_token: secret_token }
+      end
+
+      it "calls the Gitaly client with the project's repository" do
+        expect(Gitlab::GitalyClient::Notifications).
+          to receive(:new).with(gitlab_git_repository_with(path: project.repository.path)).
+          and_call_original
+        expect_any_instance_of(Gitlab::GitalyClient::Notifications).
+          to receive(:post_receive)
+
+        post api("/internal/notify_post_receive"), valid_params
+
+        expect(response).to have_http_status(200)
+      end
+
+      it "calls the Gitaly client with the wiki's repository if it's a wiki" do
+        expect(Gitlab::GitalyClient::Notifications).
+          to receive(:new).with(gitlab_git_repository_with(path: project.wiki.repository.path)).
+          and_call_original
+        expect_any_instance_of(Gitlab::GitalyClient::Notifications).
+          to receive(:post_receive)
+
+        post api("/internal/notify_post_receive"), valid_wiki_params
+
+        expect(response).to have_http_status(200)
+      end
+    end
   end
 
   def project_with_repo_path(path)
diff --git a/spec/support/matchers/gitlab_git_matchers.rb b/spec/support/matchers/gitlab_git_matchers.rb
new file mode 100644
index 0000000000000..c840cd4bf2d95
--- /dev/null
+++ b/spec/support/matchers/gitlab_git_matchers.rb
@@ -0,0 +1,6 @@
+RSpec::Matchers.define :gitlab_git_repository_with do |values|
+  match do |actual|
+    actual.is_a?(Gitlab::Git::Repository) &&
+      values.all? { |k, v| actual.send(k) == v }
+  end
+end
-- 
GitLab


From 9773b5a3d4a736d933fc8ff3b6652981507b2cbd Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Wed, 3 May 2017 14:51:28 -0600
Subject: [PATCH 175/363] make js true

---
 .../support/features/issuable_slash_commands_shared_examples.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/support/features/issuable_slash_commands_shared_examples.rb b/spec/support/features/issuable_slash_commands_shared_examples.rb
index 5e7eca1d987d4..4de61966daa5b 100644
--- a/spec/support/features/issuable_slash_commands_shared_examples.rb
+++ b/spec/support/features/issuable_slash_commands_shared_examples.rb
@@ -27,7 +27,7 @@
     wait_for_ajax
   end
 
-  describe "new #{issuable_type}" do
+  describe "new #{issuable_type}", js: true do
     context 'with commands in the description' do
       it "creates the #{issuable_type} and interpret commands accordingly" do
         visit public_send("new_namespace_project_#{issuable_type}_path", project.namespace, project, new_url_opts)
-- 
GitLab


From 0a725c80ea6a0ac0544188c743ee3755190334d8 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Wed, 3 May 2017 15:17:25 -0600
Subject: [PATCH 176/363] pass failed spinach spec

---
 features/project/issues/issues.feature | 1 +
 1 file changed, 1 insertion(+)

diff --git a/features/project/issues/issues.feature b/features/project/issues/issues.feature
index 4dee0cd23dc20..1b00d8a32a0c2 100644
--- a/features/project/issues/issues.feature
+++ b/features/project/issues/issues.feature
@@ -82,6 +82,7 @@ Feature: Project Issues
 
   # Markdown
 
+  @javascript
   Scenario: Headers inside the description should have ids generated for them.
     Given I visit issue page "Release 0.4"
     Then Header "Description header" should have correct id and link
-- 
GitLab


From 7822e4a480cd74f5ba744e936901d710239a1eeb Mon Sep 17 00:00:00 2001
From: Annabel Dunstone Gray <annabel.dunstone@gmail.com>
Date: Wed, 3 May 2017 16:30:59 -0500
Subject: [PATCH 177/363] Add tooltips to note action buttons

---
 app/views/projects/notes/_actions.html.haml                 | 6 +++---
 app/views/snippets/notes/_actions.html.haml                 | 6 +++---
 .../unreleased/31760-add-tooltips-to-note-actions.yml       | 4 ++++
 3 files changed, 10 insertions(+), 6 deletions(-)
 create mode 100644 changelogs/unreleased/31760-add-tooltips-to-note-actions.yml

diff --git a/app/views/projects/notes/_actions.html.haml b/app/views/projects/notes/_actions.html.haml
index 718b52dd82e81..d70ec8a6062a5 100644
--- a/app/views/projects/notes/_actions.html.haml
+++ b/app/views/projects/notes/_actions.html.haml
@@ -31,14 +31,14 @@
 - if current_user
   - if note.emoji_awardable?
     - user_authored = note.user_authored?(current_user)
-    = link_to '#', title: 'Award Emoji', class: "note-action-button note-emoji-button js-add-award js-note-emoji #{'js-user-authored' if user_authored}", data: { position: 'right' } do
+    = link_to '#', title: 'Award Emoji', class: "note-action-button note-emoji-button js-add-award js-note-emoji #{'js-user-authored' if user_authored} has-tooltip", data: { position: 'right' } do
       = icon('spinner spin')
       %span{ class: 'link-highlight award-control-icon-neutral' }= custom_icon('emoji_slightly_smiling_face')
       %span{ class: 'link-highlight award-control-icon-positive' }= custom_icon('emoji_smiley')
       %span{ class: 'link-highlight award-control-icon-super-positive' }= custom_icon('emoji_smile')
 
   - if note_editable
-    = link_to '#', title: 'Edit comment', class: 'note-action-button js-note-edit' do
+    = link_to '#', title: 'Edit comment', class: 'note-action-button js-note-edit has-tooltip' do
       = icon('pencil', class: 'link-highlight')
-    = link_to namespace_project_note_path(note.project.namespace, note.project, note), title: 'Remove comment', method: :delete, data: { confirm: 'Are you sure you want to remove this comment?' }, remote: true, class: 'note-action-button js-note-delete danger' do
+    = link_to namespace_project_note_path(note.project.namespace, note.project, note), title: 'Remove comment', method: :delete, data: { confirm: 'Are you sure you want to remove this comment?' }, remote: true, class: 'note-action-button js-note-delete danger has-tooltip' do
       = icon('trash-o', class: 'danger-highlight')
diff --git a/app/views/snippets/notes/_actions.html.haml b/app/views/snippets/notes/_actions.html.haml
index dace11e547461..679a5e934da0e 100644
--- a/app/views/snippets/notes/_actions.html.haml
+++ b/app/views/snippets/notes/_actions.html.haml
@@ -1,13 +1,13 @@
 - if current_user
   - if note.emoji_awardable?
     - user_authored = note.user_authored?(current_user)
-    = link_to '#', title: 'Award Emoji', class: "note-action-button note-emoji-button js-add-award js-note-emoji #{'js-user-authored' if user_authored}", data: { position: 'right' } do
+    = link_to '#', title: 'Award Emoji', class: "note-action-button note-emoji-button js-add-award js-note-emoji #{'js-user-authored' if user_authored} has-tooltip", data: { position: 'right' } do
       = icon('spinner spin')
       %span{ class: 'link-highlight award-control-icon-neutral' }= custom_icon('emoji_slightly_smiling_face')
       %span{ class: 'link-highlight award-control-icon-positive' }= custom_icon('emoji_smiley')
       %span{ class: 'link-highlight award-control-icon-super-positive' }= custom_icon('emoji_smile')
   - if note_editable
-    = link_to '#', title: 'Edit comment', class: 'note-action-button js-note-edit' do
+    = link_to '#', title: 'Edit comment', class: 'note-action-button js-note-edit has-tooltip' do
       = icon('pencil', class: 'link-highlight')
-    = link_to snippet_note_path(note.noteable, note), title: 'Remove comment', method: :delete, data: { confirm: 'Are you sure you want to remove this comment?' }, remote: true, class: 'note-action-button js-note-delete danger' do
+    = link_to snippet_note_path(note.noteable, note), title: 'Remove comment', method: :delete, data: { confirm: 'Are you sure you want to remove this comment?' }, remote: true, class: 'note-action-button js-note-delete danger has-tooltip' do
       = icon('trash-o', class: 'danger-highlight')
diff --git a/changelogs/unreleased/31760-add-tooltips-to-note-actions.yml b/changelogs/unreleased/31760-add-tooltips-to-note-actions.yml
new file mode 100644
index 0000000000000..9bbf43d652e37
--- /dev/null
+++ b/changelogs/unreleased/31760-add-tooltips-to-note-actions.yml
@@ -0,0 +1,4 @@
+---
+title: Add tooltips to note action buttons
+merge_request:
+author:
-- 
GitLab


From 4fd874219318b4ac195264f390c1b04912aff7e1 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Wed, 3 May 2017 17:09:30 -0600
Subject: [PATCH 178/363] fix failing slash command specs

---
 .../features/issuable_slash_commands_shared_examples.rb       | 4 +---
 1 file changed, 1 insertion(+), 3 deletions(-)

diff --git a/spec/support/features/issuable_slash_commands_shared_examples.rb b/spec/support/features/issuable_slash_commands_shared_examples.rb
index 4de61966daa5b..a67940e447a0b 100644
--- a/spec/support/features/issuable_slash_commands_shared_examples.rb
+++ b/spec/support/features/issuable_slash_commands_shared_examples.rb
@@ -3,7 +3,6 @@
 
 shared_examples 'issuable record that supports slash commands in its description and notes' do |issuable_type|
   include SlashCommandsHelpers
-  include WaitForVueResource
 
   let(:master) { create(:user) }
   let(:assignee) { create(:user, username: 'bob') }
@@ -19,7 +18,6 @@
     project.team << [assignee, :developer]
     project.team << [guest, :guest]
     login_with(master)
-    wait_for_vue_resource
   end
 
   after do
@@ -46,7 +44,7 @@
     end
   end
 
-  describe "note on #{issuable_type}" do
+  describe "note on #{issuable_type}", js: true do
     before do
       visit public_send("namespace_project_#{issuable_type}_path", project.namespace, project, issuable)
     end
-- 
GitLab


From 323596f68efe95b479a0dc29832b8033a1eddef0 Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Sat, 29 Apr 2017 10:54:37 +1100
Subject: [PATCH 179/363] Add system note on description change of issue/merge
 request

---
 app/helpers/system_note_helper.rb               |  1 +
 app/models/system_note_metadata.rb              |  2 +-
 app/services/issuable_base_service.rb           |  8 ++++++++
 app/services/system_note_service.rb             | 17 +++++++++++++++++
 app/views/projects/issues/show.html.haml        |  2 +-
 .../merge_requests/show/_mr_box.html.haml       |  2 +-
 .../add_system_note_for_editing_issuable.yml    |  4 ++++
 spec/services/system_note_service_spec.rb       | 14 ++++++++++++++
 8 files changed, 47 insertions(+), 3 deletions(-)
 create mode 100644 changelogs/unreleased/add_system_note_for_editing_issuable.yml

diff --git a/app/helpers/system_note_helper.rb b/app/helpers/system_note_helper.rb
index 1ea60e3938644..d889d1411018f 100644
--- a/app/helpers/system_note_helper.rb
+++ b/app/helpers/system_note_helper.rb
@@ -1,6 +1,7 @@
 module SystemNoteHelper
   ICON_NAMES_BY_ACTION = {
     'commit' => 'icon_commit',
+    'description' => 'icon_edit',
     'merge' => 'icon_merge',
     'merged' => 'icon_merged',
     'opened' => 'icon_status_open',
diff --git a/app/models/system_note_metadata.rb b/app/models/system_note_metadata.rb
index 1e6fc837a758c..b44f4fe000c73 100644
--- a/app/models/system_note_metadata.rb
+++ b/app/models/system_note_metadata.rb
@@ -1,6 +1,6 @@
 class SystemNoteMetadata < ActiveRecord::Base
   ICON_TYPES = %w[
-    commit merge confidential visible label assignee cross_reference
+    commit description merge confidential visible label assignee cross_reference
     title time_tracking branch milestone discussion task moved opened closed merged
   ].freeze
 
diff --git a/app/services/issuable_base_service.rb b/app/services/issuable_base_service.rb
index b071a3984811e..7072d78b28da1 100644
--- a/app/services/issuable_base_service.rb
+++ b/app/services/issuable_base_service.rb
@@ -24,6 +24,10 @@ def create_title_change_note(issuable, old_title)
       issuable, issuable.project, current_user, old_title)
   end
 
+  def create_description_change_note(issuable)
+    SystemNoteService.change_description(issuable, issuable.project, current_user)
+  end
+
   def create_branch_change_note(issuable, branch_type, old_branch, new_branch)
     SystemNoteService.change_branch(
       issuable, issuable.project, current_user, branch_type,
@@ -289,6 +293,10 @@ def handle_common_system_notes(issuable, old_labels: [])
       create_title_change_note(issuable, issuable.previous_changes['title'].first)
     end
 
+    if issuable.previous_changes.include?('description')
+      create_description_change_note(issuable)
+    end
+
     if issuable.previous_changes.include?('description') && issuable.tasks?
       create_task_status_note(issuable)
     end
diff --git a/app/services/system_note_service.rb b/app/services/system_note_service.rb
index c9e25c7aaa20e..82694305a9263 100644
--- a/app/services/system_note_service.rb
+++ b/app/services/system_note_service.rb
@@ -261,6 +261,23 @@ def change_title(noteable, project, author, old_title)
     create_note(NoteSummary.new(noteable, project, author, body, action: 'title'))
   end
 
+  # Called when the description of a Noteable is changed
+  #
+  # noteable  - Noteable object that responds to `description`
+  # project   - Project owning noteable
+  # author    - User performing the change
+  #
+  # Example Note text:
+  #
+  #   "changed the description"
+  #
+  # Returns the created Note object
+  def change_description(noteable, project, author)
+    body = "changed the description"
+
+    create_note(NoteSummary.new(noteable, project, author, body, action: 'description'))
+  end
+
   # Called when the confidentiality changes
   #
   # issue   - Issue object
diff --git a/app/views/projects/issues/show.html.haml b/app/views/projects/issues/show.html.haml
index 2a871966aa832..9bddc3ac44d58 100644
--- a/app/views/projects/issues/show.html.haml
+++ b/app/views/projects/issues/show.html.haml
@@ -61,7 +61,7 @@
           = markdown_field(@issue, :description)
         %textarea.hidden.js-task-list-field
           = @issue.description
-    = edited_time_ago_with_tooltip(@issue, placement: 'bottom', html_class: 'issue_edited_ago')
+    = edited_time_ago_with_tooltip(@issue, placement: 'bottom', html_class: 'issue_edited_ago', include_author: true)
 
     #merge-requests{ data: { url: referenced_merge_requests_namespace_project_issue_url(@project.namespace, @project, @issue) } }
       // This element is filled in using JavaScript.
diff --git a/app/views/projects/merge_requests/show/_mr_box.html.haml b/app/views/projects/merge_requests/show/_mr_box.html.haml
index 8a390cf870044..3b2cbb12a8509 100644
--- a/app/views/projects/merge_requests/show/_mr_box.html.haml
+++ b/app/views/projects/merge_requests/show/_mr_box.html.haml
@@ -10,4 +10,4 @@
         %textarea.hidden.js-task-list-field
           = @merge_request.description
 
-  = edited_time_ago_with_tooltip(@merge_request, placement: 'bottom')
+  = edited_time_ago_with_tooltip(@merge_request, placement: 'bottom', include_author: true)
diff --git a/changelogs/unreleased/add_system_note_for_editing_issuable.yml b/changelogs/unreleased/add_system_note_for_editing_issuable.yml
new file mode 100644
index 0000000000000..3cbc7f91bf007
--- /dev/null
+++ b/changelogs/unreleased/add_system_note_for_editing_issuable.yml
@@ -0,0 +1,4 @@
+---
+title: Add system note on description change of issue/merge request
+merge_request: 10392
+author: blackst0ne
diff --git a/spec/services/system_note_service_spec.rb b/spec/services/system_note_service_spec.rb
index 75d7caf2508cd..5e85c3c162186 100644
--- a/spec/services/system_note_service_spec.rb
+++ b/spec/services/system_note_service_spec.rb
@@ -292,6 +292,20 @@
     end
   end
 
+  describe '.change_description' do
+    subject { described_class.change_description(noteable, project, author) }
+
+    context 'when noteable responds to `description`' do
+      it_behaves_like 'a system note' do
+        let(:action) { 'description' }
+      end
+
+      it 'sets the note text' do
+        expect(subject.note).to eq 'changed the description'
+      end
+    end
+  end
+
   describe '.change_issue_confidentiality' do
     subject { described_class.change_issue_confidentiality(noteable, project, author) }
 
-- 
GitLab


From b82870afc0031cb830976fb36cd1237c6059397f Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Tue, 2 May 2017 09:49:30 +1100
Subject: [PATCH 180/363] Add author to 'Edited time ago by ...' message

---
 app/helpers/application_helper.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb
index fff57472a4f1d..b9d375d9b64d1 100644
--- a/app/helpers/application_helper.rb
+++ b/app/helpers/application_helper.rb
@@ -187,7 +187,7 @@ def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago
       output = content_tag(:span, "Edited ")
       output << time_ago_with_tooltip(object.updated_at, placement: placement, html_class: html_class)
 
-      if include_author && object.updated_by && object.updated_by != object.author
+      if include_author && object.updated_by
         output << content_tag(:span, " by ")
         output << link_to_member(object.project, object.updated_by, avatar: false, author_class: nil)
       end
-- 
GitLab


From 7ad5a1b371f5d319369a6b9d6609c40dea6292c2 Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Wed, 3 May 2017 16:32:21 +1100
Subject: [PATCH 181/363] Add last_edited_at and last_edited_by attributes

---
 app/helpers/application_helper.rb                  | 14 +++++++-------
 app/models/concerns/issuable.rb                    |  1 +
 app/models/note.rb                                 |  1 +
 app/services/issuable_base_service.rb              | 11 +++++++++++
 app/services/notes/update_service.rb               |  4 ++++
 ...st_edited_at_and_last_edited_by_id_to_issues.rb | 14 ++++++++++++++
 ...ast_edited_at_and_last_edited_by_id_to_notes.rb | 14 ++++++++++++++
 ...d_at_and_last_edited_by_id_to_merge_requests.rb | 14 ++++++++++++++
 db/schema.rb                                       |  8 +++++++-
 9 files changed, 73 insertions(+), 8 deletions(-)
 create mode 100644 db/migrate/20170503021915_add_last_edited_at_and_last_edited_by_id_to_issues.rb
 create mode 100644 db/migrate/20170503022512_add_last_edited_at_and_last_edited_by_id_to_notes.rb
 create mode 100644 db/migrate/20170503022548_add_last_edited_at_and_last_edited_by_id_to_merge_requests.rb

diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb
index b9d375d9b64d1..0ff6ab488084d 100644
--- a/app/helpers/application_helper.rb
+++ b/app/helpers/application_helper.rb
@@ -181,15 +181,15 @@ def time_ago_with_tooltip(time, placement: 'top', html_class: '', short_format:
   end
 
   def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago', include_author: false)
-    return if object.updated_at == object.created_at
+    return if object.last_edited_at == object.created_at || object.last_edited_at.blank?
 
-    content_tag :small, class: "edited-text" do
-      output = content_tag(:span, "Edited ")
-      output << time_ago_with_tooltip(object.updated_at, placement: placement, html_class: html_class)
+    content_tag :small, class: 'edited-text' do
+      output = content_tag(:span, 'Edited ')
+      output << time_ago_with_tooltip(object.last_edited_at, placement: placement, html_class: html_class)
 
-      if include_author && object.updated_by
-        output << content_tag(:span, " by ")
-        output << link_to_member(object.project, object.updated_by, avatar: false, author_class: nil)
+      if include_author && object.last_edited_by
+        output << content_tag(:span, ' by ')
+        output << link_to_member(object.project, object.last_edited_by, avatar: false, author_class: nil)
       end
 
       output
diff --git a/app/models/concerns/issuable.rb b/app/models/concerns/issuable.rb
index 26dbf4d95708f..de1ad840daa2e 100644
--- a/app/models/concerns/issuable.rb
+++ b/app/models/concerns/issuable.rb
@@ -28,6 +28,7 @@ module Issuable
     belongs_to :author, class_name: "User"
     belongs_to :assignee, class_name: "User"
     belongs_to :updated_by, class_name: "User"
+    belongs_to :last_edited_by, class_name: 'User'
     belongs_to :milestone
     has_many :notes, as: :noteable, inverse_of: :noteable, dependent: :destroy do
       def authors_loaded?
diff --git a/app/models/note.rb b/app/models/note.rb
index b06985b4a6f81..943211ca991e4 100644
--- a/app/models/note.rb
+++ b/app/models/note.rb
@@ -38,6 +38,7 @@ class Note < ActiveRecord::Base
   belongs_to :noteable, polymorphic: true, touch: true
   belongs_to :author, class_name: "User"
   belongs_to :updated_by, class_name: "User"
+  belongs_to :last_edited_by, class_name: 'User'
 
   has_many :todos, dependent: :destroy
   has_many :events, as: :target, dependent: :destroy
diff --git a/app/services/issuable_base_service.rb b/app/services/issuable_base_service.rb
index 7072d78b28da1..889e450875827 100644
--- a/app/services/issuable_base_service.rb
+++ b/app/services/issuable_base_service.rb
@@ -218,6 +218,13 @@ def update(issuable)
     if issuable.changed? || params.present?
       issuable.assign_attributes(params.merge(updated_by: current_user))
 
+      if has_title_or_description_changed?(issuable)
+        issuable.assign_attributes(params.merge(
+          last_edited_at: Time.now,
+          last_edited_by: current_user
+        ))
+      end
+
       before_update(issuable)
 
       if issuable.with_transaction_returning_status { issuable.save }
@@ -240,6 +247,10 @@ def labels_changing?(old_label_ids, new_label_ids)
     old_label_ids.sort != new_label_ids.sort
   end
 
+  def has_title_or_description_changed?(issuable)
+    issuable.title_changed? || issuable.description_changed?
+  end
+
   def change_state(issuable)
     case params.delete(:state_event)
     when 'reopen'
diff --git a/app/services/notes/update_service.rb b/app/services/notes/update_service.rb
index 75fd08ea0a95c..4b4f383abf2f0 100644
--- a/app/services/notes/update_service.rb
+++ b/app/services/notes/update_service.rb
@@ -5,7 +5,11 @@ def execute(note)
 
       old_mentioned_users = note.mentioned_users.to_a
 
+      note.assign_attributes(params)
+      params.merge!(last_edited_at: Time.now, last_edited_by: current_user) if note.note_changed?
+
       note.update_attributes(params.merge(updated_by: current_user))
+
       note.create_new_cross_references!(current_user)
 
       if note.previous_changes.include?('note')
diff --git a/db/migrate/20170503021915_add_last_edited_at_and_last_edited_by_id_to_issues.rb b/db/migrate/20170503021915_add_last_edited_at_and_last_edited_by_id_to_issues.rb
new file mode 100644
index 0000000000000..6ac10723c82f9
--- /dev/null
+++ b/db/migrate/20170503021915_add_last_edited_at_and_last_edited_by_id_to_issues.rb
@@ -0,0 +1,14 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddLastEditedAtAndLastEditedByIdToIssues < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  def change
+    add_column :issues, :last_edited_at, :timestamp
+    add_column :issues, :last_edited_by_id, :integer
+  end
+end
diff --git a/db/migrate/20170503022512_add_last_edited_at_and_last_edited_by_id_to_notes.rb b/db/migrate/20170503022512_add_last_edited_at_and_last_edited_by_id_to_notes.rb
new file mode 100644
index 0000000000000..a44a1feab16c4
--- /dev/null
+++ b/db/migrate/20170503022512_add_last_edited_at_and_last_edited_by_id_to_notes.rb
@@ -0,0 +1,14 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddLastEditedAtAndLastEditedByIdToNotes < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  def change
+    add_column :notes, :last_edited_at, :timestamp
+    add_column :notes, :last_edited_by_id, :integer
+  end
+end
diff --git a/db/migrate/20170503022548_add_last_edited_at_and_last_edited_by_id_to_merge_requests.rb b/db/migrate/20170503022548_add_last_edited_at_and_last_edited_by_id_to_merge_requests.rb
new file mode 100644
index 0000000000000..7a1acdcbf69ef
--- /dev/null
+++ b/db/migrate/20170503022548_add_last_edited_at_and_last_edited_by_id_to_merge_requests.rb
@@ -0,0 +1,14 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddLastEditedAtAndLastEditedByIdToMergeRequests < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  def change
+    add_column :merge_requests, :last_edited_at, :timestamp
+    add_column :merge_requests, :last_edited_by_id, :integer
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index be6684f3a6b4e..b173b467abf86 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170426181740) do
+ActiveRecord::Schema.define(version: 20170503022548) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -488,6 +488,8 @@
     t.integer "relative_position"
     t.datetime "closed_at"
     t.integer "cached_markdown_version"
+    t.datetime "last_edited_at"
+    t.integer "last_edited_by_id"
   end
 
   add_index "issues", ["assignee_id"], name: "index_issues_on_assignee_id", using: :btree
@@ -674,6 +676,8 @@
     t.text "description_html"
     t.integer "time_estimate"
     t.integer "cached_markdown_version"
+    t.datetime "last_edited_at"
+    t.integer "last_edited_by_id"
   end
 
   add_index "merge_requests", ["assignee_id"], name: "index_merge_requests_on_assignee_id", using: :btree
@@ -774,6 +778,8 @@
     t.string "discussion_id"
     t.text "note_html"
     t.integer "cached_markdown_version"
+    t.datetime "last_edited_at"
+    t.integer "last_edited_by_id"
   end
 
   add_index "notes", ["author_id"], name: "index_notes_on_author_id", using: :btree
-- 
GitLab


From 571c832d31690ba549d936c6cd921309eadad519 Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Wed, 3 May 2017 16:47:14 +1100
Subject: [PATCH 182/363] Simplified assign_attributes for issuable

---
 app/services/issuable_base_service.rb | 9 +++------
 1 file changed, 3 insertions(+), 6 deletions(-)

diff --git a/app/services/issuable_base_service.rb b/app/services/issuable_base_service.rb
index 889e450875827..e77799e548c5d 100644
--- a/app/services/issuable_base_service.rb
+++ b/app/services/issuable_base_service.rb
@@ -216,15 +216,12 @@ def update(issuable)
     params[:label_ids] = label_ids if labels_changing?(issuable.label_ids, label_ids)
 
     if issuable.changed? || params.present?
-      issuable.assign_attributes(params.merge(updated_by: current_user))
-
       if has_title_or_description_changed?(issuable)
-        issuable.assign_attributes(params.merge(
-          last_edited_at: Time.now,
-          last_edited_by: current_user
-        ))
+        issuable.assign_attributes(last_edited_at: Time.now, last_edited_by: current_user)
       end
 
+      issuable.assign_attributes(params.merge(updated_by: current_user))
+
       before_update(issuable)
 
       if issuable.with_transaction_returning_status { issuable.save }
-- 
GitLab


From d9f48e45a124530211e0a65d25f5d41db5842c62 Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Wed, 3 May 2017 16:49:00 +1100
Subject: [PATCH 183/363] Prefer single quotes

---
 app/services/system_note_service.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/services/system_note_service.rb b/app/services/system_note_service.rb
index 82694305a9263..7db634fbca181 100644
--- a/app/services/system_note_service.rb
+++ b/app/services/system_note_service.rb
@@ -273,7 +273,7 @@ def change_title(noteable, project, author, old_title)
   #
   # Returns the created Note object
   def change_description(noteable, project, author)
-    body = "changed the description"
+    body = 'changed the description'
 
     create_note(NoteSummary.new(noteable, project, author, body, action: 'description'))
   end
-- 
GitLab


From 35160a97f2dc781209515af4cdce6ff33e039cc9 Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Wed, 3 May 2017 17:54:00 +1100
Subject: [PATCH 184/363] Add specs for issue and note changes

---
 app/services/issuable_base_service.rb      |  4 ++--
 spec/features/issues_spec.rb               | 22 ++++++++++++++++++++++
 spec/services/notes/update_service_spec.rb |  7 +++++++
 3 files changed, 31 insertions(+), 2 deletions(-)

diff --git a/app/services/issuable_base_service.rb b/app/services/issuable_base_service.rb
index e77799e548c5d..d0805fd6eb471 100644
--- a/app/services/issuable_base_service.rb
+++ b/app/services/issuable_base_service.rb
@@ -216,12 +216,12 @@ def update(issuable)
     params[:label_ids] = label_ids if labels_changing?(issuable.label_ids, label_ids)
 
     if issuable.changed? || params.present?
+      issuable.assign_attributes(params.merge(updated_by: current_user))
+
       if has_title_or_description_changed?(issuable)
         issuable.assign_attributes(last_edited_at: Time.now, last_edited_by: current_user)
       end
 
-      issuable.assign_attributes(params.merge(updated_by: current_user))
-
       before_update(issuable)
 
       if issuable.with_transaction_returning_status { issuable.save }
diff --git a/spec/features/issues_spec.rb b/spec/features/issues_spec.rb
index 81cc8513454ef..93b7880b96f97 100644
--- a/spec/features/issues_spec.rb
+++ b/spec/features/issues_spec.rb
@@ -714,4 +714,26 @@
       expect(page).to have_text("updated title")
     end
   end
+
+  describe '"edited by" message', js: true do
+    let(:issue) { create(:issue, project: project, author: @user) }
+
+    context 'when issue is updated' do
+      before { visit edit_namespace_project_issue_path(project.namespace, project, issue) }
+
+      it 'shows "edited by" mesage on title update' do
+        fill_in 'issue_title', with: 'hello world'
+        click_button 'Save changes'
+
+        expect(page).to have_content("Edited less than a minute ago by #{@user.name}")
+      end
+
+      it 'shows "edited by" mesage on description update' do
+        fill_in 'issue_description', with: 'hello world'
+        click_button 'Save changes'
+
+        expect(page).to have_content("Edited less than a minute ago by #{@user.name}")
+      end
+    end
+  end
 end
diff --git a/spec/services/notes/update_service_spec.rb b/spec/services/notes/update_service_spec.rb
index 905e2f46bde41..8c3e4607f3e42 100644
--- a/spec/services/notes/update_service_spec.rb
+++ b/spec/services/notes/update_service_spec.rb
@@ -20,6 +20,13 @@ def update_note(opts)
       @note.reload
     end
 
+    it 'updates last_edited_at and last_edited_by attributes' do
+      update_note({ note: 'Hello world!' })
+
+      expect(@note.last_edited_at).not_to be_nil
+      expect(@note.last_edited_by).not_to be_nil
+    end
+
     context 'todos' do
       let!(:todo) { create(:todo, :assigned, user: user, project: project, target: issue, author: user2) }
 
-- 
GitLab


From c2b869a119f7400d748a8a97e4abc3137bc7c51d Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Wed, 3 May 2017 18:07:53 +1100
Subject: [PATCH 185/363] Add specs for merge requests

---
 spec/features/merge_requests/edit_mr_spec.rb | 18 ++++++++++++++++++
 1 file changed, 18 insertions(+)

diff --git a/spec/features/merge_requests/edit_mr_spec.rb b/spec/features/merge_requests/edit_mr_spec.rb
index cb3bc3929031a..b0855af95247f 100644
--- a/spec/features/merge_requests/edit_mr_spec.rb
+++ b/spec/features/merge_requests/edit_mr_spec.rb
@@ -67,5 +67,23 @@
     def get_textarea_height
       page.evaluate_script('document.getElementById("merge_request_description").offsetHeight')
     end
+
+    describe '"edited by" message', js: true do
+      context 'when merge request is updated' do
+        it 'shows "edited by" mesage on title update' do
+          fill_in 'merge_request_title', with: 'hello world'
+          click_button 'Save changes'
+
+          expect(page).to have_content("Edited less than a minute ago by #{user.name}")
+        end
+
+        it 'shows "edited by" mesage on description update' do
+          fill_in 'merge_request_description', with: 'hello world'
+          click_button 'Save changes'
+
+          expect(page).to have_content("Edited less than a minute ago by #{user.name}")
+        end
+      end
+    end
   end
 end
-- 
GitLab


From 62be3355b1cc74e085a7a046e7aca05f59a1f97a Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Wed, 3 May 2017 22:11:19 +1100
Subject: [PATCH 186/363] Add alias_attributes for notes

---
 app/models/note.rb                                 |  3 +++
 app/services/notes/update_service.rb               |  4 ----
 ...ast_edited_at_and_last_edited_by_id_to_notes.rb | 14 --------------
 db/schema.rb                                       |  2 --
 spec/services/notes/update_service_spec.rb         |  7 -------
 5 files changed, 3 insertions(+), 27 deletions(-)
 delete mode 100644 db/migrate/20170503022512_add_last_edited_at_and_last_edited_by_id_to_notes.rb

diff --git a/app/models/note.rb b/app/models/note.rb
index 943211ca991e4..002a1565d540f 100644
--- a/app/models/note.rb
+++ b/app/models/note.rb
@@ -18,6 +18,9 @@ class Note < ActiveRecord::Base
 
   cache_markdown_field :note, pipeline: :note, issuable_state_filter_enabled: true
 
+  alias_attribute :last_edited_at, :updated_at
+  alias_attribute :last_edited_by, :updated_by
+
   # Attribute containing rendered and redacted Markdown as generated by
   # Banzai::ObjectRenderer.
   attr_accessor :redacted_note_html
diff --git a/app/services/notes/update_service.rb b/app/services/notes/update_service.rb
index 4b4f383abf2f0..75fd08ea0a95c 100644
--- a/app/services/notes/update_service.rb
+++ b/app/services/notes/update_service.rb
@@ -5,11 +5,7 @@ def execute(note)
 
       old_mentioned_users = note.mentioned_users.to_a
 
-      note.assign_attributes(params)
-      params.merge!(last_edited_at: Time.now, last_edited_by: current_user) if note.note_changed?
-
       note.update_attributes(params.merge(updated_by: current_user))
-
       note.create_new_cross_references!(current_user)
 
       if note.previous_changes.include?('note')
diff --git a/db/migrate/20170503022512_add_last_edited_at_and_last_edited_by_id_to_notes.rb b/db/migrate/20170503022512_add_last_edited_at_and_last_edited_by_id_to_notes.rb
deleted file mode 100644
index a44a1feab16c4..0000000000000
--- a/db/migrate/20170503022512_add_last_edited_at_and_last_edited_by_id_to_notes.rb
+++ /dev/null
@@ -1,14 +0,0 @@
-# See http://doc.gitlab.com/ce/development/migration_style_guide.html
-# for more information on how to write migrations for GitLab.
-
-class AddLastEditedAtAndLastEditedByIdToNotes < ActiveRecord::Migration
-  include Gitlab::Database::MigrationHelpers
-
-  # Set this constant to true if this migration requires downtime.
-  DOWNTIME = false
-
-  def change
-    add_column :notes, :last_edited_at, :timestamp
-    add_column :notes, :last_edited_by_id, :integer
-  end
-end
diff --git a/db/schema.rb b/db/schema.rb
index b173b467abf86..20127cbed32c0 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -778,8 +778,6 @@
     t.string "discussion_id"
     t.text "note_html"
     t.integer "cached_markdown_version"
-    t.datetime "last_edited_at"
-    t.integer "last_edited_by_id"
   end
 
   add_index "notes", ["author_id"], name: "index_notes_on_author_id", using: :btree
diff --git a/spec/services/notes/update_service_spec.rb b/spec/services/notes/update_service_spec.rb
index 8c3e4607f3e42..905e2f46bde41 100644
--- a/spec/services/notes/update_service_spec.rb
+++ b/spec/services/notes/update_service_spec.rb
@@ -20,13 +20,6 @@ def update_note(opts)
       @note.reload
     end
 
-    it 'updates last_edited_at and last_edited_by attributes' do
-      update_note({ note: 'Hello world!' })
-
-      expect(@note.last_edited_at).not_to be_nil
-      expect(@note.last_edited_by).not_to be_nil
-    end
-
     context 'todos' do
       let!(:todo) { create(:todo, :assigned, user: user, project: project, target: issue, author: user2) }
 
-- 
GitLab


From aaa70a62b05f846a677a4b4dc5ccc83ca82fba32 Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Wed, 3 May 2017 22:16:36 +1100
Subject: [PATCH 187/363] Add comment to notes aliases

---
 app/models/note.rb | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/app/models/note.rb b/app/models/note.rb
index 002a1565d540f..46d0a4f159f87 100644
--- a/app/models/note.rb
+++ b/app/models/note.rb
@@ -18,6 +18,8 @@ class Note < ActiveRecord::Base
 
   cache_markdown_field :note, pipeline: :note, issuable_state_filter_enabled: true
 
+  # Aliases to make application_helper#edited_time_ago_with_tooltip helper work properly with notes.
+  # See https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10392/diffs#note_28719102
   alias_attribute :last_edited_at, :updated_at
   alias_attribute :last_edited_by, :updated_by
 
-- 
GitLab


From ac99441de9774333da4422eb9072777c09a2ad7a Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Wed, 3 May 2017 22:46:10 +1100
Subject: [PATCH 188/363] Change 'exclude_author' param

---
 app/helpers/application_helper.rb                        | 4 ++--
 app/views/projects/issues/show.html.haml                 | 2 +-
 app/views/projects/merge_requests/show/_mr_box.html.haml | 2 +-
 app/views/shared/notes/_note.html.haml                   | 2 +-
 app/views/shared/snippets/_header.html.haml              | 2 +-
 5 files changed, 6 insertions(+), 6 deletions(-)

diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb
index 0ff6ab488084d..6d6bcbaf88a18 100644
--- a/app/helpers/application_helper.rb
+++ b/app/helpers/application_helper.rb
@@ -180,14 +180,14 @@ def time_ago_with_tooltip(time, placement: 'top', html_class: '', short_format:
     element
   end
 
-  def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago', include_author: false)
+  def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago', exclude_author: false)
     return if object.last_edited_at == object.created_at || object.last_edited_at.blank?
 
     content_tag :small, class: 'edited-text' do
       output = content_tag(:span, 'Edited ')
       output << time_ago_with_tooltip(object.last_edited_at, placement: placement, html_class: html_class)
 
-      if include_author && object.last_edited_by
+      if !exclude_author && object.last_edited_by
         output << content_tag(:span, ' by ')
         output << link_to_member(object.project, object.last_edited_by, avatar: false, author_class: nil)
       end
diff --git a/app/views/projects/issues/show.html.haml b/app/views/projects/issues/show.html.haml
index 9bddc3ac44d58..2a871966aa832 100644
--- a/app/views/projects/issues/show.html.haml
+++ b/app/views/projects/issues/show.html.haml
@@ -61,7 +61,7 @@
           = markdown_field(@issue, :description)
         %textarea.hidden.js-task-list-field
           = @issue.description
-    = edited_time_ago_with_tooltip(@issue, placement: 'bottom', html_class: 'issue_edited_ago', include_author: true)
+    = edited_time_ago_with_tooltip(@issue, placement: 'bottom', html_class: 'issue_edited_ago')
 
     #merge-requests{ data: { url: referenced_merge_requests_namespace_project_issue_url(@project.namespace, @project, @issue) } }
       // This element is filled in using JavaScript.
diff --git a/app/views/projects/merge_requests/show/_mr_box.html.haml b/app/views/projects/merge_requests/show/_mr_box.html.haml
index 3b2cbb12a8509..8a390cf870044 100644
--- a/app/views/projects/merge_requests/show/_mr_box.html.haml
+++ b/app/views/projects/merge_requests/show/_mr_box.html.haml
@@ -10,4 +10,4 @@
         %textarea.hidden.js-task-list-field
           = @merge_request.description
 
-  = edited_time_ago_with_tooltip(@merge_request, placement: 'bottom', include_author: true)
+  = edited_time_ago_with_tooltip(@merge_request, placement: 'bottom')
diff --git a/app/views/shared/notes/_note.html.haml b/app/views/shared/notes/_note.html.haml
index 9657b4eea82f0..6868f63199a19 100644
--- a/app/views/shared/notes/_note.html.haml
+++ b/app/views/shared/notes/_note.html.haml
@@ -40,7 +40,7 @@
       .note-body{ class: note_editable ? 'js-task-list-container' : '' }
         .note-text.md
           = note.redacted_note_html
-        = edited_time_ago_with_tooltip(note, placement: 'bottom', html_class: 'note_edited_ago', include_author: true)
+        = edited_time_ago_with_tooltip(note, placement: 'bottom', html_class: 'note_edited_ago')
         - if note_editable
           - if note.for_personal_snippet?
             = render 'snippets/notes/edit', note: note
diff --git a/app/views/shared/snippets/_header.html.haml b/app/views/shared/snippets/_header.html.haml
index d084f5e968416..501c09d71d510 100644
--- a/app/views/shared/snippets/_header.html.haml
+++ b/app/views/shared/snippets/_header.html.haml
@@ -21,4 +21,4 @@
     = markdown_field(@snippet, :title)
 
   - if @snippet.updated_at != @snippet.created_at
-    = edited_time_ago_with_tooltip(@snippet, placement: 'bottom', html_class: 'snippet-edited-ago')
+    = edited_time_ago_with_tooltip(@snippet, placement: 'bottom', html_class: 'snippet-edited-ago', exclude_author: true)
-- 
GitLab


From b2e578803dce182bd8d871fa7bcac573e05c4b95 Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Thu, 4 May 2017 12:36:12 +1100
Subject: [PATCH 189/363] Add feature spec for system notes

---
 spec/features/issuables/system_notes_spec.rb | 30 ++++++++++++++++++++
 1 file changed, 30 insertions(+)
 create mode 100644 spec/features/issuables/system_notes_spec.rb

diff --git a/spec/features/issuables/system_notes_spec.rb b/spec/features/issuables/system_notes_spec.rb
new file mode 100644
index 0000000000000..e12d81e76cbaf
--- /dev/null
+++ b/spec/features/issuables/system_notes_spec.rb
@@ -0,0 +1,30 @@
+require 'spec_helper'
+
+describe 'issuable system notes', feature: true do
+  let(:issue)         { create(:issue, project: project, author: user) }
+  let(:merge_request) { create(:merge_request, :simple, source_project: project) }
+  let(:project)       { create(:project, :public) }
+  let(:user)          { create(:user) }
+
+  before do
+    project.add_user(user, :master)
+    login_as(user)
+  end
+
+  [:issue, :merge_request].each do |issuable_type|
+    context "when #{issuable_type}" do
+      before do
+        issuable = issuable_type == :issue ? issue : merge_request
+
+        visit(edit_polymorphic_path([project.namespace.becomes(Namespace), project, issuable]))
+      end
+
+      it 'adds system note "description changed"' do
+        fill_in("#{issuable_type}_description", with: 'hello world')
+        click_button('Save changes')
+
+        expect(page).to have_content("#{user.name} #{user.to_reference} changed the description")
+      end
+    end
+  end
+end
-- 
GitLab


From dbd1bdaeed596f14af89d662e73030bb02571cfd Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Wed, 3 May 2017 21:05:38 -0500
Subject: [PATCH 190/363] More updates for translations plus some refactoring.

---
 .../components/stage_code_component.js        |  4 +-
 .../components/stage_issue_component.js       |  2 +-
 .../components/stage_plan_component.js        |  4 +-
 .../components/stage_production_component.js  |  4 +-
 .../components/stage_review_component.js      |  4 +-
 .../components/stage_staging_component.js     |  2 +-
 .../cycle_analytics_service.js                |  2 +-
 .../cycle_analytics/cycle_analytics_store.js  |  2 +-
 app/assets/javascripts/locale/de/app.js       |  2 +-
 app/assets/javascripts/locale/en/app.js       |  2 +-
 app/assets/javascripts/locale/es/app.js       |  2 +-
 app/controllers/application_controller.rb     | 14 ++---
 app/serializers/analytics_stage_entity.rb     |  1 +
 app/serializers/analytics_summary_entity.rb   |  5 +-
 .../projects/cycle_analytics/show.html.haml   |  4 +-
 db/fixtures/development/17_cycle_analytics.rb |  2 +-
 lib/api/api.rb                                |  3 +
 lib/gitlab/cycle_analytics/base_stage.rb      |  2 +-
 lib/gitlab/cycle_analytics/code_stage.rb      |  6 +-
 lib/gitlab/cycle_analytics/issue_stage.rb     |  6 +-
 lib/gitlab/cycle_analytics/plan_stage.rb      |  6 +-
 .../cycle_analytics/production_stage.rb       |  6 +-
 lib/gitlab/cycle_analytics/review_stage.rb    |  6 +-
 lib/gitlab/cycle_analytics/staging_stage.rb   |  6 +-
 lib/gitlab/cycle_analytics/summary/base.rb    |  2 +-
 lib/gitlab/cycle_analytics/summary/commit.rb  |  4 ++
 lib/gitlab/cycle_analytics/summary/deploy.rb  |  4 ++
 lib/gitlab/cycle_analytics/summary/issue.rb   |  2 +-
 lib/gitlab/cycle_analytics/test_stage.rb      |  6 +-
 lib/gitlab/i18n.rb                            | 10 +++
 locale/de/gitlab.po                           | 55 ++++++++++++-----
 locale/en/gitlab.po                           | 55 ++++++++++++-----
 locale/es/gitlab.po                           | 61 +++++++++++++------
 locale/gitlab.pot                             | 59 +++++++++++++-----
 locale/unfound_translations.rb                | 10 ---
 35 files changed, 253 insertions(+), 112 deletions(-)
 delete mode 100644 locale/unfound_translations.rb

diff --git a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
index f22299898a5c4..0d9ad197abf18 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageCodeComponent = Vue.extend({
             <a :href="mergeRequest.url" class="issue-link">!{{ mergeRequest.iid }}</a>
             &middot;
             <span>
-              {{ __('Opened') }}
+              {{ __('OpenedNDaysAgo|Opened') }}
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              {{ __('Author|by') }}
+              {{ __('ByAuthor|by') }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
           </div>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
index f400fd2fb142c..c4018c8fc36e5 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageIssueComponent = Vue.extend({
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-              {{ __('Author|by') }}
+              {{ __('ByAuthor|by') }}
               <a :href="issue.author.webUrl" class="issue-author-link">
                 {{ issue.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
index b85b8902fc7c5..222084deee94c 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
@@ -31,10 +31,10 @@ global.cycleAnalytics.StagePlanComponent = Vue.extend({
               </a>
             </h5>
             <span>
-              {{ __('OfFirstTime|First') }}
+              {{ __('FirstPushedBy|First') }}
               <span class="commit-icon">${iconCommit}</span>
               <a :href="commit.commitUrl" class="commit-hash-link monospace">{{ commit.shortSha }}</a>
-              {{ __('pushed by') }}
+              {{ __('FirstPushedBy|pushed by') }}
               <a :href="commit.author.webUrl" class="commit-author-link">
                 {{ commit.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
index 18c30f490fd76..a14ebc3ece942 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageProductionComponent = Vue.extend({
             <a :href="issue.url" class="issue-link">#{{ issue.iid }}</a>
             &middot;
             <span>
-              {{ __('Opened') }}
+              {{ __('OpenedNDaysAgo|Opened') }}
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-            {{ __('Author|by') }}
+            {{ __('ByAuthor|by') }}
             <a :href="issue.author.webUrl" class="issue-author-link">
               {{ issue.author.name }}
             </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
index 988067601284b..1a5bf9bc0b511 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageReviewComponent = Vue.extend({
             <a :href="mergeRequest.url" class="issue-link">!{{ mergeRequest.iid }}</a>
             &middot;
             <span>
-              {{ __('Opened') }}
+              {{ __('OpenedNDaysAgo|Opened') }}
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              {{ __('Author|by') }}
+              {{ __('ByAuthor|by') }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
             <template v-if="mergeRequest.state === 'closed'">
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
index f13294c7af707..b1e9362434f1f 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageStagingComponent = Vue.extend({
             </h5>
             <span>
               <a :href="build.url" class="build-date">{{ build.date }}</a>
-              {{ __('Author|by') }}
+              {{ __('ByAuthor|by') }}
               <a :href="build.author.webUrl" class="issue-author-link">
                 {{ build.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
index 681d6eef56577..c176376e8cf89 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
@@ -30,7 +30,7 @@ class CycleAnalyticsService {
       startDate,
     } = options;
 
-    return $.get(`${this.requestPath}/events/${stage.title.toLowerCase()}.json`, {
+    return $.get(`${this.requestPath}/events/${stage.name.toLowerCase()}.json`, {
       cycle_analytics: {
         start_date: startDate,
       },
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
index 1c57495121141..50bd394e90e0a 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
@@ -39,7 +39,7 @@ global.cycleAnalytics.CycleAnalyticsStore = {
     });
 
     newData.stages.forEach((item) => {
-      const stageSlug = gl.text.dasherize(item.title.toLowerCase());
+      const stageSlug = gl.text.dasherize(item.name.toLowerCase());
       item.active = false;
       item.isUserAllowed = data.permissions[stageSlug];
       item.emptyStageText = EMPTY_STAGE_TEXTS[stageSlug];
diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index e1e4a2b7138f9..aee25c3439249 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"Opened":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index c14e7b41ad137..ed32f7e49c0ea 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":[""],"Commits":[""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploys":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issues":[""],"Not available":[""],"Not enough data":[""],"OfFirstTime|First":[""],"Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""],"pushed by":[""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"Opened":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index 9e09a28c0e55c..0d8152bb8635e 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-03 12:28-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"Author|by":["por"],"Commits":["Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"CycleAnalyticsStage":[""],"CycleAnalyticsStage|Code":["Código"],"CycleAnalyticsStage|Issue":["Incidencia"],"CycleAnalyticsStage|Plan":["Planificación"],"CycleAnalyticsStage|Production":["Producción"],"CycleAnalyticsStage|Review":["Revisión"],"CycleAnalyticsStage|Staging":["Puesta en escena"],"CycleAnalyticsStage|Test":["Pruebas"],"Deploys":["Despliegues"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Último %d día","Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d evento","Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issues":["Nuevas incidencias"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OfFirstTime|First":["Primer"],"Opened":["Abiertos"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Incidencias Relacionadas"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionados Originados por Cambios"],"Relative Deployed Builds":["Builds Desplegados Relacionados"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":["Tiempo Total"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"],"pushed by":["enviado por"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-03 21:03-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":["por"],"Commit":["Cambio","Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"CycleAnalyticsStage|Code":["Código"],"CycleAnalyticsStage|Issue":["Incidencia"],"CycleAnalyticsStage|Plan":["Planificación"],"CycleAnalyticsStage|Production":["Producción"],"CycleAnalyticsStage|Review":["Revisión"],"CycleAnalyticsStage|Staging":["Puesta en escena"],"CycleAnalyticsStage|Test":["Pruebas"],"Deploy":["Despliegue","Despliegues"],"FirstPushedBy|First":["Primer"],"FirstPushedBy|pushed by":["enviado por"],"From issue creation until deploy to production":["Desde la creación de la incidencia hasta el despliegue a producción"],"From merge request merge until deploy to production":["Desde la integración de la solicitud de fusión hasta el despliegue a producción"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Último %d día","Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d evento","Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issue":["Nueva incidencia","Nuevas incidencias"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"OpenedNDaysAgo|Opened":["Abierto"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Incidencias Relacionadas"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionados Originados por Cambios"],"Relative Deployed Builds":["Builds Desplegados Relacionados"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time before an issue gets scheduled":["Tiempo antes de que una incidencia sea programada"],"Time before an issue starts implementation":["Tiempo antes de que empieze la implementación de una incidencia"],"Time between merge request creation and merge/close":["Tiempo entre la creación de la solicitud de fusión y la integración o cierre de esta"],"Time until first merge request":["Tiempo hasta la primera solicitud de fusión"],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":["Tiempo Total"],"Total test time for all commits/merges":["Tiempo total de pruebas para todos los cambios o integraciones"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"]}}};
\ No newline at end of file
diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index 24017e8ea40ae..d2c13da691741 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -273,14 +273,10 @@ def u2f_app_id
   end
 
   def set_locale
-    begin
-      requested_locale = current_user&.preferred_language || I18n.default_locale
-      locale = FastGettext.set_locale(requested_locale)
-      I18n.locale = locale
-
-      yield
-    ensure
-      I18n.locale = I18n.default_locale
-    end
+    Gitlab::I18n.set_locale(current_user)
+
+    yield
+  ensure
+    Gitlab::I18n.reset_locale
   end
 end
diff --git a/app/serializers/analytics_stage_entity.rb b/app/serializers/analytics_stage_entity.rb
index 69bf693de8d8e..564612202b5ec 100644
--- a/app/serializers/analytics_stage_entity.rb
+++ b/app/serializers/analytics_stage_entity.rb
@@ -2,6 +2,7 @@ class AnalyticsStageEntity < Grape::Entity
   include EntityDateHelper
 
   expose :title
+  expose :name
   expose :legend
   expose :description
 
diff --git a/app/serializers/analytics_summary_entity.rb b/app/serializers/analytics_summary_entity.rb
index 91803ec07f5ff..9c37afd53e1ac 100644
--- a/app/serializers/analytics_summary_entity.rb
+++ b/app/serializers/analytics_summary_entity.rb
@@ -1,7 +1,4 @@
 class AnalyticsSummaryEntity < Grape::Entity
   expose :value, safe: true
-
-  expose :title do |object|
-    object.title.pluralize(object.value)
-  end
+  expose :title
 end
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index 5c1426e49953b..eadaff70e0469 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -51,7 +51,7 @@
             %ul
               %li.stage-header
                 %span.stage-name
-                  {{ s__('ProjectLifecycle|Stage') }}
+                  {{ __('ProjectLifecycle|Stage') }}
                 %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The phase of the development lifecycle."), "aria-hidden" => "true" }
               %li.median-header
                 %span.stage-name
@@ -70,7 +70,7 @@
             %ul
               %li.stage-nav-item{ ':class' => '{ active: stage.active }', '@click' => 'selectStage(stage)', "v-for" => "stage in state.stages" }
                 .stage-nav-item-cell.stage-name
-                  {{ s__('CycleAnalyticsStage', stage.title) }}
+                  {{ stage.title }}
                 .stage-nav-item-cell.stage-median
                   %template{ "v-if" => "stage.isUserAllowed" }
                     %span{ "v-if" => "stage.value" }
diff --git a/db/fixtures/development/17_cycle_analytics.rb b/db/fixtures/development/17_cycle_analytics.rb
index 0d7eb1a7c9336..040505ad52c54 100644
--- a/db/fixtures/development/17_cycle_analytics.rb
+++ b/db/fixtures/development/17_cycle_analytics.rb
@@ -227,7 +227,7 @@ def deploy_to_production(merge_requests)
 
   if ENV[flag]
     Project.all.each do |project|
-      seeder = Gitlab::Seeder::CycleAnalytics.new(project)
+      seeder = Gitlab::Seeder::CycleAnalytics.new(pro)
       seeder.seed!
     end
   elsif ENV['CYCLE_ANALYTICS_PERF_TEST']
diff --git a/lib/api/api.rb b/lib/api/api.rb
index 1bf20f76ad6a4..302ccd3e5327d 100644
--- a/lib/api/api.rb
+++ b/lib/api/api.rb
@@ -44,6 +44,9 @@ class API < Grape::API
     end
 
     before { allow_access_with_scope :api }
+    before { Gitlab::I18n.set_locale(current_user) }
+
+    after { Gitlab::I18n.reset_locale }
 
     rescue_from Gitlab::Access::AccessDeniedError do
       rack_response({ 'message' => '403 Forbidden' }.to_json, 403)
diff --git a/lib/gitlab/cycle_analytics/base_stage.rb b/lib/gitlab/cycle_analytics/base_stage.rb
index 559e3939da641..cac31ea8cff42 100644
--- a/lib/gitlab/cycle_analytics/base_stage.rb
+++ b/lib/gitlab/cycle_analytics/base_stage.rb
@@ -17,7 +17,7 @@ def as_json
       end
 
       def title
-        name.to_s.capitalize
+        raise NotImplementedError.new("Expected #{self.name} to implement title")
       end
 
       def median
diff --git a/lib/gitlab/cycle_analytics/code_stage.rb b/lib/gitlab/cycle_analytics/code_stage.rb
index 182d4c43fbda9..5f9dc9a430376 100644
--- a/lib/gitlab/cycle_analytics/code_stage.rb
+++ b/lib/gitlab/cycle_analytics/code_stage.rb
@@ -13,12 +13,16 @@ def name
         :code
       end
 
+      def title
+        s_('CycleAnalyticsStage|Code')
+      end
+
       def legend
         _("Related Merge Requests")
       end
 
       def description
-        "Time until first merge request"
+        _("Time until first merge request")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/issue_stage.rb b/lib/gitlab/cycle_analytics/issue_stage.rb
index 08e249116a44b..7b03811efb21e 100644
--- a/lib/gitlab/cycle_analytics/issue_stage.rb
+++ b/lib/gitlab/cycle_analytics/issue_stage.rb
@@ -14,12 +14,16 @@ def name
         :issue
       end
 
+      def title
+        s_('CycleAnalyticsStage|Issue')
+      end
+
       def legend
         _("Related Issues")
       end
 
       def description
-        "Time before an issue gets scheduled"
+        _("Time before an issue gets scheduled")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/plan_stage.rb b/lib/gitlab/cycle_analytics/plan_stage.rb
index c5b2809209ced..1a0afb56b4fb0 100644
--- a/lib/gitlab/cycle_analytics/plan_stage.rb
+++ b/lib/gitlab/cycle_analytics/plan_stage.rb
@@ -14,12 +14,16 @@ def name
         :plan
       end
 
+      def title
+        s_('CycleAnalyticsStage|Plan')
+      end
+
       def legend
         _("Related Commits")
       end
 
       def description
-        "Time before an issue starts implementation"
+        _("Time before an issue starts implementation")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/production_stage.rb b/lib/gitlab/cycle_analytics/production_stage.rb
index 8f05c840db195..0fa8a65cb9911 100644
--- a/lib/gitlab/cycle_analytics/production_stage.rb
+++ b/lib/gitlab/cycle_analytics/production_stage.rb
@@ -15,12 +15,16 @@ def name
         :production
       end
 
+      def title
+        s_('CycleAnalyticsStage|Production')
+      end
+
       def legend
         _("Related Issues")
       end
 
       def description
-        "From issue creation until deploy to production"
+        _("From issue creation until deploy to production")
       end
 
       def query
diff --git a/lib/gitlab/cycle_analytics/review_stage.rb b/lib/gitlab/cycle_analytics/review_stage.rb
index 4f4c94f4020d4..2cab363ba9d4e 100644
--- a/lib/gitlab/cycle_analytics/review_stage.rb
+++ b/lib/gitlab/cycle_analytics/review_stage.rb
@@ -13,12 +13,16 @@ def name
         :review
       end
 
+      def title
+        s_('CycleAnalyticsStage|Review')
+      end
+
       def legend
         _("Relative Merged Requests")
       end
 
       def description
-        "Time between merge request creation and merge/close"
+        _("Time between merge request creation and merge/close")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/staging_stage.rb b/lib/gitlab/cycle_analytics/staging_stage.rb
index da8eb8c55de74..c1753340842b0 100644
--- a/lib/gitlab/cycle_analytics/staging_stage.rb
+++ b/lib/gitlab/cycle_analytics/staging_stage.rb
@@ -14,12 +14,16 @@ def name
         :staging
       end
 
+      def title
+        s_('CycleAnalyticsStage|Staging')
+      end
+
       def legend
         _("Relative Deployed Builds")
       end
 
       def description
-        "From merge request merge until deploy to production"
+        _("From merge request merge until deploy to production")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/summary/base.rb b/lib/gitlab/cycle_analytics/summary/base.rb
index 43fa3795e5ca9..a917ddccac732 100644
--- a/lib/gitlab/cycle_analytics/summary/base.rb
+++ b/lib/gitlab/cycle_analytics/summary/base.rb
@@ -8,7 +8,7 @@ def initialize(project:, from:)
         end
 
         def title
-          self.class.name.demodulize
+          raise NotImplementedError.new("Expected #{self.name} to implement title")
         end
 
         def value
diff --git a/lib/gitlab/cycle_analytics/summary/commit.rb b/lib/gitlab/cycle_analytics/summary/commit.rb
index 7b8faa4d854ed..bea7886275718 100644
--- a/lib/gitlab/cycle_analytics/summary/commit.rb
+++ b/lib/gitlab/cycle_analytics/summary/commit.rb
@@ -2,6 +2,10 @@ module Gitlab
   module CycleAnalytics
     module Summary
       class Commit < Base
+        def title
+          n_('Commit', 'Commits', value)
+        end
+
         def value
           @value ||= count_commits
         end
diff --git a/lib/gitlab/cycle_analytics/summary/deploy.rb b/lib/gitlab/cycle_analytics/summary/deploy.rb
index 06032e9200edf..099d798aac691 100644
--- a/lib/gitlab/cycle_analytics/summary/deploy.rb
+++ b/lib/gitlab/cycle_analytics/summary/deploy.rb
@@ -2,6 +2,10 @@ module Gitlab
   module CycleAnalytics
     module Summary
       class Deploy < Base
+        def title
+          n_('Deploy', 'Deploys', value)
+        end
+
         def value
           @value ||= @project.deployments.where("created_at > ?", @from).count
         end
diff --git a/lib/gitlab/cycle_analytics/summary/issue.rb b/lib/gitlab/cycle_analytics/summary/issue.rb
index 008468f24b90a..9bbf7a2685f52 100644
--- a/lib/gitlab/cycle_analytics/summary/issue.rb
+++ b/lib/gitlab/cycle_analytics/summary/issue.rb
@@ -9,7 +9,7 @@ def initialize(project:, from:, current_user:)
         end
 
         def title
-          'New Issue'
+          n_('New Issue', 'New Issues', value)
         end
 
         def value
diff --git a/lib/gitlab/cycle_analytics/test_stage.rb b/lib/gitlab/cycle_analytics/test_stage.rb
index 3ef7132bcc398..10402a1d7cde2 100644
--- a/lib/gitlab/cycle_analytics/test_stage.rb
+++ b/lib/gitlab/cycle_analytics/test_stage.rb
@@ -13,12 +13,16 @@ def name
         :test
       end
 
+      def title
+        s_('CycleAnalyticsStage|Test')
+      end
+
       def legend
         _("Relative Builds Trigger by Commits")
       end
 
       def description
-        "Total test time for all commits/merges"
+        _("Total test time for all commits/merges")
       end
 
       def stage_query
diff --git a/lib/gitlab/i18n.rb b/lib/gitlab/i18n.rb
index a7addee0dcd15..9081ced0238ee 100644
--- a/lib/gitlab/i18n.rb
+++ b/lib/gitlab/i18n.rb
@@ -11,5 +11,15 @@ module I18n
     def available_locales
       AVAILABLE_LANGUAGES.keys
     end
+
+    def set_locale(current_user)
+      requested_locale = current_user&.preferred_language || ::I18n.default_locale
+      locale = FastGettext.set_locale(requested_locale)
+      ::I18n.locale = locale
+    end
+
+    def reset_locale
+      ::I18n.locale = ::I18n.default_locale
+    end
   end
 end
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index ea19349bac10e..3e7bcf2c90fb5 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -17,18 +17,17 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
-msgid "Author|by"
+msgid "ByAuthor|by"
 msgstr ""
 
-msgid "Commits"
-msgstr ""
+msgid "Commit"
+msgid_plural "Commits"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
-msgid "CycleAnalyticsStage"
-msgstr ""
-
 msgid "CycleAnalyticsStage|Code"
 msgstr ""
 
@@ -50,7 +49,21 @@ msgstr ""
 msgid "CycleAnalyticsStage|Test"
 msgstr ""
 
-msgid "Deploys"
+msgid "Deploy"
+msgid_plural "Deploys"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "FirstPushedBy|First"
+msgstr ""
+
+msgid "FirstPushedBy|pushed by"
+msgstr ""
+
+msgid "From issue creation until deploy to production"
+msgstr ""
+
+msgid "From merge request merge until deploy to production"
 msgstr ""
 
 msgid "Introducing Cycle Analytics"
@@ -69,8 +82,10 @@ msgstr[1] ""
 msgid "Median"
 msgstr ""
 
-msgid "New Issues"
-msgstr ""
+msgid "New Issue"
+msgid_plural "New Issues"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Not available"
 msgstr ""
@@ -78,10 +93,10 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
-msgid "OfFirstTime|First"
+msgid "Opened"
 msgstr ""
 
-msgid "Opened"
+msgid "OpenedNDaysAgo|Opened"
 msgstr ""
 
 msgid "Pipeline Health"
@@ -149,6 +164,18 @@ msgstr ""
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
 msgstr ""
 
+msgid "Time before an issue gets scheduled"
+msgstr ""
+
+msgid "Time before an issue starts implementation"
+msgstr ""
+
+msgid "Time between merge request creation and merge/close"
+msgstr ""
+
+msgid "Time until first merge request"
+msgstr ""
+
 msgid "Time|hr"
 msgid_plural "Time|hrs"
 msgstr[0] ""
@@ -165,6 +192,9 @@ msgstr ""
 msgid "Total Time"
 msgstr ""
 
+msgid "Total test time for all commits/merges"
+msgstr ""
+
 msgid "Want to see the data? Please ask an administrator for access."
 msgstr ""
 
@@ -178,6 +208,3 @@ msgid "day"
 msgid_plural "days"
 msgstr[0] ""
 msgstr[1] ""
-
-msgid "pushed by"
-msgstr ""
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 2fed2aa1f57e9..9f75b5641a4ae 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -17,18 +17,17 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "\n"
 
-msgid "Author|by"
+msgid "ByAuthor|by"
 msgstr ""
 
-msgid "Commits"
-msgstr ""
+msgid "Commit"
+msgid_plural "Commits"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
-msgid "CycleAnalyticsStage"
-msgstr ""
-
 msgid "CycleAnalyticsStage|Code"
 msgstr ""
 
@@ -50,7 +49,21 @@ msgstr ""
 msgid "CycleAnalyticsStage|Test"
 msgstr ""
 
-msgid "Deploys"
+msgid "Deploy"
+msgid_plural "Deploys"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "FirstPushedBy|First"
+msgstr ""
+
+msgid "FirstPushedBy|pushed by"
+msgstr ""
+
+msgid "From issue creation until deploy to production"
+msgstr ""
+
+msgid "From merge request merge until deploy to production"
 msgstr ""
 
 msgid "Introducing Cycle Analytics"
@@ -69,8 +82,10 @@ msgstr[1] ""
 msgid "Median"
 msgstr ""
 
-msgid "New Issues"
-msgstr ""
+msgid "New Issue"
+msgid_plural "New Issues"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Not available"
 msgstr ""
@@ -78,10 +93,10 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
-msgid "OfFirstTime|First"
+msgid "Opened"
 msgstr ""
 
-msgid "Opened"
+msgid "OpenedNDaysAgo|Opened"
 msgstr ""
 
 msgid "Pipeline Health"
@@ -149,6 +164,18 @@ msgstr ""
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
 msgstr ""
 
+msgid "Time before an issue gets scheduled"
+msgstr ""
+
+msgid "Time before an issue starts implementation"
+msgstr ""
+
+msgid "Time between merge request creation and merge/close"
+msgstr ""
+
+msgid "Time until first merge request"
+msgstr ""
+
 msgid "Time|hr"
 msgid_plural "Time|hrs"
 msgstr[0] ""
@@ -165,6 +192,9 @@ msgstr ""
 msgid "Total Time"
 msgstr ""
 
+msgid "Total test time for all commits/merges"
+msgstr ""
+
 msgid "Want to see the data? Please ask an administrator for access."
 msgstr ""
 
@@ -178,6 +208,3 @@ msgid "day"
 msgid_plural "days"
 msgstr[0] ""
 msgstr[1] ""
-
-msgid "pushed by"
-msgstr ""
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 254b7ad18020d..6806bf216774e 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-05-03 12:28-0500\n"
+"PO-Revision-Date: 2017-05-03 21:03-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -17,18 +17,17 @@ msgstr ""
 "Last-Translator: \n"
 "X-Generator: Poedit 2.0.1\n"
 
-msgid "Author|by"
+msgid "ByAuthor|by"
 msgstr "por"
 
-msgid "Commits"
-msgstr "Cambios"
+msgid "Commit"
+msgid_plural "Commits"
+msgstr[0] "Cambio"
+msgstr[1] "Cambios"
 
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr "Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."
 
-msgid "CycleAnalyticsStage"
-msgstr ""
-
 msgid "CycleAnalyticsStage|Code"
 msgstr "Código"
 
@@ -51,8 +50,22 @@ msgstr "Puesta en escena"
 msgid "CycleAnalyticsStage|Test"
 msgstr "Pruebas"
 
-msgid "Deploys"
-msgstr "Despliegues"
+msgid "Deploy"
+msgid_plural "Deploys"
+msgstr[0] "Despliegue"
+msgstr[1] "Despliegues"
+
+msgid "FirstPushedBy|First"
+msgstr "Primer"
+
+msgid "FirstPushedBy|pushed by"
+msgstr "enviado por"
+
+msgid "From issue creation until deploy to production"
+msgstr "Desde la creación de la incidencia hasta el despliegue a producción"
+
+msgid "From merge request merge until deploy to production"
+msgstr "Desde la integración de la solicitud de fusión hasta el despliegue a producción"
 
 msgid "Introducing Cycle Analytics"
 msgstr "Introducción a Cycle Analytics"
@@ -70,8 +83,10 @@ msgstr[1] "Limitado a mostrar máximo %d eventos"
 msgid "Median"
 msgstr "Mediana"
 
-msgid "New Issues"
-msgstr "Nuevas incidencias"
+msgid "New Issue"
+msgid_plural "New Issues"
+msgstr[0] "Nueva incidencia"
+msgstr[1] "Nuevas incidencias"
 
 msgid "Not available"
 msgstr "No disponible"
@@ -79,12 +94,12 @@ msgstr "No disponible"
 msgid "Not enough data"
 msgstr "No hay suficientes datos"
 
-msgid "OfFirstTime|First"
-msgstr "Primer"
-
 msgid "Opened"
 msgstr "Abiertos"
 
+msgid "OpenedNDaysAgo|Opened"
+msgstr "Abierto"
+
 msgid "Pipeline Health"
 msgstr "Estado del Pipeline"
 
@@ -150,6 +165,18 @@ msgstr "El tiempo utilizado por cada entrada de datos obtenido por esa etapa."
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
 msgstr "El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."
 
+msgid "Time before an issue gets scheduled"
+msgstr "Tiempo antes de que una incidencia sea programada"
+
+msgid "Time before an issue starts implementation"
+msgstr "Tiempo antes de que empieze la implementación de una incidencia"
+
+msgid "Time between merge request creation and merge/close"
+msgstr "Tiempo entre la creación de la solicitud de fusión y la integración o cierre de esta"
+
+msgid "Time until first merge request"
+msgstr "Tiempo hasta la primera solicitud de fusión"
+
 msgid "Time|hr"
 msgid_plural "Time|hrs"
 msgstr[0] ""
@@ -166,6 +193,9 @@ msgstr ""
 msgid "Total Time"
 msgstr "Tiempo Total"
 
+msgid "Total test time for all commits/merges"
+msgstr "Tiempo total de pruebas para todos los cambios o integraciones"
+
 msgid "Want to see the data? Please ask an administrator for access."
 msgstr "¿Quieres ver los datos? Por favor pedir acceso al administrador."
 
@@ -179,6 +209,3 @@ msgid "day"
 msgid_plural "days"
 msgstr[0] "día"
 msgstr[1] "días"
-
-msgid "pushed by"
-msgstr "enviado por"
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index 8f4dc30b8c861..e1796d519351f 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-05-03 12:32-0500\n"
-"PO-Revision-Date: 2017-05-03 12:32-0500\n"
+"POT-Creation-Date: 2017-05-03 20:53-0500\n"
+"PO-Revision-Date: 2017-05-03 20:53-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -18,18 +18,17 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
 
-msgid "Author|by"
+msgid "ByAuthor|by"
 msgstr ""
 
-msgid "Commits"
-msgstr ""
+msgid "Commit"
+msgid_plural "Commits"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
 msgstr ""
 
-msgid "CycleAnalyticsStage"
-msgstr ""
-
 msgid "CycleAnalyticsStage|Code"
 msgstr ""
 
@@ -51,7 +50,21 @@ msgstr ""
 msgid "CycleAnalyticsStage|Test"
 msgstr ""
 
-msgid "Deploys"
+msgid "Deploy"
+msgid_plural "Deploys"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "FirstPushedBy|First"
+msgstr ""
+
+msgid "FirstPushedBy|pushed by"
+msgstr ""
+
+msgid "From issue creation until deploy to production"
+msgstr ""
+
+msgid "From merge request merge until deploy to production"
 msgstr ""
 
 msgid "Introducing Cycle Analytics"
@@ -70,8 +83,10 @@ msgstr[1] ""
 msgid "Median"
 msgstr ""
 
-msgid "New Issues"
-msgstr ""
+msgid "New Issue"
+msgid_plural "New Issues"
+msgstr[0] ""
+msgstr[1] ""
 
 msgid "Not available"
 msgstr ""
@@ -79,10 +94,10 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
-msgid "OfFirstTime|First"
+msgid "Opened"
 msgstr ""
 
-msgid "Opened"
+msgid "OpenedNDaysAgo|Opened"
 msgstr ""
 
 msgid "Pipeline Health"
@@ -150,6 +165,18 @@ msgstr ""
 msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
 msgstr ""
 
+msgid "Time before an issue gets scheduled"
+msgstr ""
+
+msgid "Time before an issue starts implementation"
+msgstr ""
+
+msgid "Time between merge request creation and merge/close"
+msgstr ""
+
+msgid "Time until first merge request"
+msgstr ""
+
 msgid "Time|hr"
 msgid_plural "Time|hrs"
 msgstr[0] ""
@@ -166,6 +193,9 @@ msgstr ""
 msgid "Total Time"
 msgstr ""
 
+msgid "Total test time for all commits/merges"
+msgstr ""
+
 msgid "Want to see the data? Please ask an administrator for access."
 msgstr ""
 
@@ -179,6 +209,3 @@ msgid "day"
 msgid_plural "days"
 msgstr[0] ""
 msgstr[1] ""
-
-msgid "pushed by"
-msgstr ""
diff --git a/locale/unfound_translations.rb b/locale/unfound_translations.rb
deleted file mode 100644
index e66211c161923..0000000000000
--- a/locale/unfound_translations.rb
+++ /dev/null
@@ -1,10 +0,0 @@
-N_('Commits')
-N_('CycleAnalyticsStage|Code')
-N_('CycleAnalyticsStage|Issue')
-N_('CycleAnalyticsStage|Plan')
-N_('CycleAnalyticsStage|Production')
-N_('CycleAnalyticsStage|Review')
-N_('CycleAnalyticsStage|Staging')
-N_('CycleAnalyticsStage|Test')
-N_('Deploys')
-N_('New Issues')
-- 
GitLab


From 154610c92bec0f94067ab88f35c5afdf6f10791d Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Thu, 4 May 2017 14:54:25 +1100
Subject: [PATCH 191/363] Fix feature tests

---
 app/models/snippet.rb | 5 +++++
 1 file changed, 5 insertions(+)

diff --git a/app/models/snippet.rb b/app/models/snippet.rb
index d8860718cb581..abfbefdf9a0fb 100644
--- a/app/models/snippet.rb
+++ b/app/models/snippet.rb
@@ -12,6 +12,11 @@ class Snippet < ActiveRecord::Base
   cache_markdown_field :title, pipeline: :single_line
   cache_markdown_field :content
 
+  # Aliases to make application_helper#edited_time_ago_with_tooltip helper work properly with snippets.
+  # See https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10392/diffs#note_28719102
+  alias_attribute :last_edited_at, :updated_at
+  alias_attribute :last_edited_by, :updated_by
+
   # If file_name changes, it invalidates content
   alias_method :default_content_html_invalidator, :content_html_invalidated?
   def content_html_invalidated?
-- 
GitLab


From b44eaf8e0772d4ab076bd0df10b13085483e5e66 Mon Sep 17 00:00:00 2001
From: Timothy Andrew <mail@timothyandrew.net>
Date: Wed, 3 May 2017 09:29:49 +0000
Subject: [PATCH 192/363] Sort the network graph both by commit date and
 topographically.

- Previously, we sorted commits by date, which seemed to work okay.

- The one edge case where this failed was when multiple commits have the same
  commit date (for example: when a range of commits are cherry picked with a
  single command, they all have the same commit date [and different author
  dates]).

- Commits with the same commit date would be sorted arbitrarily, and usually
  break the network graph.

- This commit solves the problem by both sorting by date, and by sorting
  topographically (parents aren't displayed until all their children are
  displayed)

- Include review comments from @adamniedzielski

A more detailed explanation is present here:
https://gitlab.com/gitlab-org/gitlab-ce/issues/30973#note_28706230
---
 ...-network-graph-sorted-by-date-and-topo.yml |  4 ++++
 lib/gitlab/git/repository.rb                  | 17 ++++++++-------
 spec/lib/gitlab/git/repository_spec.rb        |  2 +-
 spec/models/network/graph_spec.rb             | 21 ++++++++++++++++---
 4 files changed, 33 insertions(+), 11 deletions(-)
 create mode 100644 changelogs/unreleased/30973-network-graph-sorted-by-date-and-topo.yml

diff --git a/changelogs/unreleased/30973-network-graph-sorted-by-date-and-topo.yml b/changelogs/unreleased/30973-network-graph-sorted-by-date-and-topo.yml
new file mode 100644
index 0000000000000..42426c1865e72
--- /dev/null
+++ b/changelogs/unreleased/30973-network-graph-sorted-by-date-and-topo.yml
@@ -0,0 +1,4 @@
+---
+title: Sort the network graph both by commit date and topographically
+merge_request: 11057
+author:
diff --git a/lib/gitlab/git/repository.rb b/lib/gitlab/git/repository.rb
index acd0037ee4f3b..684b2e7287533 100644
--- a/lib/gitlab/git/repository.rb
+++ b/lib/gitlab/git/repository.rb
@@ -499,8 +499,9 @@ def ref_name_for_sha(ref_path, sha)
       #     :contains is the commit contained by the refs from which to begin (SHA1 or name)
       #     :max_count is the maximum number of commits to fetch
       #     :skip is the number of commits to skip
-      #     :order is the commits order and allowed value is :none (default), :date, or :topo
-      #        commit ordering types are documented here:
+      #     :order is the commits order and allowed value is :none (default), :date,
+      #        :topo, or any combination of them (in an array). Commit ordering types
+      #        are documented here:
       #        http://www.rubydoc.info/github/libgit2/rugged/Rugged#SORT_NONE-constant)
       #
       def find_commits(options = {})
@@ -1290,16 +1291,18 @@ def gitaly_migrate(method, &block)
         raise CommandError.new(e)
       end
 
-      # Returns the `Rugged` sorting type constant for a given
-      # sort type key. Valid keys are `:none`, `:topo`, and `:date`
-      def rugged_sort_type(key)
+      # Returns the `Rugged` sorting type constant for one or more given
+      # sort types. Valid keys are `:none`, `:topo`, and `:date`, or an array
+      # containing more than one of them. `:date` uses a combination of date and
+      # topological sorting to closer mimic git's native ordering.
+      def rugged_sort_type(sort_type)
         @rugged_sort_types ||= {
           none: Rugged::SORT_NONE,
           topo: Rugged::SORT_TOPO,
-          date: Rugged::SORT_DATE
+          date: Rugged::SORT_DATE | Rugged::SORT_TOPO
         }
 
-        @rugged_sort_types.fetch(key, Rugged::SORT_NONE)
+        @rugged_sort_types.fetch(sort_type, Rugged::SORT_NONE)
       end
     end
   end
diff --git a/spec/lib/gitlab/git/repository_spec.rb b/spec/lib/gitlab/git/repository_spec.rb
index ddedb7c34438b..fea186fd4f457 100644
--- a/spec/lib/gitlab/git/repository_spec.rb
+++ b/spec/lib/gitlab/git/repository_spec.rb
@@ -1062,7 +1062,7 @@ def commit_files(commit)
       end
 
       it "allows ordering by date" do
-        expect_any_instance_of(Rugged::Walker).to receive(:sorting).with(Rugged::SORT_DATE)
+        expect_any_instance_of(Rugged::Walker).to receive(:sorting).with(Rugged::SORT_DATE | Rugged::SORT_TOPO)
 
         repository.find_commits(order: :date)
       end
diff --git a/spec/models/network/graph_spec.rb b/spec/models/network/graph_spec.rb
index 46b36e11c230f..0fe8a591a4557 100644
--- a/spec/models/network/graph_spec.rb
+++ b/spec/models/network/graph_spec.rb
@@ -10,17 +10,17 @@
     expect(graph.notes).to eq( { note_on_commit.commit_id => 1 } )
   end
 
-  describe "#commits" do
+  describe '#commits' do
     let(:graph) { described_class.new(project, 'refs/heads/master', project.repository.commit, nil) }
 
-    it "returns a list of commits" do
+    it 'returns a list of commits' do
       commits = graph.commits
 
       expect(commits).not_to be_empty
       expect(commits).to all( be_kind_of(Network::Commit) )
     end
 
-    it "sorts the commits by commit date (descending)" do
+    it 'it the commits by commit date (descending)' do
       # Remove duplicate timestamps because they make it harder to
       # assert that the commits are sorted as expected.
       commits = graph.commits.uniq(&:date)
@@ -29,5 +29,20 @@
       expect(commits).not_to be_empty
       expect(commits.map(&:id)).to eq(sorted_commits.map(&:id))
     end
+
+    it 'sorts children before parents for commits with the same timestamp' do
+      commits_by_time = graph.commits.group_by(&:date)
+
+      commits_by_time.each do |time, commits|
+        commit_ids = commits.map(&:id)
+
+        commits.each_with_index do |commit, index|
+          parent_indexes = commit.parent_ids.map { |parent_id| commit_ids.find_index(parent_id) }.compact
+
+          # All parents of the current commit should appear after it
+          expect(parent_indexes).to all( be > index )
+        end
+      end
+    end
   end
 end
-- 
GitLab


From 628d641b067fd0c0ba16fc7f6d68bcd241a7a105 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 4 May 2017 01:58:57 -0500
Subject: [PATCH 193/363] Disable FastGettext from translating AR attrs.

It isn't working fine when using POROs in forms like WikiPage,
the following error is being raised: undefined method `abstract_class?' for Object:Class
---
 config/application.rb | 3 +++
 1 file changed, 3 insertions(+)

diff --git a/config/application.rb b/config/application.rb
index f2ecc4ce77c7e..32ad239364876 100644
--- a/config/application.rb
+++ b/config/application.rb
@@ -40,6 +40,9 @@ class Application < Rails::Application
     # config.i18n.default_locale = :de
     config.i18n.enforce_available_locales = false
 
+    # Translation for AR attrs is not working well for POROs like WikiPage
+    config.gettext_i18n_rails.use_for_active_record_attributes = false
+
     # Configure the default encoding used in templates for Ruby 1.9.
     config.encoding = "utf-8"
 
-- 
GitLab


From 38c29f8775e393164e04fb1faf6c6b05fd105df6 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=A9my=20Coutable?= <remy@rymai.me>
Date: Thu, 4 May 2017 09:02:39 +0200
Subject: [PATCH 194/363] Improving copy of CONTRIBUTING.md, PROCESS.md, and
 code_review.md
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Signed-off-by: Rémy Coutable <remy@rymai.me>
---
 CONTRIBUTING.md                | 23 ++++++++++-------------
 PROCESS.md                     |  2 +-
 doc/development/code_review.md |  3 ---
 3 files changed, 11 insertions(+), 17 deletions(-)

diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index f27efb0ae85b2..600dad563a6a3 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -136,7 +136,7 @@ and ~"direction".
 A number of type labels have a priority assigned to them, which automatically
 makes them float to the top, depending on their importance.
 
-Type labels are always lowercase, but can have any color, besides blue (which is
+Type labels are always lowercase, and can have any color, besides blue (which is
 already reserved for subject labels).
 
 The descriptions on the [labels page][labels-page] explain what falls under each type label.
@@ -153,7 +153,7 @@ issue is labelled with a subject label corresponding to your expertise.
 Examples of subject labels are ~wiki, ~"container registry", ~ldap, ~api,
 ~issues, ~"merge requests", ~labels, and ~"container registry".
 
-Subject labels are always colored blue and all-lowercase.
+Subject labels are always all-lowercase.
 
 ### Team labels (~CI, ~Discussion, ~Edge, ~Frontend, ~Platform, etc.)
 
@@ -167,8 +167,8 @@ The current team labels are ~Build, ~CI, ~Discussion, ~Documentation, ~Edge,
 The descriptions on the [labels page][labels-page] explain what falls under the
 responsibility of each team.
 
-Team labels are always colored aqua, and are capitalized so that they show up as
-the first label for any issue.
+Team labels are always capitalized so that they show up as the first label for
+any issue.
 
 ### Priority labels (~Deliverable and ~Stretch)
 
@@ -255,12 +255,9 @@ every quarter.
 
 The most important thing is making sure valid issues receive feedback from the
 development team. Therefore the priority is mentioning developers that can help
-on those issues. Please select someone with relevant experience from
-[GitLab team][team]. If there is nobody mentioned with that expertise
-look in the commit history for the affected files to find someone. Avoid
-mentioning the lead developer, this is the person that is least likely to give a
-timely response. If the involvement of the lead developer is needed the other
-core team members will mention this person.
+on those issues. Please select someone with relevant experience from the
+[GitLab team][team]. If there is nobody mentioned with that expertise look in
+the commit history for the affected files to find someone.
 
 [described in our handbook]: https://about.gitlab.com/handbook/engineering/issues/issue-triage-policies/
 [issue bash events]: https://gitlab.com/gitlab-org/gitlab-ce/issues/17815
@@ -535,11 +532,11 @@ the feature you contribute through all of these steps.
 1. [Unit and system tests][testing] that pass on the CI server
 1. Performance/scalability implications have been considered, addressed, and tested
 1. [Documented][doc-styleguide] in the `/doc` directory
-1. [Changelog entry added][changelog]
+1. [Changelog entry added][changelog], if necessary
 1. Reviewed and any concerns are addressed
 1. Merged by a project maintainer
-1. Added to the release blog article if relevant
-1. Added to [the website](https://gitlab.com/gitlab-com/www-gitlab-com/) if relevant
+1. Added to the release blog article, if relevant
+1. Added to [the website](https://gitlab.com/gitlab-com/www-gitlab-com/), if relevant
 1. Community questions answered
 1. Answers to questions radiated (in docs/wiki/support etc.)
 
diff --git a/PROCESS.md b/PROCESS.md
index 6d7d155ca6cf1..3b97a4e8c75fc 100644
--- a/PROCESS.md
+++ b/PROCESS.md
@@ -1,4 +1,4 @@
-## GitLab Core Team & GitLab Inc. Team Contributing Process
+## GitLab Core Team & GitLab Inc. Contribution Process
 
 ---
 
diff --git a/doc/development/code_review.md b/doc/development/code_review.md
index 138817f5440f4..be3dd1e2cc69b 100644
--- a/doc/development/code_review.md
+++ b/doc/development/code_review.md
@@ -95,8 +95,6 @@ experience, refactors the existing code). Then:
   "LGTM :thumbsup:", or "Just a couple things to address."
 - Assign the merge request to the author if changes are required following your
   review.
-- You should try to resolve merge conflicts yourself, using the [merge conflict
-  resolution][conflict-resolution] tool.
 - Set the milestone before merging a merge request.
 - Avoid accepting a merge request before the job succeeds. Of course, "Merge
   When Pipeline Succeeds" (MWPS) is fine.
@@ -105,7 +103,6 @@ experience, refactors the existing code). Then:
 - Consider using the [Squash and
   merge][squash-and-merge] feature when the merge request has a lot of commits.
 
-[conflict-resolution]: https://docs.gitlab.com/ce/user/project/merge_requests/resolve_conflicts.html#merge-conflict-resolution
 [squash-and-merge]: https://docs.gitlab.com/ee/user/project/merge_requests/squash_and_merge.html#squash-and-merge
 
 ### The right balance
-- 
GitLab


From fdacc4ee6a3341b2c44ddd85a41f2a04d0d417ad Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Thu, 4 May 2017 08:54:42 +0100
Subject: [PATCH 195/363] Moved to a view spec

---
 app/views/projects/tags/index.html.haml       |  1 +
 spec/features/projects/tags/sort_spec.rb      | 15 --------------
 .../projects/tags/index.html.haml_spec.rb     | 20 +++++++++++++++++++
 3 files changed, 21 insertions(+), 15 deletions(-)
 delete mode 100644 spec/features/projects/tags/sort_spec.rb
 create mode 100644 spec/views/projects/tags/index.html.haml_spec.rb

diff --git a/app/views/projects/tags/index.html.haml b/app/views/projects/tags/index.html.haml
index c14bbf4f05f72..56656ea3d8600 100644
--- a/app/views/projects/tags/index.html.haml
+++ b/app/views/projects/tags/index.html.haml
@@ -1,4 +1,5 @@
 - @no_container = true
+- @sort ||= sort_value_recently_updated
 - page_title "Tags"
 = render "projects/commits/head"
 
diff --git a/spec/features/projects/tags/sort_spec.rb b/spec/features/projects/tags/sort_spec.rb
deleted file mode 100644
index 835cd1507fbfc..0000000000000
--- a/spec/features/projects/tags/sort_spec.rb
+++ /dev/null
@@ -1,15 +0,0 @@
-require 'spec_helper'
-
-feature 'Tags sort dropdown', :feature do
-  let(:project) { create(:project) }
-
-  before do
-    login_as(:admin)
-
-    visit namespace_project_tags_path(project.namespace, project)
-  end
-
-  it 'defaults sort dropdown to last updated' do
-    expect(page).to have_button('Last updated')
-  end
-end
diff --git a/spec/views/projects/tags/index.html.haml_spec.rb b/spec/views/projects/tags/index.html.haml_spec.rb
new file mode 100644
index 0000000000000..33122365e9ab7
--- /dev/null
+++ b/spec/views/projects/tags/index.html.haml_spec.rb
@@ -0,0 +1,20 @@
+require 'spec_helper'
+
+describe 'projects/tags/index', :view do
+  let(:project) { create(:project) }
+
+  before do
+    assign(:project, project)
+    assign(:repository, project.repository)
+    assign(:tags, [])
+
+    allow(view).to receive(:current_ref).and_return('master')
+    allow(view).to receive(:can?).and_return(false)
+  end
+
+  it 'defaults sort dropdown toggle to last updated' do
+    render
+
+    expect(rendered).to have_button('Last updated')
+  end
+end
-- 
GitLab


From 2f7f1ce4e66db847414e2fc3de09556e75c51eb4 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Thu, 4 May 2017 11:32:39 +0200
Subject: [PATCH 196/363] fix sidekiq spec, add changelog

---
 changelogs/unreleased/fix-admin-integrations.yml | 4 ++++
 config/sidekiq_queues.yml                        | 1 +
 2 files changed, 5 insertions(+)
 create mode 100644 changelogs/unreleased/fix-admin-integrations.yml

diff --git a/changelogs/unreleased/fix-admin-integrations.yml b/changelogs/unreleased/fix-admin-integrations.yml
new file mode 100644
index 0000000000000..7689623501ffc
--- /dev/null
+++ b/changelogs/unreleased/fix-admin-integrations.yml
@@ -0,0 +1,4 @@
+---
+title: Fix new admin integrations not taking effect on existing projects
+merge_request:
+author:
diff --git a/config/sidekiq_queues.yml b/config/sidekiq_queues.yml
index c3bd73533d079..91ea1c0f779ba 100644
--- a/config/sidekiq_queues.yml
+++ b/config/sidekiq_queues.yml
@@ -53,3 +53,4 @@
   - [pages, 1]
   - [system_hook_push, 1]
   - [update_user_activity, 1]
+  - [propagate_project_service, 1]
-- 
GitLab


From f81cf84035213002ce7931af6c3ffa917fe7fcbd Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Thu, 4 May 2017 12:13:33 +0200
Subject: [PATCH 197/363] refactor worker into service

---
 app/services/projects/propagate_service.rb    | 47 +++++++++++++++++
 .../propagate_project_service_worker.rb       | 34 ++----------
 .../projects/propagate_service_spec.rb        | 40 ++++++++++++++
 .../propagate_project_service_worker_spec.rb  | 52 +++++++------------
 4 files changed, 110 insertions(+), 63 deletions(-)
 create mode 100644 app/services/projects/propagate_service.rb
 create mode 100644 spec/services/projects/propagate_service_spec.rb

diff --git a/app/services/projects/propagate_service.rb b/app/services/projects/propagate_service.rb
new file mode 100644
index 0000000000000..3c05dcce07c7a
--- /dev/null
+++ b/app/services/projects/propagate_service.rb
@@ -0,0 +1,47 @@
+module Projects
+  class PropagateService
+    BATCH_SIZE = 100
+
+    def self.propagate!(*args)
+      new(*args).propagate!
+    end
+
+    def initialize(template)
+      @template = template
+    end
+
+    def propagate!
+      return unless @template&.active
+
+      Rails.logger.info("Propagating services for template #{@template.id}")
+
+      propagate_projects_with_template
+    end
+
+    private
+
+    def propagate_projects_with_template
+      offset = 0
+
+      loop do
+        batch = project_ids_batch(offset)
+
+        batch.each { |project_id| create_from_template(project_id) }
+
+        break if batch.count < BATCH_SIZE
+
+        offset += BATCH_SIZE
+      end
+    end
+
+    def create_from_template(project_id)
+      Service.build_from_template(project_id, @template).save!
+    end
+
+    def project_ids_batch(offset)
+      Project.joins('LEFT JOIN services ON services.project_id = projects.id').
+        where('services.type != ? OR services.id IS NULL', @template.type).
+        limit(BATCH_SIZE).offset(offset).pluck(:id)
+    end
+  end
+end
diff --git a/app/workers/propagate_project_service_worker.rb b/app/workers/propagate_project_service_worker.rb
index 535517709682f..ab2b7738f9ad9 100644
--- a/app/workers/propagate_project_service_worker.rb
+++ b/app/workers/propagate_project_service_worker.rb
@@ -3,44 +3,18 @@ class PropagateProjectServiceWorker
   include Sidekiq::Worker
   include DedicatedSidekiqQueue
 
+  sidekiq_options retry: 3
+
   LEASE_TIMEOUT = 30.minutes.to_i
 
   def perform(template_id)
-    template = Service.find_by(id: template_id)
-
-    return unless template&.active
-    return unless try_obtain_lease_for(template.id)
-
-    Rails.logger.info("Propagating services for template #{template.id}")
+    return unless try_obtain_lease_for(template_id)
 
-    project_ids_for_template(template) do |project_id|
-      Service.build_from_template(project_id, template).save!
-    end
+    Projects::PropagateService.propagate!(Service.find_by(id: template_id))
   end
 
   private
 
-  def project_ids_for_template(template)
-    limit = 100
-    offset = 0
-
-    loop do
-      batch = project_ids_batch(limit, offset, template.type)
-
-      batch.each { |project_id| yield(project_id) }
-
-      break if batch.count < limit
-
-      offset += limit
-    end
-  end
-
-  def project_ids_batch(limit, offset, template_type)
-    Project.joins('LEFT JOIN services ON services.project_id = projects.id').
-      where('services.type != ? OR services.id IS NULL', template_type).
-      limit(limit).offset(offset).pluck(:id)
-  end
-
   def try_obtain_lease_for(template_id)
     Gitlab::ExclusiveLease.
       new("propagate_project_service_worker:#{template_id}", timeout: LEASE_TIMEOUT).
diff --git a/spec/services/projects/propagate_service_spec.rb b/spec/services/projects/propagate_service_spec.rb
new file mode 100644
index 0000000000000..ee40a7ecbabce
--- /dev/null
+++ b/spec/services/projects/propagate_service_spec.rb
@@ -0,0 +1,40 @@
+require 'spec_helper'
+
+describe Projects::PropagateService, services: true do
+  describe '.propagate!' do
+    let!(:service_template) do
+      PushoverService.create(
+        template: true,
+        active: true,
+        properties: {
+          device: 'MyDevice',
+          sound: 'mic',
+          priority: 4,
+          user_key: 'asdf',
+          api_key: '123456789'
+        })
+    end
+
+    let!(:project) { create(:empty_project) }
+
+    it 'creates services for projects' do
+      expect { described_class.propagate!(service_template) }.
+        to change { Service.count }.by(1)
+    end
+
+    it 'does not create the service if it exists already' do
+      Service.build_from_template(project.id, service_template).save!
+
+      expect { described_class.propagate!(service_template) }.
+        not_to change { Service.count }
+    end
+
+    it 'creates the service containing the template attributes' do
+      described_class.propagate!(service_template)
+
+      service = Service.find_by(type: service_template.type, template: false)
+
+      expect(service.properties).to eq(service_template.properties)
+    end
+  end
+end
diff --git a/spec/workers/propagate_project_service_worker_spec.rb b/spec/workers/propagate_project_service_worker_spec.rb
index d525a8b4a236b..c16e95bd49b6e 100644
--- a/spec/workers/propagate_project_service_worker_spec.rb
+++ b/spec/workers/propagate_project_service_worker_spec.rb
@@ -1,43 +1,29 @@
 require 'spec_helper'
 
 describe PropagateProjectServiceWorker do
-  describe '#perform' do
-    let!(:service_template) do
-      PushoverService.create(
-        template: true,
-        active: true,
-        properties: {
-          device: 'MyDevice',
-          sound: 'mic',
-          priority: 4,
-          user_key: 'asdf',
-          api_key: '123456789'
-        })
-    end
-
-    let!(:project) { create(:empty_project) }
-
-    before do
-      allow_any_instance_of(Gitlab::ExclusiveLease).to receive(:try_obtain).
-        and_return(true)
-    end
-
-    it 'creates services for projects' do
-      expect { subject.perform(service_template.id) }.to change { Service.count }.by(1)
-    end
+  let!(:service_template) do
+    PushoverService.create(
+      template: true,
+      active: true,
+      properties: {
+        device: 'MyDevice',
+        sound: 'mic',
+        priority: 4,
+        user_key: 'asdf',
+        api_key: '123456789'
+      })
+  end
 
-    it 'does not create the service if it exists already' do
-      Service.build_from_template(project.id, service_template).save!
+  before do
+    allow_any_instance_of(Gitlab::ExclusiveLease).to receive(:try_obtain).
+      and_return(true)
+  end
 
-      expect { subject.perform(service_template.id) }.not_to change { Service.count }
-    end
+  describe '#perform' do
+    it 'calls the propagate service with the template' do
+      expect(Projects::PropagateService).to receive(:propagate!).with(service_template)
 
-    it 'creates the service containing the template attributes' do
       subject.perform(service_template.id)
-
-      service = Service.find_by(type: service_template.type, template: false)
-
-      expect(service.properties).to eq(service_template.properties)
     end
   end
 end
-- 
GitLab


From 77bbc1c4af087d009fdfa3d7d4c9a720b7cc9c5b Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 4 May 2017 10:47:44 +0100
Subject: [PATCH 198/363] Reset schema and migration

---
 ...ientside_sentry_to_application_settings.rb | 65 +++++++++++++++++++
 1 file changed, 65 insertions(+)
 create mode 100644 db/migrate/20170504094058_add_clientside_sentry_to_application_settings.rb

diff --git a/db/migrate/20170504094058_add_clientside_sentry_to_application_settings.rb b/db/migrate/20170504094058_add_clientside_sentry_to_application_settings.rb
new file mode 100644
index 0000000000000..9462d85833412
--- /dev/null
+++ b/db/migrate/20170504094058_add_clientside_sentry_to_application_settings.rb
@@ -0,0 +1,65 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using the methods "add_concurrent_index", "remove_concurrent_index" or
+  # "add_column_with_default" you must disable the use of transactions
+  # as these methods can not run in an existing transaction.
+  # When using "add_concurrent_index" or "remove_concurrent_index" methods make sure
+  # that either of them is the _only_ method called in the migration,
+  # any other changes should go in a separate migration.
+  # This ensures that upon failure _only_ the index creation or removing fails
+  # and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  disable_ddl_transaction!
+
+  # Reversability was tested when I made an update the originally irreversible
+  # migration after I ran it, so I could test that the down method worked and then
+  # the up method worked.
+
+  def up
+    add_column_with_default :application_settings, :clientside_sentry_enabled, :boolean, default: false
+    add_column :application_settings, :clientside_sentry_dsn, :string
+  end
+
+  def down
+    remove_columns :application_settings, :clientside_sentry_enabled, :clientside_sentry_dsn
+  end
+end
+
+class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using the methods "add_concurrent_index" or "add_column_with_default"
+  # you must disable the use of transactions as these methods can not run in an
+  # existing transaction. When using "add_concurrent_index" make sure that this
+  # method is the _only_ method called in the migration, any other changes
+  # should go in a separate migration. This ensures that upon failure _only_ the
+  # index creation fails and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  disable_ddl_transaction!
+
+
+end
-- 
GitLab


From 3d807dc81b27c6366390b2355e40a5c65bbf02c2 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Thu, 4 May 2017 12:21:35 +0200
Subject: [PATCH 199/363] update lease timeout

---
 app/workers/propagate_project_service_worker.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/workers/propagate_project_service_worker.rb b/app/workers/propagate_project_service_worker.rb
index ab2b7738f9ad9..6cc3fe846356a 100644
--- a/app/workers/propagate_project_service_worker.rb
+++ b/app/workers/propagate_project_service_worker.rb
@@ -5,7 +5,7 @@ class PropagateProjectServiceWorker
 
   sidekiq_options retry: 3
 
-  LEASE_TIMEOUT = 30.minutes.to_i
+  LEASE_TIMEOUT = 4.hours.to_i
 
   def perform(template_id)
     return unless try_obtain_lease_for(template_id)
-- 
GitLab


From 1a8dc206a49a3812ce920ad94deda1ef57929c1b Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 4 May 2017 11:28:12 +0100
Subject: [PATCH 200/363] Reset migration to head

---
 ...ientside_sentry_to_application_settings.rb | 31 ---------
 ...ientside_sentry_to_application_settings.rb | 65 -------------------
 db/schema.rb                                  |  2 -
 3 files changed, 98 deletions(-)
 delete mode 100644 db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
 delete mode 100644 db/migrate/20170504094058_add_clientside_sentry_to_application_settings.rb

diff --git a/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb b/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
deleted file mode 100644
index f3e08aed81988..0000000000000
--- a/db/migrate/20170428123910_add_clientside_sentry_to_application_settings.rb
+++ /dev/null
@@ -1,31 +0,0 @@
-class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
-  include Gitlab::Database::MigrationHelpers
-
-  # Set this constant to true if this migration requires downtime.
-  DOWNTIME = false
-
-  # When a migration requires downtime you **must** uncomment the following
-  # constant and define a short and easy to understand explanation as to why the
-  # migration requires downtime.
-  # DOWNTIME_REASON = ''
-
-  # When using the methods "add_concurrent_index" or "add_column_with_default"
-  # you must disable the use of transactions as these methods can not run in an
-  # existing transaction. When using "add_concurrent_index" make sure that this
-  # method is the _only_ method called in the migration, any other changes
-  # should go in a separate migration. This ensures that upon failure _only_ the
-  # index creation fails and can be retried or reverted easily.
-  #
-  # To disable transactions uncomment the following line and remove these
-  # comments:
-  disable_ddl_transaction!
-
-  def up
-    add_column_with_default :application_settings, :clientside_sentry_enabled, :boolean, default: false
-    add_column :application_settings, :clientside_sentry_dsn, :string
-  end
-
-  def down
-    remove_columns :application_settings, :clientside_sentry_enabled, :clientside_sentry_dsn
-  end
-end
diff --git a/db/migrate/20170504094058_add_clientside_sentry_to_application_settings.rb b/db/migrate/20170504094058_add_clientside_sentry_to_application_settings.rb
deleted file mode 100644
index 9462d85833412..0000000000000
--- a/db/migrate/20170504094058_add_clientside_sentry_to_application_settings.rb
+++ /dev/null
@@ -1,65 +0,0 @@
-# See http://doc.gitlab.com/ce/development/migration_style_guide.html
-# for more information on how to write migrations for GitLab.
-
-class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
-  include Gitlab::Database::MigrationHelpers
-
-  # Set this constant to true if this migration requires downtime.
-  DOWNTIME = false
-
-  # When a migration requires downtime you **must** uncomment the following
-  # constant and define a short and easy to understand explanation as to why the
-  # migration requires downtime.
-  # DOWNTIME_REASON = ''
-
-  # When using the methods "add_concurrent_index", "remove_concurrent_index" or
-  # "add_column_with_default" you must disable the use of transactions
-  # as these methods can not run in an existing transaction.
-  # When using "add_concurrent_index" or "remove_concurrent_index" methods make sure
-  # that either of them is the _only_ method called in the migration,
-  # any other changes should go in a separate migration.
-  # This ensures that upon failure _only_ the index creation or removing fails
-  # and can be retried or reverted easily.
-  #
-  # To disable transactions uncomment the following line and remove these
-  # comments:
-  disable_ddl_transaction!
-
-  # Reversability was tested when I made an update the originally irreversible
-  # migration after I ran it, so I could test that the down method worked and then
-  # the up method worked.
-
-  def up
-    add_column_with_default :application_settings, :clientside_sentry_enabled, :boolean, default: false
-    add_column :application_settings, :clientside_sentry_dsn, :string
-  end
-
-  def down
-    remove_columns :application_settings, :clientside_sentry_enabled, :clientside_sentry_dsn
-  end
-end
-
-class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
-  include Gitlab::Database::MigrationHelpers
-
-  # Set this constant to true if this migration requires downtime.
-  DOWNTIME = false
-
-  # When a migration requires downtime you **must** uncomment the following
-  # constant and define a short and easy to understand explanation as to why the
-  # migration requires downtime.
-  # DOWNTIME_REASON = ''
-
-  # When using the methods "add_concurrent_index" or "add_column_with_default"
-  # you must disable the use of transactions as these methods can not run in an
-  # existing transaction. When using "add_concurrent_index" make sure that this
-  # method is the _only_ method called in the migration, any other changes
-  # should go in a separate migration. This ensures that upon failure _only_ the
-  # index creation fails and can be retried or reverted easily.
-  #
-  # To disable transactions uncomment the following line and remove these
-  # comments:
-  disable_ddl_transaction!
-
-
-end
diff --git a/db/schema.rb b/db/schema.rb
index 519a444833b3a..01c0f00c92496 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -121,8 +121,6 @@
     t.integer "cached_markdown_version"
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
-    t.boolean "clientside_sentry_enabled", default: false, null: false
-    t.string "clientside_sentry_dsn"
   end
 
   create_table "audit_events", force: :cascade do |t|
-- 
GitLab


From 944ed8da556a067746ac6e98e2e81781748eab48 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 4 May 2017 11:32:28 +0100
Subject: [PATCH 201/363] clientside_sentry migration and schema changes commit

---
 ...ientside_sentry_to_application_settings.rb | 33 +++++++++++++++++++
 db/schema.rb                                  |  4 ++-
 2 files changed, 36 insertions(+), 1 deletion(-)
 create mode 100644 db/migrate/20170504102911_add_clientside_sentry_to_application_settings.rb

diff --git a/db/migrate/20170504102911_add_clientside_sentry_to_application_settings.rb b/db/migrate/20170504102911_add_clientside_sentry_to_application_settings.rb
new file mode 100644
index 0000000000000..141112f8b5084
--- /dev/null
+++ b/db/migrate/20170504102911_add_clientside_sentry_to_application_settings.rb
@@ -0,0 +1,33 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using "add_concurrent_index" or "remove_concurrent_index" methods make sure
+  # that either of them is the _only_ method called in the migration,
+  # any other changes should go in a separate migration.
+  # This ensures that upon failure _only_ the index creation or removing fails
+  # and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  disable_ddl_transaction!
+
+  def up
+    add_column_with_default :application_settings, :clientside_sentry_enabled, :boolean, default: false
+    add_column :application_settings, :clientside_sentry_dsn, :string
+  end
+
+  def down
+    remove_columns :application_settings, :clientside_sentry_enabled, :clientside_sentry_dsn
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 01c0f00c92496..44148a9bc318d 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170502091007) do
+ActiveRecord::Schema.define(version: 20170504102911) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -121,6 +121,8 @@
     t.integer "cached_markdown_version"
     t.boolean "usage_ping_enabled", default: true, null: false
     t.string "uuid"
+    t.boolean "clientside_sentry_enabled", default: false, null: false
+    t.string "clientside_sentry_dsn"
   end
 
   create_table "audit_events", force: :cascade do |t|
-- 
GitLab


From 3bff8da8c1e3223e81bccd5343902b840f005fcf Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Thu, 4 May 2017 12:37:54 +0200
Subject: [PATCH 202/363] fix service spec

---
 app/models/service.rb       |  2 +-
 spec/models/service_spec.rb | 11 ++++++-----
 2 files changed, 7 insertions(+), 6 deletions(-)

diff --git a/app/models/service.rb b/app/models/service.rb
index dea22fd96a7aa..18c046aff5418 100644
--- a/app/models/service.rb
+++ b/app/models/service.rb
@@ -257,7 +257,7 @@ def self.build_from_template(project_id, template)
   def update_and_propagate(service_params)
     return false unless update_attributes(service_params)
 
-    if service_params[:active] == 1
+    if service_params[:active]
       PropagateProjectServiceWorker.perform_async(service_params[:id])
     end
 
diff --git a/spec/models/service_spec.rb b/spec/models/service_spec.rb
index e9acbb81ad99e..7a5fb509bf5d3 100644
--- a/spec/models/service_spec.rb
+++ b/spec/models/service_spec.rb
@@ -256,26 +256,27 @@
   end
 
   describe "#update_and_propagate" do
+    let(:project) { create(:empty_project) }
     let!(:service) do
       RedmineService.new(
         project: project,
         active: false,
         properties: {
-          project_url: 'http://redmine/projects/project_name_in_redmine',
-          issues_url: "http://redmine/#{project.id}/project_name_in_redmine/:id",
-          new_issue_url: 'http://redmine/projects/project_name_in_redmine/issues/new'
+          project_url: 'http://abc',
+          issues_url: 'http://abc',
+          new_issue_url: 'http://abc'
         }
       )
     end
 
     it 'updates the service params successfully and calls the propagation worker' do
-      expect(PropagateProjectServiceWorker).to receve(:perform_async)
+      expect(PropagateProjectServiceWorker).to receive(:perform_async)
 
       expect(service.update_and_propagate(active: true)).to be true
     end
 
     it 'updates the service params successfully' do
-      expect(PropagateProjectServiceWorker).not_to receve(:perform_asyncs)
+      expect(PropagateProjectServiceWorker).not_to receive(:perform_async)
 
       expect(service.update_and_propagate(properties: {})).to be true
     end
-- 
GitLab


From b871564383cbade7fff312b8f045cee6c871f1e0 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Thu, 4 May 2017 12:46:03 +0200
Subject: [PATCH 203/363] fix service spec

---
 app/models/service.rb       | 2 +-
 spec/models/service_spec.rb | 4 ++--
 2 files changed, 3 insertions(+), 3 deletions(-)

diff --git a/app/models/service.rb b/app/models/service.rb
index 18c046aff5418..f85343877035e 100644
--- a/app/models/service.rb
+++ b/app/models/service.rb
@@ -258,7 +258,7 @@ def update_and_propagate(service_params)
     return false unless update_attributes(service_params)
 
     if service_params[:active]
-      PropagateProjectServiceWorker.perform_async(service_params[:id])
+      PropagateProjectServiceWorker.perform_async(id)
     end
 
     true
diff --git a/spec/models/service_spec.rb b/spec/models/service_spec.rb
index 7a5fb509bf5d3..3a7d8b729938b 100644
--- a/spec/models/service_spec.rb
+++ b/spec/models/service_spec.rb
@@ -258,7 +258,7 @@
   describe "#update_and_propagate" do
     let(:project) { create(:empty_project) }
     let!(:service) do
-      RedmineService.new(
+      RedmineService.create(
         project: project,
         active: false,
         properties: {
@@ -270,7 +270,7 @@
     end
 
     it 'updates the service params successfully and calls the propagation worker' do
-      expect(PropagateProjectServiceWorker).to receive(:perform_async)
+      expect(PropagateProjectServiceWorker).to receive(:perform_async).with(service.id)
 
       expect(service.update_and_propagate(active: true)).to be true
     end
-- 
GitLab


From 7a37ada12f8c2dc033594fecddf63345f1addf37 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Thu, 4 May 2017 06:07:36 -0600
Subject: [PATCH 204/363] use refs when possible - style changes - better
 initial title and description data types

---
 .../issue_show/issue_title_description.vue    | 52 ++++++++++++-------
 1 file changed, 34 insertions(+), 18 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index f6c3308388cd5..8d2ef39040b36 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -6,8 +6,14 @@ import tasks from './actions/tasks';
 
 export default {
   props: {
-    endpoint: { required: true, type: String },
-    candescription: { required: true, type: String },
+    endpoint: {
+      required: true,
+      type: String,
+    },
+    candescription: {
+      required: true,
+      type: String,
+    },
   },
   data() {
     const resource = new Service(this.$http, this.endpoint);
@@ -32,17 +38,18 @@ export default {
       poll,
       apiData: {},
       timeoutId: null,
-      title: '<span></span>',
+      title: null,
       titleText: '',
-      description: '<span></span>',
+      description: null,
       descriptionText: '',
       descriptionChange: false,
       tasks: '0 of 0',
+      titleEl: document.querySelector('title'),
     };
   },
   methods: {
     renderResponse(res) {
-      this.apiData = JSON.parse(res.body);
+      this.apiData = res.json();
       this.triggerAnimation();
     },
     updateTaskHTML() {
@@ -53,7 +60,7 @@ export default {
 
       if (!noTitleChange) {
         this.titleText = this.apiData.title_text;
-        elementStack.push(this.$el.querySelector('.title'));
+        elementStack.push(this.$refs['issue-title']);
       }
 
       if (!noDescriptionChange) {
@@ -61,21 +68,22 @@ export default {
         this.descriptionChange = true;
         this.updateTaskHTML();
         this.tasks = this.apiData.task_status;
-        elementStack.push(this.$el.querySelector('.wiki'));
+        elementStack.push(this.$refs['issue-content-container-gfm-entry']);
       }
 
       elementStack.forEach((element) => {
-        element.classList.remove('issue-realtime-trigger-pulse');
-        element.classList.add('issue-realtime-pre-pulse');
+        if (element) {
+          element.classList.remove('issue-realtime-trigger-pulse');
+          element.classList.add('issue-realtime-pre-pulse');
+        }
       });
 
       return elementStack;
     },
     setTabTitle() {
-      const currentTabTitle = document.querySelector('title');
-      const currentTabTitleScope = currentTabTitle.innerText.split('·');
+      const currentTabTitleScope = this.titleEl.innerText.split('·');
       currentTabTitleScope[0] = `${this.titleText} (#${this.apiData.issue_number}) `;
-      currentTabTitle.innerText = currentTabTitleScope.join('·');
+      this.titleEl.innerText = currentTabTitleScope.join('·');
     },
     animate(title, description, elementsToVisualize) {
       this.timeoutId = setTimeout(() => {
@@ -84,11 +92,11 @@ export default {
         this.setTabTitle();
 
         elementsToVisualize.forEach((element) => {
-          element.classList.remove('issue-realtime-pre-pulse');
-          element.classList.add('issue-realtime-trigger-pulse');
+          if (element) {
+            element.classList.remove('issue-realtime-pre-pulse');
+            element.classList.add('issue-realtime-trigger-pulse');
+          }
         });
-
-        clearTimeout(this.timeoutId);
       }, 0);
     },
     triggerAnimation() {
@@ -163,7 +171,12 @@ export default {
 
 <template>
   <div>
-    <h2 class="title issue-realtime-trigger-pulse" v-html="title"></h2>
+    <h2
+      class="title issue-realtime-trigger-pulse"
+      ref="issue-title"
+      v-html="title"
+    >
+    </h2>
     <div
       :class="descriptionClass"
       v-if="description"
@@ -174,7 +187,10 @@ export default {
         ref="issue-content-container-gfm-entry"
       >
       </div>
-      <textarea class="hidden js-task-list-field" v-if="descriptionText">{{descriptionText}}</textarea>
+      <textarea
+        class="hidden js-task-list-field"
+        v-if="descriptionText"
+      >{{descriptionText}}</textarea>
     </div>
   </div>
 </template>
-- 
GitLab


From 05c409d28eaacde527a9260a1828fcd2b720a93c Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Thu, 4 May 2017 06:24:50 -0600
Subject: [PATCH 205/363] use nextTick and cache timeAgoEl

---
 .../issue_show/issue_title_description.vue    | 20 +++++++++----------
 1 file changed, 10 insertions(+), 10 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 8d2ef39040b36..0593d95cf40e3 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -1,4 +1,5 @@
 <script>
+import Vue from 'vue';
 import Visibility from 'visibilityjs';
 import Poll from './../lib/utils/poll';
 import Service from './services/index';
@@ -37,13 +38,13 @@ export default {
     return {
       poll,
       apiData: {},
-      timeoutId: null,
       title: null,
       titleText: '',
+      tasks: '0 of 0',
       description: null,
       descriptionText: '',
       descriptionChange: false,
-      tasks: '0 of 0',
+      timeAgoEl: $('.issue_edited_ago'),
       titleEl: document.querySelector('title'),
     };
   },
@@ -86,18 +87,18 @@ export default {
       this.titleEl.innerText = currentTabTitleScope.join('·');
     },
     animate(title, description, elementsToVisualize) {
-      this.timeoutId = setTimeout(() => {
-        this.title = title;
-        this.description = description;
-        this.setTabTitle();
+      this.title = title;
+      this.description = description;
+      this.setTabTitle();
 
+      Vue.nextTick(() => {
         elementsToVisualize.forEach((element) => {
           if (element) {
             element.classList.remove('issue-realtime-pre-pulse');
             element.classList.add('issue-realtime-trigger-pulse');
           }
         });
-      }, 0);
+      });
     },
     triggerAnimation() {
       // always reset to false before checking the change
@@ -124,10 +125,9 @@ export default {
     },
     updateEditedTimeAgo() {
       const toolTipTime = gl.utils.formatDate(this.apiData.updated_at);
-      const $timeAgoNode = $('.issue_edited_ago');
 
-      $timeAgoNode.attr('datetime', this.apiData.updated_at);
-      $timeAgoNode.attr('data-original-title', toolTipTime);
+      this.timeAgoEl.attr('datetime', this.apiData.updated_at);
+      this.timeAgoEl.attr('data-original-title', toolTipTime);
     },
   },
   computed: {
-- 
GitLab


From 57731bbc6063f781a07511a3acb434a3eadf6afc Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Thu, 4 May 2017 06:37:27 -0600
Subject: [PATCH 206/363] better name for zeroData - use  insteaf importing Vue

---
 app/assets/javascripts/issue_show/actions/tasks.js | 14 +++++++-------
 .../issue_show/issue_title_description.vue         |  3 +--
 2 files changed, 8 insertions(+), 9 deletions(-)

diff --git a/app/assets/javascripts/issue_show/actions/tasks.js b/app/assets/javascripts/issue_show/actions/tasks.js
index 0f6e71ce7ac97..654d6d64f0fb9 100644
--- a/app/assets/javascripts/issue_show/actions/tasks.js
+++ b/app/assets/javascripts/issue_show/actions/tasks.js
@@ -2,25 +2,25 @@ export default (apiData, tasks) => {
   const $tasks = $('#task_status');
   const $tasksShort = $('#task_status_short');
   const $issueableHeader = $('.issuable-header');
-  const zeroData = { api: null, tasks: null };
+  const tasksStates = { api: null, tasks: null };
 
   if ($tasks.length === 0) {
-    if (!(apiData.task_status.indexOf('0 of 0') >= 0)) {
+    if (!(apiData.task_status.indexOf('0 of 0') === 0)) {
       $issueableHeader.append(`<span id="task_status">${apiData.task_status}</span>`);
     } else {
       $issueableHeader.append('<span id="task_status"></span>');
     }
   } else {
-    zeroData.api = apiData.task_status.indexOf('0 of 0') >= 0;
-    zeroData.tasks = tasks.indexOf('0 of 0') >= 0;
+    tasksStates.api = apiData.task_status.indexOf('0 of 0') === 0;
+    tasksStates.tasks = tasks.indexOf('0 of 0') === 0;
   }
 
-  if ($tasks && !zeroData.api) {
+  if ($tasks && !tasksStates.api) {
     $tasks.text(apiData.task_status);
     $tasksShort.text(apiData.task_status);
-  } else if (zeroData.tasks) {
+  } else if (tasksStates.tasks) {
     $issueableHeader.append(`<span id="task_status">${apiData.task_status}</span>`);
-  } else if (zeroData.api) {
+  } else if (tasksStates.api) {
     $tasks.remove();
     $tasksShort.remove();
   }
diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 0593d95cf40e3..7aa019ac41d9a 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -1,5 +1,4 @@
 <script>
-import Vue from 'vue';
 import Visibility from 'visibilityjs';
 import Poll from './../lib/utils/poll';
 import Service from './services/index';
@@ -91,7 +90,7 @@ export default {
       this.description = description;
       this.setTabTitle();
 
-      Vue.nextTick(() => {
+      this.$nextTick(() => {
         elementsToVisualize.forEach((element) => {
           if (element) {
             element.classList.remove('issue-realtime-pre-pulse');
-- 
GitLab


From 3af533f7b62c6b94f230f66e9cbd3bbd775d969f Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Thu, 4 May 2017 06:41:28 -0600
Subject: [PATCH 207/363] rename candescription to canUpdateIssue

---
 app/assets/javascripts/issue_show/index.js                    | 4 ++--
 app/assets/javascripts/issue_show/issue_title_description.vue | 4 ++--
 app/views/projects/issues/show.html.haml                      | 2 +-
 3 files changed, 5 insertions(+), 5 deletions(-)

diff --git a/app/assets/javascripts/issue_show/index.js b/app/assets/javascripts/issue_show/index.js
index db1cdb6d498a6..2de072c9778d5 100644
--- a/app/assets/javascripts/issue_show/index.js
+++ b/app/assets/javascripts/issue_show/index.js
@@ -4,13 +4,13 @@ import '../vue_shared/vue_resource_interceptor';
 
 (() => {
   const issueTitleData = document.querySelector('.issue-title-data').dataset;
-  const { candescription, endpoint } = issueTitleData;
+  const { canupdateissue, endpoint } = issueTitleData;
 
   const vm = new Vue({
     el: '.issue-title-entrypoint',
     render: createElement => createElement(IssueTitle, {
       props: {
-        candescription,
+        canUpdateIssue: canupdateissue,
         endpoint,
       },
     }),
diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 7aa019ac41d9a..424e8b88bd7b5 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -10,7 +10,7 @@ export default {
       required: true,
       type: String,
     },
-    candescription: {
+    canUpdateIssue: {
       required: true,
       type: String,
     },
@@ -131,7 +131,7 @@ export default {
   },
   computed: {
     descriptionClass() {
-      return `description ${this.candescription} is-task-list-enabled`;
+      return `description ${this.canUpdateIssue} is-task-list-enabled`;
     },
   },
   created() {
diff --git a/app/views/projects/issues/show.html.haml b/app/views/projects/issues/show.html.haml
index 3971ea44ef33a..aa024be1c55d3 100644
--- a/app/views/projects/issues/show.html.haml
+++ b/app/views/projects/issues/show.html.haml
@@ -52,7 +52,7 @@
 .issue-details.issuable-details
   .detail-page-description.content-block
     .issue-title-data.hidden{ "data" => { "endpoint" => rendered_title_namespace_project_issue_path(@project.namespace, @project, @issue),
-      "canDescription" => can?(current_user, :update_issue, @issue) ? 'js-task-list-container' : '',
+      "canUpdateIssue" => can?(current_user, :update_issue, @issue) ? 'js-task-list-container' : '',
     } }
     .issue-title-entrypoint
 
-- 
GitLab


From 082b868ecda0b30193a93b32d9614aef1f204484 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Thu, 4 May 2017 07:24:47 -0600
Subject: [PATCH 208/363] fix js specs

---
 .../javascripts/issue_show/actions/tasks.js   | 24 +++++-----
 .../issue_title_description_spec.js           | 22 ++++-----
 spec/javascripts/issue_spec.js                | 48 -------------------
 3 files changed, 23 insertions(+), 71 deletions(-)

diff --git a/app/assets/javascripts/issue_show/actions/tasks.js b/app/assets/javascripts/issue_show/actions/tasks.js
index 654d6d64f0fb9..719b604a86581 100644
--- a/app/assets/javascripts/issue_show/actions/tasks.js
+++ b/app/assets/javascripts/issue_show/actions/tasks.js
@@ -1,26 +1,26 @@
-export default (apiData, tasks) => {
+export default (newStateData, tasks) => {
   const $tasks = $('#task_status');
   const $tasksShort = $('#task_status_short');
   const $issueableHeader = $('.issuable-header');
-  const tasksStates = { api: null, tasks: null };
+  const tasksStates = { newState: null, currentState: null };
 
   if ($tasks.length === 0) {
-    if (!(apiData.task_status.indexOf('0 of 0') === 0)) {
-      $issueableHeader.append(`<span id="task_status">${apiData.task_status}</span>`);
+    if (!(newStateData.task_status.indexOf('0 of 0') === 0)) {
+      $issueableHeader.append(`<span id="task_status">${newStateData.task_status}</span>`);
     } else {
       $issueableHeader.append('<span id="task_status"></span>');
     }
   } else {
-    tasksStates.api = apiData.task_status.indexOf('0 of 0') === 0;
-    tasksStates.tasks = tasks.indexOf('0 of 0') === 0;
+    tasksStates.newState = newStateData.task_status.indexOf('0 of 0') === 0;
+    tasksStates.currentState = tasks.indexOf('0 of 0') === 0;
   }
 
-  if ($tasks && !tasksStates.api) {
-    $tasks.text(apiData.task_status);
-    $tasksShort.text(apiData.task_status);
-  } else if (tasksStates.tasks) {
-    $issueableHeader.append(`<span id="task_status">${apiData.task_status}</span>`);
-  } else if (tasksStates.api) {
+  if ($tasks && !tasksStates.newState) {
+    $tasks.text(newStateData.task_status);
+    $tasksShort.text(newStateData.task_status);
+  } else if (tasksStates.currentState) {
+    $issueableHeader.append(`<span id="task_status">${newStateData.task_status}</span>`);
+  } else if (tasksStates.newState) {
     $tasks.remove();
     $tasksShort.remove();
   }
diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index 3877ed2164b90..30455663e5097 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -2,7 +2,7 @@ import Vue from 'vue';
 import $ from 'jquery';
 import '~/render_math';
 import '~/render_gfm';
-import issueTitle from '~/issue_show/issue_title_description.vue';
+import issueTitleDescription from '~/issue_show/issue_title_description.vue';
 import issueShowData from './mock_data';
 
 window.$ = $;
@@ -11,18 +11,18 @@ const issueShowInterceptor = data => (request, next) => {
   next(request.respondWith(JSON.stringify(data), {
     status: 200,
     headers: {
-      'POLL-INTERVAL': 10,
+      'POLL-INTERVAL': 1,
     },
   }));
 };
 
 describe('Issue Title', () => {
-  const comps = {
-    IssueTitleComponent: {},
-  };
+  document.body.innerHTML = '<span id="task_status"></span>';
+
+  let IssueTitleDescriptionComponent;
 
   beforeEach(() => {
-    comps.IssueTitleComponent = Vue.extend(issueTitle);
+    IssueTitleDescriptionComponent = Vue.extend(issueTitleDescription);
   });
 
   afterEach(() => {
@@ -32,14 +32,14 @@ describe('Issue Title', () => {
   it('should render a title/description and update title/description on update', (done) => {
     Vue.http.interceptors.push(issueShowInterceptor(issueShowData.initialRequest));
 
-    const issueShowComponent = new comps.IssueTitleComponent({
+    const issueShowComponent = new IssueTitleDescriptionComponent({
       propsData: {
-        candescription: '.css-stuff',
+        canUpdateIssue: '.css-stuff',
         endpoint: '/gitlab-org/gitlab-shell/issues/9/rendered_title',
       },
     }).$mount();
 
-    // need setTimeout because actual setTimeout in code :P
+    // need setTimeout because of api call/v-html
     setTimeout(() => {
       expect(document.querySelector('title').innerText).toContain('this is a title (#1)');
       expect(issueShowComponent.$el.querySelector('.title').innerHTML).toContain('<p>this is a title</p>');
@@ -55,8 +55,8 @@ describe('Issue Title', () => {
         expect(issueShowComponent.$el.querySelector('.js-task-list-field').innerText).toContain('42');
 
         done();
-      }, 20);
-    }, 10);
+      });
+    });
     // 10ms is just long enough for the update hook to fire
   });
 });
diff --git a/spec/javascripts/issue_spec.js b/spec/javascripts/issue_spec.js
index 788d7ccc5e42c..445ecea8e643c 100644
--- a/spec/javascripts/issue_spec.js
+++ b/spec/javascripts/issue_spec.js
@@ -1,10 +1,5 @@
 /* eslint-disable space-before-function-paren, one-var, one-var-declaration-per-line, no-use-before-define, comma-dangle, max-len */
 import Issue from '~/issue';
-import Vue from 'vue';
-import '~/render_math';
-import '~/render_gfm';
-import IssueTitle from '~/issue_show/issue_title_description.vue';
-import issueShowData from './issue_show/mock_data';
 
 require('~/lib/utils/text_utility');
 
@@ -81,52 +76,9 @@ describe('Issue', function() {
   }
 
   describe('task lists', function() {
-    const issueShowInterceptor = data => (request, next) => {
-      next(request.respondWith(JSON.stringify(data), {
-        status: 200,
-      }));
-    };
-
     beforeEach(function() {
       loadFixtures('issues/issue-with-task-list.html.raw');
       this.issue = new Issue();
-      Vue.http.interceptors.push(issueShowInterceptor(issueShowData.issueSpecRequest));
-    });
-
-    afterEach(function() {
-      Vue.http.interceptors = _.without(Vue.http.interceptors, issueShowInterceptor);
-    });
-
-    it('modifies the Markdown field', function(done) {
-      // gotta actually render it for jquery to find elements
-      const vm = new Vue({
-        el: document.querySelector('.issue-title-entrypoint'),
-        components: {
-          IssueTitle,
-        },
-        render: createElement => createElement(IssueTitle, {
-          props: {
-            candescription: '.js-task-list-container',
-            endpoint: '/gitlab-org/gitlab-shell/issues/9/rendered_title',
-          },
-        }),
-      });
-
-      setTimeout(() => {
-        spyOn(jQuery, 'ajax').and.stub();
-
-        const description = '<li class="task-list-item enabled"><input type="checkbox" class="task-list-item-checkbox"> Task List Item</li>';
-
-        expect(document.querySelector('title').innerText).toContain('this is a title (#1)');
-        expect(vm.$el.querySelector('.title').innerHTML).toContain('<p>this is a title</p>');
-        expect(vm.$el.querySelector('.wiki').innerHTML).toContain(description);
-        expect(vm.$el.querySelector('.js-task-list-field').value).toContain('- [ ] Task List Item');
-
-        // somehow the dom does not have a closest `.js-task-list.field` to the `.task-list-item-checkbox`
-        $('input[type=checkbox]').attr('checked', true).trigger('change');
-        expect($('.js-task-list-field').val()).toBe('- [x] Task List Item');
-        done();
-      }, 10);
     });
 
     it('submits an ajax request on tasklist:changed', function() {
-- 
GitLab


From c59f80ff2d50fccb31625eb617be1374a7ad1264 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Thu, 4 May 2017 07:32:23 -0600
Subject: [PATCH 209/363] change error handling for api callback - check length
 is not zero on jquery el

---
 app/assets/javascripts/issue_show/actions/tasks.js         | 2 +-
 .../javascripts/issue_show/issue_title_description.vue     | 7 +------
 2 files changed, 2 insertions(+), 7 deletions(-)

diff --git a/app/assets/javascripts/issue_show/actions/tasks.js b/app/assets/javascripts/issue_show/actions/tasks.js
index 719b604a86581..0740a9f559c26 100644
--- a/app/assets/javascripts/issue_show/actions/tasks.js
+++ b/app/assets/javascripts/issue_show/actions/tasks.js
@@ -15,7 +15,7 @@ export default (newStateData, tasks) => {
     tasksStates.currentState = tasks.indexOf('0 of 0') === 0;
   }
 
-  if ($tasks && !tasksStates.newState) {
+  if ($tasks.length !== 0 && !tasksStates.newState) {
     $tasks.text(newStateData.task_status);
     $tasksShort.text(newStateData.task_status);
   } else if (tasksStates.currentState) {
diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 424e8b88bd7b5..d74d59531ddd7 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -25,12 +25,7 @@ export default {
         this.renderResponse(res);
       },
       errorCallback: (err) => {
-        if (process.env.NODE_ENV !== 'production') {
-          // eslint-disable-next-line no-console
-          console.error('ISSUE SHOW REALTIME ERROR', err, err.stack);
-        } else {
-          throw new Error(err);
-        }
+        throw new Error(err);
       },
     });
 
-- 
GitLab


From 0f58eb6bde35009b69ef871534d9ff80fc38bbf7 Mon Sep 17 00:00:00 2001
From: Douwe Maan <douwe@selenight.nl>
Date: Tue, 2 May 2017 17:42:37 -0500
Subject: [PATCH 210/363] Add artifact file page that uses the blob viewer

---
 app/assets/javascripts/dispatcher.js          |   3 +
 .../projects/artifacts_controller.rb          |  28 ++-
 app/helpers/blob_helper.rb                    |   6 +-
 app/helpers/gitlab_routing_helper.rb          |   2 +
 app/models/ci/artifact_blob.rb                |  35 ++++
 .../projects/artifacts/_tree_file.html.haml   |   9 +-
 app/views/projects/artifacts/file.html.haml   |  33 +++
 .../unreleased/dm-artifact-blob-viewer.yml    |   4 +
 config/routes/project.rb                      |   1 +
 features/project/builds/artifacts.feature     |   3 +-
 features/steps/project/builds/artifacts.rb    |  15 +-
 .../ci/build/artifacts/metadata/entry.rb      |   6 +
 .../projects/artifacts_controller_spec.rb     | 188 ++++++++++++++++++
 spec/features/projects/artifacts/file_spec.rb |  59 ++++++
 .../ci/build/artifacts/metadata/entry_spec.rb |  11 +
 spec/models/ci/artifact_blob_spec.rb          |  44 ++++
 .../projects/artifacts_controller_spec.rb     | 117 -----------
 17 files changed, 425 insertions(+), 139 deletions(-)
 create mode 100644 app/models/ci/artifact_blob.rb
 create mode 100644 app/views/projects/artifacts/file.html.haml
 create mode 100644 changelogs/unreleased/dm-artifact-blob-viewer.yml
 create mode 100644 spec/controllers/projects/artifacts_controller_spec.rb
 create mode 100644 spec/features/projects/artifacts/file_spec.rb
 create mode 100644 spec/models/ci/artifact_blob_spec.rb
 delete mode 100644 spec/requests/projects/artifacts_controller_spec.rb

diff --git a/app/assets/javascripts/dispatcher.js b/app/assets/javascripts/dispatcher.js
index 0bdce52cc89c1..7e46915310682 100644
--- a/app/assets/javascripts/dispatcher.js
+++ b/app/assets/javascripts/dispatcher.js
@@ -344,6 +344,9 @@ const ShortcutsBlob = require('./shortcuts_blob');
         case 'projects:artifacts:browse':
           new BuildArtifacts();
           break;
+        case 'projects:artifacts:file':
+          new BlobViewer();
+          break;
         case 'help:index':
           gl.VersionCheckImage.bindErrorEvent($('img.js-version-status-badge'));
           break;
diff --git a/app/controllers/projects/artifacts_controller.rb b/app/controllers/projects/artifacts_controller.rb
index a13588b4218c2..1224e9503c93d 100644
--- a/app/controllers/projects/artifacts_controller.rb
+++ b/app/controllers/projects/artifacts_controller.rb
@@ -1,11 +1,13 @@
 class Projects::ArtifactsController < Projects::ApplicationController
   include ExtractsPath
+  include RendersBlob
 
   layout 'project'
   before_action :authorize_read_build!
   before_action :authorize_update_build!, only: [:keep]
   before_action :extract_ref_name_and_path
   before_action :validate_artifacts!
+  before_action :set_path_and_entry, only: [:file, :raw]
 
   def download
     if artifacts_file.file_storage?
@@ -24,15 +26,24 @@ def browse
   end
 
   def file
-    entry = build.artifacts_metadata_entry(params[:path])
+    blob = @entry.blob
+    override_max_blob_size(blob)
 
-    if entry.exists?
-      send_artifacts_entry(build, entry)
-    else
-      render_404
+    respond_to do |format|
+      format.html do
+        render 'file'
+      end
+
+      format.json do
+        render_blob_json(blob)
+      end
     end
   end
 
+  def raw
+    send_artifacts_entry(build, @entry)
+  end
+
   def keep
     build.keep_artifacts!
     redirect_to namespace_project_build_path(project.namespace, project, build)
@@ -81,4 +92,11 @@ def build_from_ref
   def artifacts_file
     @artifacts_file ||= build.artifacts_file
   end
+
+  def set_path_and_entry
+    @path = params[:path]
+    @entry = build.artifacts_metadata_entry(@path)
+
+    render_404 unless @entry.exists?
+  end
 end
diff --git a/app/helpers/blob_helper.rb b/app/helpers/blob_helper.rb
index 37b6f4ad5cc6b..af430270ae424 100644
--- a/app/helpers/blob_helper.rb
+++ b/app/helpers/blob_helper.rb
@@ -119,7 +119,9 @@ def blob_icon(mode, name)
   end
 
   def blob_raw_url
-    if @snippet
+    if @build && @entry
+      raw_namespace_project_build_artifacts_path(@project.namespace, @project, @build, path: @entry.path)
+    elsif @snippet
       if @snippet.project_id
         raw_namespace_project_snippet_path(@project.namespace, @project, @snippet)
       else
@@ -250,6 +252,8 @@ def blob_render_error_reason(viewer)
       case viewer.blob.external_storage
       when :lfs
         'it is stored in LFS'
+      when :build_artifact
+        'it is stored as a job artifact'
       else
         'it is stored externally'
       end
diff --git a/app/helpers/gitlab_routing_helper.rb b/app/helpers/gitlab_routing_helper.rb
index e9b7cbbad6a5e..1336c676134eb 100644
--- a/app/helpers/gitlab_routing_helper.rb
+++ b/app/helpers/gitlab_routing_helper.rb
@@ -208,6 +208,8 @@ def artifacts_action_path(path, project, build)
       browse_namespace_project_build_artifacts_path(*args)
     when 'file'
       file_namespace_project_build_artifacts_path(*args)
+    when 'raw'
+      raw_namespace_project_build_artifacts_path(*args)
     end
   end
 
diff --git a/app/models/ci/artifact_blob.rb b/app/models/ci/artifact_blob.rb
new file mode 100644
index 0000000000000..b35febc9ac5a5
--- /dev/null
+++ b/app/models/ci/artifact_blob.rb
@@ -0,0 +1,35 @@
+module Ci
+  class ArtifactBlob
+    include BlobLike
+
+    attr_reader :entry
+
+    def initialize(entry)
+      @entry = entry
+    end
+
+    delegate :name, :path, to: :entry
+
+    def id
+      Digest::SHA1.hexdigest(path)
+    end
+
+    def size
+      entry.metadata[:size]
+    end
+
+    def data
+      "Build artifact #{path}"
+    end
+
+    def mode
+      entry.metadata[:mode]
+    end
+
+    def external_storage
+      :build_artifact
+    end
+
+    alias_method :external_size, :size
+  end
+end
diff --git a/app/views/projects/artifacts/_tree_file.html.haml b/app/views/projects/artifacts/_tree_file.html.haml
index 36fb4c998c96b..ce7e25d774b62 100644
--- a/app/views/projects/artifacts/_tree_file.html.haml
+++ b/app/views/projects/artifacts/_tree_file.html.haml
@@ -1,9 +1,10 @@
 - path_to_file = file_namespace_project_build_artifacts_path(@project.namespace, @project, @build, path: file.path)
 
 %tr.tree-item{ 'data-link' => path_to_file }
+  - blob = file.blob
   %td.tree-item-file-name
-    = tree_icon('file', '664', file.name)
-    %span.str-truncated
-      = link_to file.name, path_to_file
+    = tree_icon('file', blob.mode, blob.name)
+    = link_to path_to_file do
+      %span.str-truncated= blob.name
   %td
-    = number_to_human_size(file.metadata[:size], precision: 2)
+    = number_to_human_size(blob.size, precision: 2)
diff --git a/app/views/projects/artifacts/file.html.haml b/app/views/projects/artifacts/file.html.haml
new file mode 100644
index 0000000000000..d8da83b9a80ba
--- /dev/null
+++ b/app/views/projects/artifacts/file.html.haml
@@ -0,0 +1,33 @@
+- page_title @path, 'Artifacts', "#{@build.name} (##{@build.id})", 'Jobs'
+= render "projects/pipelines/head"
+
+= render "projects/builds/header", show_controls: false
+
+#tree-holder.tree-holder
+  .nav-block
+    %ul.breadcrumb.repo-breadcrumb
+      %li
+        = link_to 'Artifacts', browse_namespace_project_build_artifacts_path(@project.namespace, @project, @build)
+      - path_breadcrumbs do |title, path|
+        - title = truncate(title, length: 40)
+        %li
+          - if path == @path
+            = link_to file_namespace_project_build_artifacts_path(@project.namespace, @project, @build, path) do
+              %strong= title
+          - else
+            = link_to title, browse_namespace_project_build_artifacts_path(@project.namespace, @project, @build, path)
+
+
+  %article.file-holder
+    - blob = @entry.blob
+    .js-file-title.file-title-flex-parent
+      = render 'projects/blob/header_content', blob: blob
+
+      .file-actions.hidden-xs
+        = render 'projects/blob/viewer_switcher', blob: blob
+
+        .btn-group{ role: "group" }<
+          = copy_blob_source_button(blob)
+          = open_raw_blob_button(blob)
+
+    = render 'projects/blob/content', blob: blob
diff --git a/changelogs/unreleased/dm-artifact-blob-viewer.yml b/changelogs/unreleased/dm-artifact-blob-viewer.yml
new file mode 100644
index 0000000000000..38f5cbb73e18e
--- /dev/null
+++ b/changelogs/unreleased/dm-artifact-blob-viewer.yml
@@ -0,0 +1,4 @@
+---
+title: Add artifact file page that uses the blob viewer
+merge_request:
+author:
diff --git a/config/routes/project.rb b/config/routes/project.rb
index a15e365cc2fb8..5aba469b6e4d8 100644
--- a/config/routes/project.rb
+++ b/config/routes/project.rb
@@ -183,6 +183,7 @@
           get :download
           get :browse, path: 'browse(/*path)', format: false
           get :file, path: 'file/*path', format: false
+          get :raw, path: 'raw/*path', format: false
           post :keep
         end
       end
diff --git a/features/project/builds/artifacts.feature b/features/project/builds/artifacts.feature
index 09094d638c966..5abc24949cf5c 100644
--- a/features/project/builds/artifacts.feature
+++ b/features/project/builds/artifacts.feature
@@ -46,13 +46,14 @@ Feature: Project Builds Artifacts
     And I navigate to parent directory of directory with invalid name
     Then I should not see directory with invalid name on the list
 
+  @javascript
   Scenario: I download a single file from build artifacts
     Given recent build has artifacts available
     And recent build has artifacts metadata available
     When I visit recent build details page
     And I click artifacts browse button
     And I click a link to file within build artifacts
-    Then download of a file extracted from build artifacts should start
+    Then I see a download link
 
   @javascript
   Scenario: I click on a row in an artifacts table
diff --git a/features/steps/project/builds/artifacts.rb b/features/steps/project/builds/artifacts.rb
index 3ec5c8e2f4ffc..eec375b0532fc 100644
--- a/features/steps/project/builds/artifacts.rb
+++ b/features/steps/project/builds/artifacts.rb
@@ -3,6 +3,7 @@ class Spinach::Features::ProjectBuildsArtifacts < Spinach::FeatureSteps
   include SharedProject
   include SharedBuilds
   include RepoHelpers
+  include WaitForAjax
 
   step 'I click artifacts download button' do
     click_link 'Download'
@@ -78,19 +79,11 @@ class Spinach::Features::ProjectBuildsArtifacts < Spinach::FeatureSteps
 
   step 'I click a link to file within build artifacts' do
     page.within('.tree-table') { find_link('ci_artifacts.txt').click }
+    wait_for_ajax
   end
 
-  step 'download of a file extracted from build artifacts should start' do
-    send_data = response_headers[Gitlab::Workhorse::SEND_DATA_HEADER]
-
-    expect(send_data).to start_with('artifacts-entry:')
-
-    base64_params = send_data.sub(/\Aartifacts\-entry:/, '')
-    params = JSON.parse(Base64.urlsafe_decode64(base64_params))
-
-    expect(params.keys).to eq(%w(Archive Entry))
-    expect(params['Archive']).to end_with('build_artifacts.zip')
-    expect(params['Entry']).to eq(Base64.encode64('ci_artifacts.txt'))
+  step 'I see a download link' do
+    expect(page).to have_link 'download it'
   end
 
   step 'I click a first row within build artifacts table' do
diff --git a/lib/gitlab/ci/build/artifacts/metadata/entry.rb b/lib/gitlab/ci/build/artifacts/metadata/entry.rb
index 6f799c2f031b0..2e073334abca9 100644
--- a/lib/gitlab/ci/build/artifacts/metadata/entry.rb
+++ b/lib/gitlab/ci/build/artifacts/metadata/entry.rb
@@ -37,6 +37,12 @@ def file?
           !directory?
         end
 
+        def blob
+          return unless file?
+
+          @blob ||= Blob.decorate(::Ci::ArtifactBlob.new(self), nil)
+        end
+
         def has_parent?
           nodes > 0
         end
diff --git a/spec/controllers/projects/artifacts_controller_spec.rb b/spec/controllers/projects/artifacts_controller_spec.rb
new file mode 100644
index 0000000000000..eff9fab8da2a5
--- /dev/null
+++ b/spec/controllers/projects/artifacts_controller_spec.rb
@@ -0,0 +1,188 @@
+require 'spec_helper'
+
+describe Projects::ArtifactsController do
+  let(:user) { create(:user) }
+  let(:project) { create(:project, :repository) }
+
+  let(:pipeline) do
+    create(:ci_pipeline,
+            project: project,
+            sha: project.commit.sha,
+            ref: project.default_branch,
+            status: 'success')
+  end
+
+  let(:build) { create(:ci_build, :success, :artifacts, pipeline: pipeline) }
+
+  before do
+    project.team << [user, :developer]
+
+    sign_in(user)
+  end
+
+  describe 'GET download' do
+    it 'sends the artifacts file' do
+      expect(controller).to receive(:send_file).with(build.artifacts_file.path, disposition: 'attachment').and_call_original
+
+      get :download, namespace_id: project.namespace, project_id: project, build_id: build
+    end
+  end
+
+  describe 'GET browse' do
+    context 'when the directory exists' do
+      it 'renders the browse view' do
+        get :browse, namespace_id: project.namespace, project_id: project, build_id: build, path: 'other_artifacts_0.1.2'
+
+        expect(response).to render_template('projects/artifacts/browse')
+      end
+    end
+
+    context 'when the directory does not exist' do
+      it 'responds Not Found' do
+        get :browse, namespace_id: project.namespace, project_id: project, build_id: build, path: 'unknown'
+
+        expect(response).to be_not_found
+      end
+    end
+  end
+
+  describe 'GET file' do
+    context 'when the file exists' do
+      it 'renders the file view' do
+        get :file, namespace_id: project.namespace, project_id: project, build_id: build, path: 'ci_artifacts.txt'
+
+        expect(response).to render_template('projects/artifacts/file')
+      end
+    end
+
+    context 'when the file does not exist' do
+      it 'responds Not Found' do
+        get :file, namespace_id: project.namespace, project_id: project, build_id: build, path: 'unknown'
+
+        expect(response).to be_not_found
+      end
+    end
+  end
+
+  describe 'GET raw' do
+    context 'when the file exists' do
+      it 'serves the file using workhorse' do
+        get :raw, namespace_id: project.namespace, project_id: project, build_id: build, path: 'ci_artifacts.txt'
+
+        send_data = response.headers[Gitlab::Workhorse::SEND_DATA_HEADER]
+
+        expect(send_data).to start_with('artifacts-entry:')
+
+        base64_params = send_data.sub(/\Aartifacts\-entry:/, '')
+        params = JSON.parse(Base64.urlsafe_decode64(base64_params))
+
+        expect(params.keys).to eq(%w(Archive Entry))
+        expect(params['Archive']).to end_with('build_artifacts.zip')
+        expect(params['Entry']).to eq(Base64.encode64('ci_artifacts.txt'))
+      end
+    end
+
+    context 'when the file does not exist' do
+      it 'responds Not Found' do
+        get :raw, namespace_id: project.namespace, project_id: project, build_id: build, path: 'unknown'
+
+        expect(response).to be_not_found
+      end
+    end
+  end
+
+  describe 'GET latest_succeeded' do
+    def params_from_ref(ref = pipeline.ref, job = build.name, path = 'browse')
+      {
+        namespace_id: project.namespace,
+        project_id: project,
+        ref_name_and_path: File.join(ref, path),
+        job: job
+      }
+    end
+
+    context 'cannot find the build' do
+      shared_examples 'not found' do
+        it { expect(response).to have_http_status(:not_found) }
+      end
+
+      context 'has no such ref' do
+        before do
+          get :latest_succeeded, params_from_ref('TAIL', build.name)
+        end
+
+        it_behaves_like 'not found'
+      end
+
+      context 'has no such build' do
+        before do
+          get :latest_succeeded, params_from_ref(pipeline.ref, 'NOBUILD')
+        end
+
+        it_behaves_like 'not found'
+      end
+
+      context 'has no path' do
+        before do
+          get :latest_succeeded, params_from_ref(pipeline.sha, build.name, '')
+        end
+
+        it_behaves_like 'not found'
+      end
+    end
+
+    context 'found the build and redirect' do
+      shared_examples 'redirect to the build' do
+        it 'redirects' do
+          path = browse_namespace_project_build_artifacts_path(
+            project.namespace,
+            project,
+            build)
+
+          expect(response).to redirect_to(path)
+        end
+      end
+
+      context 'with regular branch' do
+        before do
+          pipeline.update(ref: 'master',
+                          sha: project.commit('master').sha)
+
+          get :latest_succeeded, params_from_ref('master')
+        end
+
+        it_behaves_like 'redirect to the build'
+      end
+
+      context 'with branch name containing slash' do
+        before do
+          pipeline.update(ref: 'improve/awesome',
+                          sha: project.commit('improve/awesome').sha)
+
+          get :latest_succeeded, params_from_ref('improve/awesome')
+        end
+
+        it_behaves_like 'redirect to the build'
+      end
+
+      context 'with branch name and path containing slashes' do
+        before do
+          pipeline.update(ref: 'improve/awesome',
+                          sha: project.commit('improve/awesome').sha)
+
+          get :latest_succeeded, params_from_ref('improve/awesome', build.name, 'file/README.md')
+        end
+
+        it 'redirects' do
+          path = file_namespace_project_build_artifacts_path(
+            project.namespace,
+            project,
+            build,
+            'README.md')
+
+          expect(response).to redirect_to(path)
+        end
+      end
+    end
+  end
+end
diff --git a/spec/features/projects/artifacts/file_spec.rb b/spec/features/projects/artifacts/file_spec.rb
new file mode 100644
index 0000000000000..74308a7e8dd39
--- /dev/null
+++ b/spec/features/projects/artifacts/file_spec.rb
@@ -0,0 +1,59 @@
+require 'spec_helper'
+
+feature 'Artifact file', :js, feature: true do
+  let(:project) { create(:project, :public) }
+  let(:pipeline) { create(:ci_empty_pipeline, project: project, sha: project.commit.sha, ref: 'master') }
+  let(:build) { create(:ci_build, :artifacts, pipeline: pipeline) }
+
+  def visit_file(path)
+    visit file_namespace_project_build_artifacts_path(project.namespace, project, build, path)
+  end
+
+  context 'Text file' do
+    before do
+      visit_file('other_artifacts_0.1.2/doc_sample.txt')
+
+      wait_for_ajax
+    end
+
+    it 'displays an error' do
+      aggregate_failures do
+        # shows an error message
+        expect(page).to have_content('The source could not be displayed because it is stored as a job artifact. You can download it instead.')
+
+        # does not show a viewer switcher
+        expect(page).not_to have_selector('.js-blob-viewer-switcher')
+
+        # does not show a copy button
+        expect(page).not_to have_selector('.js-copy-blob-source-btn')
+
+        # shows a download button
+        expect(page).to have_link('Download')
+      end
+    end
+  end
+
+  context 'JPG file' do
+    before do
+      visit_file('rails_sample.jpg')
+
+      wait_for_ajax
+    end
+
+    it 'displays the blob' do
+      aggregate_failures do
+        # shows rendered image
+        expect(page).to have_selector('.image_file img')
+
+        # does not show a viewer switcher
+        expect(page).not_to have_selector('.js-blob-viewer-switcher')
+
+        # does not show a copy button
+        expect(page).not_to have_selector('.js-copy-blob-source-btn')
+
+        # shows a download button
+        expect(page).to have_link('Download')
+      end
+    end
+  end
+end
diff --git a/spec/lib/gitlab/ci/build/artifacts/metadata/entry_spec.rb b/spec/lib/gitlab/ci/build/artifacts/metadata/entry_spec.rb
index abc93e1b44a92..3b9056114674f 100644
--- a/spec/lib/gitlab/ci/build/artifacts/metadata/entry_spec.rb
+++ b/spec/lib/gitlab/ci/build/artifacts/metadata/entry_spec.rb
@@ -135,6 +135,17 @@ def entry(path)
       subject { |example| path(example).nodes }
       it { is_expected.to eq 4 }
     end
+
+    describe '#blob' do
+      let(:file_entry) { |example| path(example) }
+      subject { file_entry.blob }
+
+      it 'returns a blob representing the entry data' do
+        expect(subject).to be_a(Blob)
+        expect(subject.path).to eq(file_entry.path)
+        expect(subject.size).to eq(file_entry.metadata[:size])
+      end
+    end
   end
 
   describe 'non-existent/', path: 'non-existent/' do
diff --git a/spec/models/ci/artifact_blob_spec.rb b/spec/models/ci/artifact_blob_spec.rb
new file mode 100644
index 0000000000000..968593d7e9bef
--- /dev/null
+++ b/spec/models/ci/artifact_blob_spec.rb
@@ -0,0 +1,44 @@
+require 'spec_helper'
+
+describe Ci::ArtifactBlob, models: true do
+  let(:build) { create(:ci_build, :artifacts) }
+  let(:entry) { build.artifacts_metadata_entry('other_artifacts_0.1.2/another-subdirectory/banana_sample.gif') }
+
+  subject { described_class.new(entry) }
+
+  describe '#id' do
+    it 'returns a hash of the path' do
+      expect(subject.id).to eq(Digest::SHA1.hexdigest(entry.path))
+    end
+  end
+
+  describe '#name' do
+    it 'returns the entry name' do
+      expect(subject.name).to eq(entry.name)
+    end
+  end
+
+  describe '#path' do
+    it 'returns the entry path' do
+      expect(subject.path).to eq(entry.path)
+    end
+  end
+
+  describe '#size' do
+    it 'returns the entry size' do
+      expect(subject.size).to eq(entry.metadata[:size])
+    end
+  end
+
+  describe '#mode' do
+    it 'returns the entry mode' do
+      expect(subject.mode).to eq(entry.metadata[:mode])
+    end
+  end
+
+  describe '#external_storage' do
+    it 'returns :build_artifact' do
+      expect(subject.external_storage).to eq(:build_artifact)
+    end
+  end
+end
diff --git a/spec/requests/projects/artifacts_controller_spec.rb b/spec/requests/projects/artifacts_controller_spec.rb
deleted file mode 100644
index d20866c0d44d4..0000000000000
--- a/spec/requests/projects/artifacts_controller_spec.rb
+++ /dev/null
@@ -1,117 +0,0 @@
-require 'spec_helper'
-
-describe Projects::ArtifactsController do
-  let(:user) { create(:user) }
-  let(:project) { create(:project, :repository) }
-
-  let(:pipeline) do
-    create(:ci_pipeline,
-            project: project,
-            sha: project.commit.sha,
-            ref: project.default_branch,
-            status: 'success')
-  end
-
-  let(:build) { create(:ci_build, :success, :artifacts, pipeline: pipeline) }
-
-  describe 'GET /:project/builds/artifacts/:ref_name/browse?job=name' do
-    before do
-      project.team << [user, :developer]
-
-      login_as(user)
-    end
-
-    def path_from_ref(
-      ref = pipeline.ref, job = build.name, path = 'browse')
-      latest_succeeded_namespace_project_artifacts_path(
-        project.namespace,
-        project,
-        [ref, path].join('/'),
-        job: job)
-    end
-
-    context 'cannot find the build' do
-      shared_examples 'not found' do
-        it { expect(response).to have_http_status(:not_found) }
-      end
-
-      context 'has no such ref' do
-        before do
-          get path_from_ref('TAIL', build.name)
-        end
-
-        it_behaves_like 'not found'
-      end
-
-      context 'has no such build' do
-        before do
-          get path_from_ref(pipeline.ref, 'NOBUILD')
-        end
-
-        it_behaves_like 'not found'
-      end
-
-      context 'has no path' do
-        before do
-          get path_from_ref(pipeline.sha, build.name, '')
-        end
-
-        it_behaves_like 'not found'
-      end
-    end
-
-    context 'found the build and redirect' do
-      shared_examples 'redirect to the build' do
-        it 'redirects' do
-          path = browse_namespace_project_build_artifacts_path(
-            project.namespace,
-            project,
-            build)
-
-          expect(response).to redirect_to(path)
-        end
-      end
-
-      context 'with regular branch' do
-        before do
-          pipeline.update(ref: 'master',
-                          sha: project.commit('master').sha)
-
-          get path_from_ref('master')
-        end
-
-        it_behaves_like 'redirect to the build'
-      end
-
-      context 'with branch name containing slash' do
-        before do
-          pipeline.update(ref: 'improve/awesome',
-                          sha: project.commit('improve/awesome').sha)
-
-          get path_from_ref('improve/awesome')
-        end
-
-        it_behaves_like 'redirect to the build'
-      end
-
-      context 'with branch name and path containing slashes' do
-        before do
-          pipeline.update(ref: 'improve/awesome',
-                          sha: project.commit('improve/awesome').sha)
-
-          get path_from_ref('improve/awesome', build.name, 'file/README.md')
-        end
-
-        it 'redirects' do
-          path = file_namespace_project_build_artifacts_path(
-            project.namespace,
-            project,
-            build,
-            'README.md')
-
-          expect(response).to redirect_to(path)
-        end
-      end
-    end
-  end
-end
-- 
GitLab


From f68f0cd1eacbd9137b014c486c6f6cd6b6378e58 Mon Sep 17 00:00:00 2001
From: Filipa Lacerda <filipa@gitlab.com>
Date: Thu, 4 May 2017 14:53:00 +0100
Subject: [PATCH 211/363] Adds numbered lists to easily point to documentation

---
 doc/development/fe_guide/style_guide_js.md | 544 ++++++++++-----------
 doc/development/fe_guide/vue.md            |  15 +
 2 files changed, 277 insertions(+), 282 deletions(-)

diff --git a/doc/development/fe_guide/style_guide_js.md b/doc/development/fe_guide/style_guide_js.md
index 1d2b055894830..c8a77f6760708 100644
--- a/doc/development/fe_guide/style_guide_js.md
+++ b/doc/development/fe_guide/style_guide_js.md
@@ -11,207 +11,197 @@ See [our current .eslintrc][eslintrc] for specific rules and patterns.
 
 #### ESlint
 
-- **Never** disable eslint rules unless you have a good reason.  You may see a lot of legacy files with `/* eslint-disable some-rule, some-other-rule */` at the top, but legacy files are a special case.  Any time you develop a new feature or refactor an existing one, you should abide by the eslint rules.
-
-- **Never Ever EVER** disable eslint globally for a file
+1. **Never** disable eslint rules unless you have a good reason.  You may see a lot of legacy files with `/* eslint-disable some-rule, some-other-rule */` at the top, but legacy files are a special case.  Any time you develop a new feature or refactor an existing one, you should abide by the eslint rules.
 
+1. **Never Ever EVER** disable eslint globally for a file
   ```javascript
-  // bad
-  /* eslint-disable */
+    // bad
+    /* eslint-disable */
 
-  // better
-  /* eslint-disable some-rule, some-other-rule */
+    // better
+    /* eslint-disable some-rule, some-other-rule */
 
-  // best
-  // nothing :)
+    // best
+    // nothing :)
   ```
 
-- If you do need to disable a rule for a single violation, try to do it as locally as possible
-
+1. If you do need to disable a rule for a single violation, try to do it as locally as possible
   ```javascript
-  // bad
-  /* eslint-disable no-new */
+    // bad
+    /* eslint-disable no-new */
 
-  import Foo from 'foo';
+    import Foo from 'foo';
 
-  new Foo();
+    new Foo();
 
-  // better
-  import Foo from 'foo';
+    // better
+    import Foo from 'foo';
 
-  // eslint-disable-next-line no-new
-  new Foo();
+    // eslint-disable-next-line no-new
+    new Foo();
   ```
+1. There are few rules that we need to disable due to technical debt. Which are:
+  1. [no-new][eslint-new]
+  1. [class-methods-use-this][eslint-this]
 
-- When they are needed _always_ place ESlint directive comment blocks on the first line of a script, followed by any global declarations, then a blank newline prior to any imports or code.
-
+1. When they are needed _always_ place ESlint directive comment blocks on the first line of a script, followed by any global declarations, then a blank newline prior to any imports or code.
   ```javascript
-  // bad
-  /* global Foo */
-  /* eslint-disable no-new */
-  import Bar from './bar';
+    // bad
+    /* global Foo */
+    /* eslint-disable no-new */
+    import Bar from './bar';
 
-  // good
-  /* eslint-disable no-new */
-  /* global Foo */
+    // good
+    /* eslint-disable no-new */
+    /* global Foo */
 
-  import Bar from './bar';
+    import Bar from './bar';
   ```
 
-- **Never** disable the `no-undef` rule. Declare globals with `/* global Foo */` instead.
-
-- When declaring multiple globals, always use one `/* global [name] */` line per variable.
+1. **Never** disable the `no-undef` rule. Declare globals with `/* global Foo */` instead.
 
+1. When declaring multiple globals, always use one `/* global [name] */` line per variable.
   ```javascript
-  // bad
-  /* globals Flash, Cookies, jQuery */
+    // bad
+    /* globals Flash, Cookies, jQuery */
 
-  // good
-  /* global Flash */
-  /* global Cookies */
-  /* global jQuery */
+    // good
+    /* global Flash */
+    /* global Cookies */
+    /* global jQuery */
   ```
-  
-- Use up to 3 parameters for a function or class. If you need more accept an Object instead.
 
+1. Use up to 3 parameters for a function or class. If you need more accept an Object instead.
   ```javascript
-  // bad
-  fn(p1, p2, p3, p4) {}
+    // bad
+    fn(p1, p2, p3, p4) {}
 
-  // good
-  fn(options) {}
+    // good
+    fn(options) {}
   ```
 
 #### Modules, Imports, and Exports
-- Use ES module syntax to import modules
-
+1. Use ES module syntax to import modules
   ```javascript
-  // bad
-  require('foo');
+    // bad
+    require('foo');
 
-  // good
-  import Foo from 'foo';
+    // good
+    import Foo from 'foo';
 
-  // bad
-  module.exports = Foo;
+    // bad
+    module.exports = Foo;
 
-  // good
-  export default Foo;
+    // good
+    export default Foo;
   ```
 
-- Relative paths
+1. Relative paths: Unless you are writing a test, always reference other scripts using relative paths instead of `~`
+  * In **app/assets/javascripts**:
 
-  Unless you are writing a test, always reference other scripts using relative paths instead of `~`
+    ```javascript
+      // bad
+      import Foo from '~/foo'
 
-  In **app/assets/javascripts**:
-  ```javascript
-  // bad
-  import Foo from '~/foo'
+      // good
+      import Foo from '../foo';
+    ```
+  * In **spec/javascripts**:
 
-  // good
-  import Foo from '../foo';
-  ```
-
-  In **spec/javascripts**:
-  ```javascript
-  // bad
-  import Foo from '../../app/assets/javascripts/foo'
-
-  // good
-  import Foo from '~/foo';
-  ```
+    ```javascript
+      // bad
+      import Foo from '../../app/assets/javascripts/foo'
 
-- Avoid using IIFE. Although we have a lot of examples of files which wrap their contents in IIFEs (immediately-invoked function expressions), this is no longer necessary after the transition from Sprockets to webpack. Do not use them anymore and feel free to remove them when refactoring legacy code.
+      // good
+      import Foo from '~/foo';
+    ```
 
-- Avoid adding to the global namespace.
+1. Avoid using IIFE. Although we have a lot of examples of files which wrap their contents in IIFEs (immediately-invoked function expressions), this is no longer necessary after the transition from Sprockets to webpack. Do not use them anymore and feel free to remove them when refactoring legacy code.
 
+1. Avoid adding to the global namespace.
   ```javascript
-  // bad
-  window.MyClass = class { /* ... */ };
+    // bad
+    window.MyClass = class { /* ... */ };
 
-  // good
-  export default class MyClass { /* ... */ }
+    // good
+    export default class MyClass { /* ... */ }
   ```
 
-- Side effects are forbidden in any script which contains exports
-
+1. Side effects are forbidden in any script which contains exports
   ```javascript
-  // bad
-  export default class MyClass { /* ... */ }
+    // bad
+    export default class MyClass { /* ... */ }
 
-  document.addEventListener("DOMContentLoaded", function(event) {
-    new MyClass();
-  }
+    document.addEventListener("DOMContentLoaded", function(event) {
+      new MyClass();
+    }
   ```
 
 
 #### Data Mutation and Pure functions
-- Strive to write many small pure functions, and minimize where mutations occur.
-
+1. Strive to write many small pure functions, and minimize where mutations occur.
   ```javascript
-  // bad
-  const values = {foo: 1};
+    // bad
+    const values = {foo: 1};
 
-  function impureFunction(items) {
-    const bar = 1;
+    function impureFunction(items) {
+      const bar = 1;
 
-    items.foo = items.a * bar + 2;
+      items.foo = items.a * bar + 2;
 
-    return items.a;
-  }
+      return items.a;
+    }
 
-  const c = impureFunction(values);
+    const c = impureFunction(values);
 
-  // good
-  var values = {foo: 1};
+    // good
+    var values = {foo: 1};
 
-  function pureFunction (foo) {
-    var bar = 1;
+    function pureFunction (foo) {
+      var bar = 1;
 
-    foo = foo * bar + 2;
+      foo = foo * bar + 2;
 
-    return foo;
-  }
+      return foo;
+    }
 
-  var c = pureFunction(values.foo);
+    var c = pureFunction(values.foo);
   ```
 
-- Avoid constructors with side-effects
+1. Avoid constructors with side-effects
 
-- Prefer `.map`, `.reduce` or `.filter` over `.forEach`
+1. Prefer `.map`, `.reduce` or `.filter` over `.forEach`
 A forEach will cause side effects, it will be mutating the array being iterated. Prefer using `.map`,
 `.reduce` or `.filter`
-
   ```javascript
-  const users = [ { name: 'Foo' }, { name: 'Bar' } ];
+    const users = [ { name: 'Foo' }, { name: 'Bar' } ];
 
-  // bad
-  users.forEach((user, index) => {
-    user.id = index;
-  });
+    // bad
+    users.forEach((user, index) => {
+      user.id = index;
+    });
 
-  // good
-  const usersWithId = users.map((user, index) => {
-    return Object.assign({}, user, { id: index });
-  });
+    // good
+    const usersWithId = users.map((user, index) => {
+      return Object.assign({}, user, { id: index });
+    });
   ```
 
 #### Parse Strings into Numbers
-- `parseInt()` is preferable over `Number()` or `+`
-
+1. `parseInt()` is preferable over `Number()` or `+`
   ```javascript
-  // bad
-  +'10' // 10
+    // bad
+    +'10' // 10
 
-  // good
-  Number('10') // 10
+    // good
+    Number('10') // 10
 
-  // better
-  parseInt('10', 10);
+    // better
+    parseInt('10', 10);
   ```
 
 #### CSS classes used for JavaScript
-- If the class is being used in Javascript it needs to be prepend with `js-`
+1. If the class is being used in Javascript it needs to be prepend with `js-`
   ```html
     // bad
     <button class="add-user">
@@ -226,234 +216,222 @@ A forEach will cause side effects, it will be mutating the array being iterated.
 
 ### Vue.js
 
-
 #### Basic Rules
-- Only include one Vue.js component per file.
-- Export components as plain objects:
-
+1. Only include one Vue.js component per file.
+1. Export components as plain objects:
   ```javascript
-  export default {
-    template: `<h1>I'm a component</h1>
-  }
+    export default {
+      template: `<h1>I'm a component</h1>
+    }
   ```
+1.
 
 #### Naming
-- **Extensions**: Use `.vue` extension for Vue components.
-- **Reference Naming**: Use PascalCase for Vue components and camelCase for their instances:
-
+1. **Extensions**: Use `.vue` extension for Vue components.
+1. **Reference Naming**: Use PascalCase for Vue components and camelCase for their instances:
   ```javascript
-  // bad
-  import cardBoard from 'cardBoard';
+    // bad
+    import cardBoard from 'cardBoard';
 
-  // good
-  import CardBoard from 'cardBoard'
+    // good
+    import CardBoard from 'cardBoard'
 
-  // bad
-  components: {
-    CardBoard: CardBoard
-  };
+    // bad
+    components: {
+      CardBoard: CardBoard
+    };
 
-  // good
-  components: {
-    cardBoard: CardBoard
-  };
+    // good
+    components: {
+      cardBoard: CardBoard
+    };
   ```
 
-- **Props Naming:**
-- Avoid using DOM component prop names.
-- Use kebab-case instead of camelCase to provide props in templates.
-
+1. **Props Naming:**
+1. Avoid using DOM component prop names.
+1. Use kebab-case instead of camelCase to provide props in templates.
   ```javascript
-  // bad
-  <component class="btn">
+    // bad
+    <component class="btn">
 
-  // good
-  <component css-class="btn">
+    // good
+    <component css-class="btn">
 
-  // bad
-  <component myProp="prop" />
+    // bad
+    <component myProp="prop" />
 
-  // good
-  <component my-prop="prop" />
-```
+    // good
+    <component my-prop="prop" />
+  ```
 
 #### Alignment
-- Follow these alignment styles for the template method:
-
+1. Follow these alignment styles for the template method:
   ```javascript
-  // bad
-  <component v-if="bar"
-      param="baz" />
+    // bad
+    <component v-if="bar"
+        param="baz" />
 
-  <button class="btn">Click me</button>
+    <button class="btn">Click me</button>
 
-  // good
-  <component
-    v-if="bar"
-    param="baz"
-  />
+    // good
+    <component
+      v-if="bar"
+      param="baz"
+    />
 
-  <button class="btn">
-    Click me
-  </button>
+    <button class="btn">
+      Click me
+    </button>
 
-  // if props fit in one line then keep it on the same line
-  <component bar="bar" />
+    // if props fit in one line then keep it on the same line
+    <component bar="bar" />
   ```
 
 #### Quotes
-- Always use double quotes `"` inside templates and single quotes `'` for all other JS.
-
+1. Always use double quotes `"` inside templates and single quotes `'` for all other JS.
   ```javascript
-  // bad
-  template: `
-    <button :class='style'>Button</button>
-  `
-
-  // good
-  template: `
-    <button :class="style">Button</button>
-  `
+    // bad
+    template: `
+      <button :class='style'>Button</button>
+    `
+
+    // good
+    template: `
+      <button :class="style">Button</button>
+    `
   ```
 
 #### Props
-- Props should be declared as an object
-
+1. Props should be declared as an object
   ```javascript
-  // bad
-  props: ['foo']
-
-  // good
-  props: {
-    foo: {
-      type: String,
-      required: false,
-      default: 'bar'
+    // bad
+    props: ['foo']
+
+    // good
+    props: {
+      foo: {
+        type: String,
+        required: false,
+        default: 'bar'
+      }
     }
-  }
   ```
 
-- Required key should always be provided when declaring a prop
-
+1. Required key should always be provided when declaring a prop
   ```javascript
-  // bad
-  props: {
-    foo: {
-      type: String,
+    // bad
+    props: {
+      foo: {
+        type: String,
+      }
     }
-  }
-
-  // good
-  props: {
-    foo: {
-      type: String,
-      required: false,
-      default: 'bar'
+
+    // good
+    props: {
+      foo: {
+        type: String,
+        required: false,
+        default: 'bar'
+      }
     }
-  }
   ```
 
-- Default key should always be provided if the prop is not required:
-
+1. Default key should always be provided if the prop is not required:
   ```javascript
-  // bad
-  props: {
-    foo: {
-      type: String,
-      required: false,
+    // bad
+    props: {
+      foo: {
+        type: String,
+        required: false,
+      }
     }
-  }
-
-  // good
-  props: {
-    foo: {
-      type: String,
-      required: false,
-      default: 'bar'
+
+    // good
+    props: {
+      foo: {
+        type: String,
+        required: false,
+        default: 'bar'
+      }
     }
-  }
 
-  // good
-  props: {
-    foo: {
-      type: String,
-      required: true
+    // good
+    props: {
+      foo: {
+        type: String,
+        required: true
+      }
     }
-  }
   ```
 
 #### Data
-- `data` method should always be a function
+1. `data` method should always be a function
 
   ```javascript
-  // bad
-  data: {
-    foo: 'foo'
-  }
-
-  // good
-  data() {
-    return {
+    // bad
+    data: {
       foo: 'foo'
-    };
-  }
+    }
+
+    // good
+    data() {
+      return {
+        foo: 'foo'
+      };
+    }
   ```
 
 #### Directives
 
-- Shorthand `@` is preferable over `v-on`
-
+1. Shorthand `@` is preferable over `v-on`
   ```javascript
-  // bad
-  <component v-on:click="eventHandler"/>
+    // bad
+    <component v-on:click="eventHandler"/>
 
 
-  // good
-  <component @click="eventHandler"/>
+    // good
+    <component @click="eventHandler"/>
   ```
 
-- Shorthand `:` is preferable over `v-bind`
-
+1. Shorthand `:` is preferable over `v-bind`
   ```javascript
-  // bad
-  <component v-bind:class="btn"/>
+    // bad
+    <component v-bind:class="btn"/>
 
 
-  // good
-  <component :class="btn"/>
+    // good
+    <component :class="btn"/>
   ```
 
 #### Closing tags
-- Prefer self closing component tags
-
+1. Prefer self closing component tags
   ```javascript
-  // bad
-  <component></component>
+    // bad
+    <component></component>
 
-  // good
-  <component />
+    // good
+    <component />
   ```
 
 #### Ordering
-- Order for a Vue Component:
+1. Order for a Vue Component:
   1. `name`
-  2. `props`
-  3. `data`
-  4. `components`
-  5. `computedProps`
-  6. `methods`
-  7. lifecycle methods
-    1. `beforeCreate`
-    2. `created`
-    3. `beforeMount`
-    4. `mounted`
-    5. `beforeUpdate`
-    6. `updated`
-    7. `activated`
-    8. `deactivated`
-    9. `beforeDestroy`
-    10. `destroyed`
-  8. `template`
+  1. `props`
+  1. `mixins`
+  1. `data`
+  1. `components`
+  1. `computedProps`
+  1. `methods`
+  1. `beforeCreate`
+  1. `created`
+  1. `beforeMount`
+  1. `mounted`
+  1. `beforeUpdate`
+  1. `updated`
+  1. `activated`
+  1. `deactivated`
+  1. `beforeDestroy`
+  1. `destroyed`
 
 
 ## SCSS
@@ -461,3 +439,5 @@ A forEach will cause side effects, it will be mutating the array being iterated.
 
 [airbnb-js-style-guide]: https://github.com/airbnb/javascript
 [eslintrc]: https://gitlab.com/gitlab-org/gitlab-ce/blob/master/.eslintrc
+[eslint-this]: http://eslint.org/docs/rules/class-methods-use-this
+[eslint-new]: http://eslint.org/docs/rules/no-new
diff --git a/doc/development/fe_guide/vue.md b/doc/development/fe_guide/vue.md
index 73d2ffc1bdc7b..a984bb6c94ce6 100644
--- a/doc/development/fe_guide/vue.md
+++ b/doc/development/fe_guide/vue.md
@@ -387,6 +387,10 @@ describe('Todos App', () => {
   });
 });
 ```
+#### Test the component's output
+The main return value of a Vue component is the rendered output. In order to test the component we
+need to test the rendered output. [Vue][vue-test] guide's to unit test show us exactly that:
+
 
 ### Stubbing API responses
 [Vue Resource Interceptors][vue-resource-interceptor] allow us to add a interceptor with
@@ -419,6 +423,16 @@ the response we need:
   });
 ```
 
+1. Use `$.mount()` to mount the component
+```javascript
+  // bad
+  new Component({
+    el: document.createElement('div')
+  });
+
+  // good
+  new Component().$mount();
+```
 
 [vue-docs]: http://vuejs.org/guide/index.html
 [issue-boards]: https://gitlab.com/gitlab-org/gitlab-ce/tree/master/app/assets/javascripts/boards
@@ -429,5 +443,6 @@ the response we need:
 [one-way-data-flow]: https://vuejs.org/v2/guide/components.html#One-Way-Data-Flow
 [vue-resource-repo]: https://github.com/pagekit/vue-resource
 [vue-resource-interceptor]: https://github.com/pagekit/vue-resource/blob/develop/docs/http.md#interceptors
+[vue-test]: https://vuejs.org/v2/guide/unit-testing.html
 [issue-boards-service]: https://gitlab.com/gitlab-org/gitlab-ce/blob/master/app/assets/javascripts/boards/services/board_service.js.es6
 [flux]: https://facebook.github.io/flux
-- 
GitLab


From 387c4b2c21a44360386a9b8ce6849e7f1b8a3de9 Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Thu, 4 May 2017 15:11:15 +0300
Subject: [PATCH 212/363] Backport of multiple_assignees_feature [ci skip]

---
 .../blob/target_branch_dropdown.js            |   4 +-
 .../javascripts/blob/template_selector.js     |   7 +-
 .../javascripts/boards/boards_bundle.js       |   2 +-
 .../boards/components/board_new_issue.js      |   1 +
 .../boards/components/board_sidebar.js        |  61 ++-
 .../boards/components/issue_card_inner.js     | 104 +++--
 .../boards/components/new_list_dropdown.js    |   4 +-
 .../boards/models/{user.js => assignee.js}    |   6 +-
 app/assets/javascripts/boards/models/issue.js |  32 +-
 app/assets/javascripts/gl_dropdown.js         |  56 ++-
 .../javascripts/issuable/issuable_bundle.js   |   1 -
 .../components/no_tracking_pane.js            |  12 -
 .../time_tracking/time_tracking_bundle.js     |  66 ---
 app/assets/javascripts/issue_status_select.js |   4 +-
 .../javascripts/issues_bulk_assignment.js     |   3 +
 app/assets/javascripts/labels_select.js       |   7 +-
 app/assets/javascripts/main.js                |   1 -
 app/assets/javascripts/members.js             |  19 +
 app/assets/javascripts/milestone_select.js    |  23 ++
 app/assets/javascripts/namespace_select.js    |   3 +-
 app/assets/javascripts/project.js             |   3 +-
 .../protected_branch_access_dropdown.js       |   5 +-
 .../protected_branch_dropdown.js              |   3 +-
 .../components/assignees/assignee_title.js    |  41 ++
 .../sidebar/components/assignees/assignees.js | 224 +++++++++++
 .../components/assignees/sidebar_assignees.js |  84 ++++
 .../time_tracking/collapsed_state.js          |  97 +++++
 .../time_tracking/comparison_pane.js          |  98 +++++
 .../time_tracking/estimate_only_pane.js       |  17 +
 .../components/time_tracking/help_state.js    |  44 ++
 .../time_tracking/no_tracking_pane.js         |  10 +
 .../time_tracking/sidebar_time_tracking.js    |  45 +++
 .../time_tracking/spent_only_pane.js          |  15 +
 .../components/time_tracking/time_tracker.js  | 163 ++++++++
 app/assets/javascripts/sidebar/event_hub.js   |   3 +
 .../sidebar/services/sidebar_service.js       |  28 ++
 .../javascripts/sidebar/sidebar_bundle.js     |  21 +
 .../javascripts/sidebar/sidebar_mediator.js   |  38 ++
 .../sidebar/stores/sidebar_store.js           |  52 +++
 app/assets/javascripts/subbable_resource.js   |  51 ---
 app/assets/javascripts/subscription_select.js |   4 +-
 app/assets/javascripts/users_select.js        | 377 ++++++++++++++----
 app/assets/stylesheets/framework/avatar.scss  |  11 +
 .../stylesheets/framework/dropdowns.scss      |  12 +-
 app/assets/stylesheets/framework/lists.scss   |   1 +
 app/assets/stylesheets/pages/boards.scss      |  74 +++-
 app/assets/stylesheets/pages/diff.scss        |   9 +-
 app/assets/stylesheets/pages/issuable.scss    | 125 +++++-
 app/controllers/concerns/issuable_actions.rb  |   1 +
 .../concerns/issuable_collections.rb          |   2 +-
 .../projects/boards/issues_controller.rb      |   2 +-
 app/controllers/projects/issues_controller.rb |   6 +-
 app/finders/issuable_finder.rb                |   2 +-
 app/finders/issues_finder.rb                  |  19 +-
 app/helpers/form_helper.rb                    |  33 ++
 app/helpers/issuables_helper.rb               |  10 +
 app/mailers/emails/issues.rb                  |   6 +-
 app/models/concerns/issuable.rb               |  27 +-
 app/models/concerns/milestoneish.rb           |   2 +-
 app/models/global_milestone.rb                |   6 +-
 app/models/issue.rb                           |  35 +-
 app/models/issue_assignee.rb                  |  29 ++
 app/models/merge_request.rb                   |  32 ++
 app/models/milestone.rb                       |   5 +-
 app/models/user.rb                            |   5 +
 app/serializers/issuable_entity.rb            |   1 -
 app/serializers/issue_entity.rb               |   1 +
 app/serializers/merge_request_entity.rb       |   1 +
 app/services/issuable/bulk_update_service.rb  |   6 +-
 app/services/issuable_base_service.rb         |  23 +-
 app/services/issues/base_service.rb           |  19 +
 app/services/issues/update_service.rb         |  14 +-
 .../merge_requests/assign_issues_service.rb   |   4 +-
 app/services/merge_requests/base_service.rb   |   5 +
 app/services/merge_requests/update_service.rb |   5 +-
 .../notification_recipient_service.rb         |   7 +-
 app/services/notification_service.rb          |  27 +-
 .../slash_commands/interpret_service.rb       |  23 +-
 app/services/system_note_service.rb           |  38 ++
 app/services/todo_service.rb                  |   4 +-
 app/views/issues/_issue.atom.builder          |  10 +-
 .../_reassigned_issuable_email.text.erb       |   6 -
 app/views/notify/new_issue_email.html.haml    |   4 +-
 app/views/notify/new_issue_email.text.erb     |   2 +-
 .../new_mention_in_issue_email.text.erb       |   2 +-
 .../notify/reassigned_issue_email.html.haml   |  11 +-
 .../notify/reassigned_issue_email.text.erb    |   7 +-
 .../reassigned_merge_request_email.html.haml  |  10 +-
 .../reassigned_merge_request_email.text.erb   |   7 +-
 .../components/sidebar/_assignee.html.haml    |  48 +--
 app/views/projects/issues/_issue.html.haml    |   4 +-
 .../shared/issuable/_assignees.html.haml      |  15 +
 .../shared/issuable/_participants.html.haml   |   8 +-
 .../shared/issuable/_search_bar.html.haml     |   9 +
 app/views/shared/issuable/_sidebar.html.haml  |  95 +++--
 .../issuable/form/_issue_assignee.html.haml   |  30 ++
 .../form/_merge_request_assignee.html.haml    |  31 ++
 .../shared/issuable/form/_metadata.html.haml  |  25 +-
 .../shared/milestones/_issuable.html.haml     |   8 +-
 .../update-issue-board-cards-design.yml       |   4 +
 config/webpack.config.js                      |   7 +-
 db/fixtures/development/09_issues.rb          |   2 +-
 ...0320171632_create_issue_assignees_table.rb |  40 ++
 .../20170320173259_migrate_assignees.rb       |  52 +++
 db/schema.rb                                  |  10 +
 doc/api/issues.md                             |  84 +++-
 doc/user/project/integrations/webhooks.md     |  12 +
 features/steps/dashboard/dashboard.rb         |   2 +-
 features/steps/dashboard/todos.rb             |   2 +-
 features/steps/group/milestones.rb            |   4 +-
 features/steps/groups.rb                      |   4 +-
 lib/api/api.rb                                |   2 +
 lib/api/entities.rb                           |   6 +-
 lib/api/helpers/common_helpers.rb             |  13 +
 lib/api/issues.rb                             |   9 +-
 lib/api/v3/entities.rb                        |   7 +
 lib/api/v3/issues.rb                          |  33 +-
 lib/api/v3/merge_requests.rb                  |   2 +-
 lib/api/v3/milestones.rb                      |   4 +-
 lib/banzai/reference_parser/issue_parser.rb   |   2 +-
 .../chat_commands/presenters/issue_base.rb    |   2 +-
 lib/gitlab/fogbugz_import/importer.rb         |  18 +-
 lib/gitlab/github_import/issue_formatter.rb   |   2 +-
 lib/gitlab/google_code_import/importer.rb     |  14 +-
 .../dashboard/todos_controller_spec.rb        |   2 +-
 .../projects/boards/issues_controller_spec.rb |   2 +-
 .../projects/issues_controller_spec.rb        |   8 +-
 .../merge_requests_controller_spec.rb         |   2 +-
 spec/features/atom/dashboard_issues_spec.rb   |   8 +-
 spec/features/atom/issues_spec.rb             |   6 +-
 spec/features/boards/boards_spec.rb           |   2 +-
 spec/features/boards/modal_filter_spec.rb     |   2 +-
 spec/features/boards/sidebar_spec.rb          |  35 +-
 .../dashboard/issuables_counter_spec.rb       |   4 +-
 spec/features/dashboard/issues_spec.rb        |   7 +-
 spec/features/dashboard_issues_spec.rb        |   4 +-
 .../features/gitlab_flavored_markdown_spec.rb |   4 +-
 spec/features/issues/award_emoji_spec.rb      |   2 +-
 .../filtered_search/filter_issues_spec.rb     |  11 +-
 spec/features/issues/form_spec.rb             |  54 ++-
 spec/features/issues/issue_sidebar_spec.rb    |  15 +
 spec/features/issues/update_issues_spec.rb    |   2 +-
 spec/features/issues_spec.rb                  |  40 +-
 .../merge_requests/assign_issues_spec.rb      |   2 +-
 spec/features/milestones/show_spec.rb         |   2 +-
 .../projects/issuable_templates_spec.rb       |   4 +-
 spec/features/search_spec.rb                  |   2 +-
 spec/features/unsubscribe_links_spec.rb       |   2 +-
 spec/finders/issues_finder_spec.rb            |  26 +-
 spec/fixtures/api/schemas/issue.json          |  20 +
 .../api/schemas/public_api/v4/issues.json     |  17 +-
 spec/helpers/issuables_helper_spec.rb         |  17 +
 spec/javascripts/boards/boards_store_spec.js  |  19 +-
 spec/javascripts/boards/issue_card_spec.js    | 110 ++++-
 spec/javascripts/boards/issue_spec.js         |  76 +++-
 spec/javascripts/boards/list_spec.js          |  19 +-
 spec/javascripts/boards/mock_data.js          |   3 +-
 spec/javascripts/boards/modal_store_spec.js   |  12 +-
 .../javascripts/issuable_time_tracker_spec.js | 250 ++++++------
 spec/javascripts/subbable_resource_spec.js    |  63 ---
 .../lib/banzai/filter/redactor_filter_spec.rb |   2 +-
 .../github_import/issue_formatter_spec.rb     |  10 +-
 .../google_code_import/importer_spec.rb       |   2 +-
 spec/lib/gitlab/import_export/all_models.yml  |   3 +-
 .../import_export/project_tree_saver_spec.rb  |   2 +-
 .../lib/gitlab/project_search_results_spec.rb |   2 +-
 spec/lib/gitlab/search_results_spec.rb        |   4 +-
 spec/mailers/notify_spec.rb                   |  10 +-
 spec/models/concerns/issuable_spec.rb         | 109 +----
 spec/models/concerns/milestoneish_spec.rb     |   6 +-
 spec/models/event_spec.rb                     |   4 +-
 spec/models/issue_collection_spec.rb          |   2 +-
 spec/models/issue_spec.rb                     |  91 ++++-
 spec/models/merge_request_spec.rb             |  91 ++++-
 spec/requests/api/issues_spec.rb              |  76 +++-
 spec/requests/api/v3/issues_spec.rb           |  72 +++-
 .../issuable/bulk_update_service_spec.rb      |  62 ++-
 spec/services/issues/close_service_spec.rb    |   2 +-
 spec/services/issues/create_service_spec.rb   |  86 +++-
 spec/services/issues/update_service_spec.rb   |  59 ++-
 .../assign_issues_service_spec.rb             |  10 +-
 .../merge_requests/create_service_spec.rb     |  82 +++-
 .../merge_requests/update_service_spec.rb     |  48 +++
 .../notes/slash_commands_service_spec.rb      |   4 +-
 spec/services/notification_service_spec.rb    |  80 ++--
 .../projects/autocomplete_service_spec.rb     |   2 +-
 .../slash_commands/interpret_service_spec.rb  |  70 +++-
 spec/services/system_note_service_spec.rb     |  45 +++
 spec/services/todo_service_spec.rb            |  26 +-
 ...issuable_slash_commands_shared_examples.rb |   4 +-
 .../import_export/export_file_helper.rb       |   2 +-
 ...issuable_create_service_shared_examples.rb |  52 ---
 ..._service_slash_commands_shared_examples.rb |  18 +-
 ...issuable_update_service_shared_examples.rb |  48 ---
 spec/support/time_tracking_shared_examples.rb |  10 +-
 195 files changed, 3966 insertions(+), 1224 deletions(-)
 rename app/assets/javascripts/boards/models/{user.js => assignee.js} (61%)
 delete mode 100644 app/assets/javascripts/issuable/issuable_bundle.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/no_tracking_pane.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/time_tracking_bundle.js
 create mode 100644 app/assets/javascripts/sidebar/components/assignees/assignee_title.js
 create mode 100644 app/assets/javascripts/sidebar/components/assignees/assignees.js
 create mode 100644 app/assets/javascripts/sidebar/components/assignees/sidebar_assignees.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/collapsed_state.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/comparison_pane.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/estimate_only_pane.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/help_state.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/no_tracking_pane.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/spent_only_pane.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/time_tracker.js
 create mode 100644 app/assets/javascripts/sidebar/event_hub.js
 create mode 100644 app/assets/javascripts/sidebar/services/sidebar_service.js
 create mode 100644 app/assets/javascripts/sidebar/sidebar_bundle.js
 create mode 100644 app/assets/javascripts/sidebar/sidebar_mediator.js
 create mode 100644 app/assets/javascripts/sidebar/stores/sidebar_store.js
 delete mode 100644 app/assets/javascripts/subbable_resource.js
 create mode 100644 app/models/issue_assignee.rb
 delete mode 100644 app/views/notify/_reassigned_issuable_email.text.erb
 create mode 100644 app/views/shared/issuable/_assignees.html.haml
 create mode 100644 app/views/shared/issuable/form/_issue_assignee.html.haml
 create mode 100644 app/views/shared/issuable/form/_merge_request_assignee.html.haml
 create mode 100644 changelogs/unreleased/update-issue-board-cards-design.yml
 create mode 100644 db/migrate/20170320171632_create_issue_assignees_table.rb
 create mode 100644 db/migrate/20170320173259_migrate_assignees.rb
 create mode 100644 lib/api/helpers/common_helpers.rb
 delete mode 100644 spec/javascripts/subbable_resource_spec.js
 delete mode 100644 spec/support/services/issuable_create_service_shared_examples.rb

diff --git a/app/assets/javascripts/blob/target_branch_dropdown.js b/app/assets/javascripts/blob/target_branch_dropdown.js
index 216f069ef71cd..d52d69b127488 100644
--- a/app/assets/javascripts/blob/target_branch_dropdown.js
+++ b/app/assets/javascripts/blob/target_branch_dropdown.js
@@ -37,8 +37,8 @@ class TargetBranchDropDown {
         }
         return SELECT_ITEM_MSG;
       },
-      clicked(item, el, e) {
-        e.preventDefault();
+      clicked(options) {
+        options.e.preventDefault();
         self.onClick.call(self);
       },
       fieldName: self.fieldName,
diff --git a/app/assets/javascripts/blob/template_selector.js b/app/assets/javascripts/blob/template_selector.js
index d7c1c32efbd40..888883163c5b2 100644
--- a/app/assets/javascripts/blob/template_selector.js
+++ b/app/assets/javascripts/blob/template_selector.js
@@ -24,7 +24,7 @@ export default class TemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (item, el, e) => this.fetchFileTemplate(item, el, e),
+      clicked: options => this.fetchFileTemplate(options),
       text: item => item.name,
     });
   }
@@ -51,7 +51,10 @@ export default class TemplateSelector {
     return this.$dropdownContainer.removeClass('hidden');
   }
 
-  fetchFileTemplate(item, el, e) {
+  fetchFileTemplate(options) {
+    const { e } = options;
+    const item = options.selectedObj;
+
     e.preventDefault();
     return this.requestFile(item);
   }
diff --git a/app/assets/javascripts/boards/boards_bundle.js b/app/assets/javascripts/boards/boards_bundle.js
index b6dee8177d276..dbbfb6de3ea33 100644
--- a/app/assets/javascripts/boards/boards_bundle.js
+++ b/app/assets/javascripts/boards/boards_bundle.js
@@ -11,7 +11,7 @@ require('./models/issue');
 require('./models/label');
 require('./models/list');
 require('./models/milestone');
-require('./models/user');
+require('./models/assignee');
 require('./stores/boards_store');
 require('./stores/modal_store');
 require('./services/board_service');
diff --git a/app/assets/javascripts/boards/components/board_new_issue.js b/app/assets/javascripts/boards/components/board_new_issue.js
index 0fa85b6fe1406..1ce95b62138d9 100644
--- a/app/assets/javascripts/boards/components/board_new_issue.js
+++ b/app/assets/javascripts/boards/components/board_new_issue.js
@@ -26,6 +26,7 @@ export default {
         title: this.title,
         labels,
         subscribed: true,
+        assignees: [],
       });
 
       this.list.newIssue(issue)
diff --git a/app/assets/javascripts/boards/components/board_sidebar.js b/app/assets/javascripts/boards/components/board_sidebar.js
index f0066d4ec5d72..317cef9f22750 100644
--- a/app/assets/javascripts/boards/components/board_sidebar.js
+++ b/app/assets/javascripts/boards/components/board_sidebar.js
@@ -3,8 +3,13 @@
 /* global MilestoneSelect */
 /* global LabelsSelect */
 /* global Sidebar */
+/* global Flash */
 
 import Vue from 'vue';
+import eventHub from '../../sidebar/event_hub';
+
+import AssigneeTitle from '../../sidebar/components/assignees/assignee_title';
+import Assignees from '../../sidebar/components/assignees/assignees';
 
 require('./sidebar/remove_issue');
 
@@ -22,6 +27,7 @@ gl.issueBoards.BoardSidebar = Vue.extend({
       detail: Store.detail,
       issue: {},
       list: {},
+      loadingAssignees: false,
     };
   },
   computed: {
@@ -43,6 +49,10 @@ gl.issueBoards.BoardSidebar = Vue.extend({
 
         this.issue = this.detail.issue;
         this.list = this.detail.list;
+
+        this.$nextTick(() => {
+          this.endpoint = this.$refs.assigneeDropdown.dataset.issueUpdate;
+        });
       },
       deep: true
     },
@@ -53,12 +63,57 @@ gl.issueBoards.BoardSidebar = Vue.extend({
           $('.right-sidebar').getNiceScroll().resize();
         });
       }
-    }
+
+      this.issue = this.detail.issue;
+      this.list = this.detail.list;
+    },
+    deep: true
   },
   methods: {
     closeSidebar () {
       this.detail.issue = {};
-    }
+    },
+    assignSelf () {
+      // Notify gl dropdown that we are now assigning to current user
+      this.$refs.assigneeBlock.dispatchEvent(new Event('assignYourself'));
+
+      this.addAssignee(this.currentUser);
+      this.saveAssignees();
+    },
+    removeAssignee (a) {
+      gl.issueBoards.BoardsStore.detail.issue.removeAssignee(a);
+    },
+    addAssignee (a) {
+      gl.issueBoards.BoardsStore.detail.issue.addAssignee(a);
+    },
+    removeAllAssignees () {
+      gl.issueBoards.BoardsStore.detail.issue.removeAllAssignees();
+    },
+    saveAssignees () {
+      this.loadingAssignees = true;
+
+      gl.issueBoards.BoardsStore.detail.issue.update(this.endpoint)
+        .then(() => {
+          this.loadingAssignees = false;
+        })
+        .catch(() => {
+          this.loadingAssignees = false;
+          return new Flash('An error occurred while saving assignees');
+        });
+    },
+  },
+  created () {
+    // Get events from glDropdown
+    eventHub.$on('sidebar.removeAssignee', this.removeAssignee);
+    eventHub.$on('sidebar.addAssignee', this.addAssignee);
+    eventHub.$on('sidebar.removeAllAssignees', this.removeAllAssignees);
+    eventHub.$on('sidebar.saveAssignees', this.saveAssignees);
+  },
+  beforeDestroy() {
+    eventHub.$off('sidebar.removeAssignee', this.removeAssignee);
+    eventHub.$off('sidebar.addAssignee', this.addAssignee);
+    eventHub.$off('sidebar.removeAllAssignees', this.removeAllAssignees);
+    eventHub.$off('sidebar.saveAssignees', this.saveAssignees);
   },
   mounted () {
     new IssuableContext(this.currentUser);
@@ -70,5 +125,7 @@ gl.issueBoards.BoardSidebar = Vue.extend({
   },
   components: {
     removeBtn: gl.issueBoards.RemoveIssueBtn,
+    'assignee-title': AssigneeTitle,
+    assignees: Assignees,
   },
 });
diff --git a/app/assets/javascripts/boards/components/issue_card_inner.js b/app/assets/javascripts/boards/components/issue_card_inner.js
index fc154ee7b8b03..2f06d186c508b 100644
--- a/app/assets/javascripts/boards/components/issue_card_inner.js
+++ b/app/assets/javascripts/boards/components/issue_card_inner.js
@@ -31,18 +31,36 @@ gl.issueBoards.IssueCardInner = Vue.extend({
       default: false,
     },
   },
+  data() {
+    return {
+      limitBeforeCounter: 3,
+      maxRender: 4,
+      maxCounter: 99,
+    };
+  },
   computed: {
-    cardUrl() {
-      return `${this.issueLinkBase}/${this.issue.id}`;
+    numberOverLimit() {
+      return this.issue.assignees.length - this.limitBeforeCounter;
     },
-    assigneeUrl() {
-      return `${this.rootPath}${this.issue.assignee.username}`;
+    assigneeCounterTooltip() {
+      return `${this.assigneeCounterLabel} more`;
+    },
+    assigneeCounterLabel() {
+      if (this.numberOverLimit > this.maxCounter) {
+        return `${this.maxCounter}+`;
+      }
+
+      return `+${this.numberOverLimit}`;
     },
-    assigneeUrlTitle() {
-      return `Assigned to ${this.issue.assignee.name}`;
+    shouldRenderCounter() {
+      if (this.issue.assignees.length <= this.maxRender) {
+        return false;
+      }
+
+      return this.issue.assignees.length > this.numberOverLimit;
     },
-    avatarUrlTitle() {
-      return `Avatar for ${this.issue.assignee.name}`;
+    cardUrl() {
+      return `${this.issueLinkBase}/${this.issue.id}`;
     },
     issueId() {
       return `#${this.issue.id}`;
@@ -52,6 +70,28 @@ gl.issueBoards.IssueCardInner = Vue.extend({
     },
   },
   methods: {
+    isIndexLessThanlimit(index) {
+      return index < this.limitBeforeCounter;
+    },
+    shouldRenderAssignee(index) {
+      // Eg. maxRender is 4,
+      // Render up to all 4 assignees if there are only 4 assigness
+      // Otherwise render up to the limitBeforeCounter
+      if (this.issue.assignees.length <= this.maxRender) {
+        return index < this.maxRender;
+      }
+
+      return index < this.limitBeforeCounter;
+    },
+    assigneeUrl(assignee) {
+      return `${this.rootPath}${assignee.username}`;
+    },
+    assigneeUrlTitle(assignee) {
+      return `Assigned to ${assignee.name}`;
+    },
+    avatarUrlTitle(assignee) {
+      return `Avatar for ${assignee.name}`;
+    },
     showLabel(label) {
       if (!this.list) return true;
 
@@ -105,25 +145,39 @@ gl.issueBoards.IssueCardInner = Vue.extend({
             {{ issueId }}
           </span>
         </h4>
-        <a
-          class="card-assignee has-tooltip js-no-trigger"
-          :href="assigneeUrl"
-          :title="assigneeUrlTitle"
-          v-if="issue.assignee"
-          data-container="body"
-        >
-          <img
-            class="avatar avatar-inline s20 js-no-trigger"
-            :src="issue.assignee.avatar"
-            width="20"
-            height="20"
-            :alt="avatarUrlTitle"
-          />
-        </a>
+        <div class="card-assignee">
+          <a
+            class="has-tooltip js-no-trigger"
+            :href="assigneeUrl(assignee)"
+            :title="assigneeUrlTitle(assignee)"
+            v-for="(assignee, index) in issue.assignees"
+            v-if="shouldRenderAssignee(index)"
+            data-container="body"
+            data-placement="bottom"
+          >
+            <img
+              class="avatar avatar-inline s20"
+              :src="assignee.avatarUrl"
+              width="20"
+              height="20"
+              :alt="avatarUrlTitle(assignee)"
+            />
+          </a>
+          <span
+            class="avatar-counter has-tooltip"
+            :title="assigneeCounterTooltip"
+            v-if="shouldRenderCounter"
+          >
+           {{ assigneeCounterLabel }}
+          </span>
+        </div>
       </div>
-      <div class="card-footer" v-if="showLabelFooter">
+      <div
+        class="card-footer"
+        v-if="showLabelFooter"
+      >
         <button
-          class="label color-label has-tooltip js-no-trigger"
+          class="label color-label has-tooltip"
           v-for="label in issue.labels"
           type="button"
           v-if="showLabel(label)"
diff --git a/app/assets/javascripts/boards/components/new_list_dropdown.js b/app/assets/javascripts/boards/components/new_list_dropdown.js
index 7e3bb79af1d4a..f29b6caa1acec 100644
--- a/app/assets/javascripts/boards/components/new_list_dropdown.js
+++ b/app/assets/javascripts/boards/components/new_list_dropdown.js
@@ -52,7 +52,9 @@ gl.issueBoards.newListDropdownInit = () => {
       filterable: true,
       selectable: true,
       multiSelect: true,
-      clicked (label, $el, e) {
+      clicked (options) {
+        const { e } = options;
+        const label = options.selectedObj;
         e.preventDefault();
 
         if (!Store.findList('title', label.title)) {
diff --git a/app/assets/javascripts/boards/models/user.js b/app/assets/javascripts/boards/models/assignee.js
similarity index 61%
rename from app/assets/javascripts/boards/models/user.js
rename to app/assets/javascripts/boards/models/assignee.js
index 8e9de4d4cbb69..ee0f4e608c943 100644
--- a/app/assets/javascripts/boards/models/user.js
+++ b/app/assets/javascripts/boards/models/assignee.js
@@ -1,12 +1,12 @@
 /* eslint-disable no-unused-vars */
 
-class ListUser {
+class ListAssignee {
   constructor(user) {
     this.id = user.id;
     this.name = user.name;
     this.username = user.username;
-    this.avatar = user.avatar_url;
+    this.avatarUrl = user.avatar_url;
   }
 }
 
-window.ListUser = ListUser;
+window.ListAssignee = ListAssignee;
diff --git a/app/assets/javascripts/boards/models/issue.js b/app/assets/javascripts/boards/models/issue.js
index d6175069e370e..76ebd4c9eab2d 100644
--- a/app/assets/javascripts/boards/models/issue.js
+++ b/app/assets/javascripts/boards/models/issue.js
@@ -1,7 +1,7 @@
 /* eslint-disable no-unused-vars, space-before-function-paren, arrow-body-style, arrow-parens, comma-dangle, max-len */
 /* global ListLabel */
 /* global ListMilestone */
-/* global ListUser */
+/* global ListAssignee */
 
 import Vue from 'vue';
 
@@ -14,14 +14,10 @@ class ListIssue {
     this.dueDate = obj.due_date;
     this.subscribed = obj.subscribed;
     this.labels = [];
+    this.assignees = [];
     this.selected = false;
-    this.assignee = false;
     this.position = obj.relative_position || Infinity;
 
-    if (obj.assignee) {
-      this.assignee = new ListUser(obj.assignee);
-    }
-
     if (obj.milestone) {
       this.milestone = new ListMilestone(obj.milestone);
     }
@@ -29,6 +25,8 @@ class ListIssue {
     obj.labels.forEach((label) => {
       this.labels.push(new ListLabel(label));
     });
+
+    this.assignees = obj.assignees.map(a => new ListAssignee(a));
   }
 
   addLabel (label) {
@@ -51,6 +49,26 @@ class ListIssue {
     labels.forEach(this.removeLabel.bind(this));
   }
 
+  addAssignee (assignee) {
+    if (!this.findAssignee(assignee)) {
+      this.assignees.push(new ListAssignee(assignee));
+    }
+  }
+
+  findAssignee (findAssignee) {
+    return this.assignees.filter(assignee => assignee.id === findAssignee.id)[0];
+  }
+
+  removeAssignee (removeAssignee) {
+    if (removeAssignee) {
+      this.assignees = this.assignees.filter(assignee => assignee.id !== removeAssignee.id);
+    }
+  }
+
+  removeAllAssignees () {
+    this.assignees = [];
+  }
+
   getLists () {
     return gl.issueBoards.BoardsStore.state.lists.filter(list => list.findIssue(this.id));
   }
@@ -60,7 +78,7 @@ class ListIssue {
       issue: {
         milestone_id: this.milestone ? this.milestone.id : null,
         due_date: this.dueDate,
-        assignee_id: this.assignee ? this.assignee.id : null,
+        assignee_ids: this.assignees.length > 0 ? this.assignees.map((u) => u.id) : [0],
         label_ids: this.labels.map((label) => label.id)
       }
     };
diff --git a/app/assets/javascripts/gl_dropdown.js b/app/assets/javascripts/gl_dropdown.js
index a03f1202a6d4c..0c9eb84f0ebe6 100644
--- a/app/assets/javascripts/gl_dropdown.js
+++ b/app/assets/javascripts/gl_dropdown.js
@@ -255,7 +255,8 @@ GitLabDropdown = (function() {
               }
             };
           // Remote data
-          })(this)
+          })(this),
+          instance: this,
         });
       }
     }
@@ -269,6 +270,7 @@ GitLabDropdown = (function() {
         remote: this.options.filterRemote,
         query: this.options.data,
         keys: searchFields,
+        instance: this,
         elements: (function(_this) {
           return function() {
             selector = '.dropdown-content li:not(' + NON_SELECTABLE_CLASSES + ')';
@@ -343,21 +345,26 @@ GitLabDropdown = (function() {
       }
       this.dropdown.on("click", selector, function(e) {
         var $el, selected, selectedObj, isMarking;
-        $el = $(this);
+        $el = $(e.currentTarget);
         selected = self.rowClicked($el);
         selectedObj = selected ? selected[0] : null;
         isMarking = selected ? selected[1] : null;
-        if (self.options.clicked) {
-          self.options.clicked(selectedObj, $el, e, isMarking);
+        if (this.options.clicked) {
+          this.options.clicked.call(this, {
+            selectedObj,
+            $el,
+            e,
+            isMarking,
+          });
         }
 
         // Update label right after all modifications in dropdown has been done
-        if (self.options.toggleLabel) {
-          self.updateLabel(selectedObj, $el, self);
+        if (this.options.toggleLabel) {
+          this.updateLabel(selectedObj, $el, this);
         }
 
         $el.trigger('blur');
-      });
+      }.bind(this));
     }
   }
 
@@ -439,15 +446,34 @@ GitLabDropdown = (function() {
     }
   };
 
+  GitLabDropdown.prototype.filteredFullData = function() {
+    return this.fullData.filter(r => typeof r === 'object'
+      && !Object.prototype.hasOwnProperty.call(r, 'beforeDivider')
+      && !Object.prototype.hasOwnProperty.call(r, 'header')
+    );
+  };
+
   GitLabDropdown.prototype.opened = function(e) {
     var contentHtml;
     this.resetRows();
     this.addArrowKeyEvent();
 
+    const dropdownToggle = this.dropdown.find('.dropdown-menu-toggle');
+    const hasFilterBulkUpdate = dropdownToggle.hasClass('js-filter-bulk-update');
+    const hasMultiSelect = dropdownToggle.hasClass('js-multiselect');
+
     // Makes indeterminate items effective
-    if (this.fullData && this.dropdown.find('.dropdown-menu-toggle').hasClass('js-filter-bulk-update')) {
+    if (this.fullData && hasFilterBulkUpdate) {
       this.parseData(this.fullData);
     }
+
+    // Process the data to make sure rendered data
+    // matches the correct layout
+    if (this.fullData && hasMultiSelect && this.options.processData) {
+      const inputValue = this.filterInput.val();
+      this.options.processData.call(this.options, inputValue, this.filteredFullData(), this.parseData.bind(this));
+    }
+
     contentHtml = $('.dropdown-content', this.dropdown).html();
     if (this.remote && contentHtml === "") {
       this.remote.execute();
@@ -709,6 +735,11 @@ GitLabDropdown = (function() {
     if (this.options.inputId != null) {
       $input.attr('id', this.options.inputId);
     }
+
+    if (this.options.inputMeta) {
+      $input.attr('data-meta', selectedObject[this.options.inputMeta]);
+    }
+
     return this.dropdown.before($input);
   };
 
@@ -829,7 +860,14 @@ GitLabDropdown = (function() {
     if (instance == null) {
       instance = null;
     }
-    return $(this.el).find(".dropdown-toggle-text").text(this.options.toggleLabel(selected, el, instance));
+
+    let toggleText = this.options.toggleLabel(selected, el, instance);
+    if (this.options.updateLabel) {
+      // Option to override the dropdown label text
+      toggleText = this.options.updateLabel;
+    }
+
+    return $(this.el).find(".dropdown-toggle-text").text(toggleText);
   };
 
   GitLabDropdown.prototype.clearField = function(field, isInput) {
diff --git a/app/assets/javascripts/issuable/issuable_bundle.js b/app/assets/javascripts/issuable/issuable_bundle.js
deleted file mode 100644
index e927cc0077c5b..0000000000000
--- a/app/assets/javascripts/issuable/issuable_bundle.js
+++ /dev/null
@@ -1 +0,0 @@
-require('./time_tracking/time_tracking_bundle');
diff --git a/app/assets/javascripts/issuable/time_tracking/components/no_tracking_pane.js b/app/assets/javascripts/issuable/time_tracking/components/no_tracking_pane.js
deleted file mode 100644
index b081adf5e643b..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/no_tracking_pane.js
+++ /dev/null
@@ -1,12 +0,0 @@
-import Vue from 'vue';
-
-(() => {
-  Vue.component('time-tracking-no-tracking-pane', {
-    name: 'time-tracking-no-tracking-pane',
-    template: `
-      <div class='time-tracking-no-tracking-pane'>
-        <span class='no-value'>No estimate or time spent</span>
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/time_tracking_bundle.js b/app/assets/javascripts/issuable/time_tracking/time_tracking_bundle.js
deleted file mode 100644
index 1689a69e1ed94..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/time_tracking_bundle.js
+++ /dev/null
@@ -1,66 +0,0 @@
-import Vue from 'vue';
-import VueResource from 'vue-resource';
-
-require('./components/time_tracker');
-require('../../smart_interval');
-require('../../subbable_resource');
-
-Vue.use(VueResource);
-
-(() => {
-  /* This Vue instance represents what will become the parent instance for the
-    * sidebar. It will be responsible for managing `issuable` state and propagating
-    * changes to sidebar components. We will want to create a separate service to
-    * interface with the server at that point.
-   */
-
-  class IssuableTimeTracking {
-    constructor(issuableJSON) {
-      const parsedIssuable = JSON.parse(issuableJSON);
-      return this.initComponent(parsedIssuable);
-    }
-
-    initComponent(parsedIssuable) {
-      this.parentInstance = new Vue({
-        el: '#issuable-time-tracker',
-        data: {
-          issuable: parsedIssuable,
-        },
-        methods: {
-          fetchIssuable() {
-            return gl.IssuableResource.get.call(gl.IssuableResource, {
-              type: 'GET',
-              url: gl.IssuableResource.endpoint,
-            });
-          },
-          updateState(data) {
-            this.issuable = data;
-          },
-          subscribeToUpdates() {
-            gl.IssuableResource.subscribe(data => this.updateState(data));
-          },
-          listenForSlashCommands() {
-            $(document).on('ajax:success', '.gfm-form', (e, data) => {
-              const subscribedCommands = ['spend_time', 'time_estimate'];
-              const changedCommands = data.commands_changes
-                ? Object.keys(data.commands_changes)
-                : [];
-              if (changedCommands && _.intersection(subscribedCommands, changedCommands).length) {
-                this.fetchIssuable();
-              }
-            });
-          },
-        },
-        created() {
-          this.fetchIssuable();
-        },
-        mounted() {
-          this.subscribeToUpdates();
-          this.listenForSlashCommands();
-        },
-      });
-    }
-  }
-
-  gl.IssuableTimeTracking = IssuableTimeTracking;
-})(window.gl || (window.gl = {}));
diff --git a/app/assets/javascripts/issue_status_select.js b/app/assets/javascripts/issue_status_select.js
index b2cfd3ef2a3cb..56cb536dcde7d 100644
--- a/app/assets/javascripts/issue_status_select.js
+++ b/app/assets/javascripts/issue_status_select.js
@@ -19,8 +19,8 @@
               return label;
             };
           })(this),
-          clicked: function(item, $el, e) {
-            return e.preventDefault();
+          clicked: function(options) {
+            return options.e.preventDefault();
           },
           id: function(obj, el) {
             return $(el).data("id");
diff --git a/app/assets/javascripts/issues_bulk_assignment.js b/app/assets/javascripts/issues_bulk_assignment.js
index e0ebd36a65ce8..fee3429e2b846 100644
--- a/app/assets/javascripts/issues_bulk_assignment.js
+++ b/app/assets/javascripts/issues_bulk_assignment.js
@@ -88,7 +88,10 @@
       const formData = {
         update: {
           state_event: this.form.find('input[name="update[state_event]"]').val(),
+          // For Merge Requests
           assignee_id: this.form.find('input[name="update[assignee_id]"]').val(),
+          // For Issues
+          assignee_ids: [this.form.find('input[name="update[assignee_ids][]"]').val()],
           milestone_id: this.form.find('input[name="update[milestone_id]"]').val(),
           issuable_ids: this.form.find('input[name="update[issuable_ids]"]').val(),
           subscription_event: this.form.find('input[name="update[subscription_event]"]').val(),
diff --git a/app/assets/javascripts/labels_select.js b/app/assets/javascripts/labels_select.js
index 9a60f5464df95..ac5ce84e31b1f 100644
--- a/app/assets/javascripts/labels_select.js
+++ b/app/assets/javascripts/labels_select.js
@@ -330,7 +330,10 @@
           },
           multiSelect: $dropdown.hasClass('js-multiselect'),
           vue: $dropdown.hasClass('js-issue-board-sidebar'),
-          clicked: function(label, $el, e, isMarking) {
+          clicked: function(options) {
+            const { $el, e, isMarking } = options;
+            const label = options.selectedObj;
+
             var isIssueIndex, isMRIndex, page, boardsModel;
             var fadeOutLoader = () => {
               $loading.fadeOut();
@@ -352,7 +355,7 @@
 
             if ($dropdown.hasClass('js-filter-bulk-update')) {
               _this.enableBulkLabelDropdown();
-              _this.setDropdownData($dropdown, isMarking, this.id(label));
+              _this.setDropdownData($dropdown, isMarking, label.id);
               return;
             }
 
diff --git a/app/assets/javascripts/main.js b/app/assets/javascripts/main.js
index be3c2c9fbb102..1b0d5fc92e38b 100644
--- a/app/assets/javascripts/main.js
+++ b/app/assets/javascripts/main.js
@@ -158,7 +158,6 @@ import './single_file_diff';
 import './smart_interval';
 import './snippets_list';
 import './star';
-import './subbable_resource';
 import './subscription';
 import './subscription_select';
 import './syntax_highlight';
diff --git a/app/assets/javascripts/members.js b/app/assets/javascripts/members.js
index e3f367a11eb94..bfc4e55116834 100644
--- a/app/assets/javascripts/members.js
+++ b/app/assets/javascripts/members.js
@@ -31,8 +31,27 @@
           toggleLabel(selected, $el) {
             return $el.text();
           },
+<<<<<<< HEAD
           clicked: (selected, $link) => {
             this.formSubmit(null, $link);
+=======
+          clicked: (options) => {
+            const $link = options.$el;
+
+            if (!$link.data('revert')) {
+              this.formSubmit(null, $link);
+            } else {
+              const { $memberListItem, $toggle, $dateInput } = this.getMemberListItems($link);
+
+              $toggle.disable();
+              $dateInput.disable();
+
+              this.overrideLdap($memberListItem, $link.data('endpoint'), false).fail(() => {
+                $toggle.enable();
+                $dateInput.enable();
+              });
+            }
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
           },
         });
       });
diff --git a/app/assets/javascripts/milestone_select.js b/app/assets/javascripts/milestone_select.js
index bebd0aa357e0e..0ca7aedb5bd4b 100644
--- a/app/assets/javascripts/milestone_select.js
+++ b/app/assets/javascripts/milestone_select.js
@@ -121,7 +121,30 @@
             return $value.css('display', '');
           },
           vue: $dropdown.hasClass('js-issue-board-sidebar'),
+<<<<<<< HEAD
           clicked: function(selected, $el, e) {
+=======
+          hideRow: function(milestone) {
+            if ($('html').hasClass('issue-boards-page') && !$dropdown.hasClass('js-issue-board-sidebar') &&
+              !$dropdown.closest('.add-issues-modal').length && gl.issueBoards.BoardsStore.state.currentBoard.milestone) {
+              return milestone !== gl.issueBoards.BoardsStore.state.currentBoard.milestone.title;
+            }
+
+            return false;
+          },
+          isSelectable: function() {
+            if ($('html').hasClass('issue-boards-page') && !$dropdown.hasClass('js-issue-board-sidebar') &&
+              !$dropdown.closest('.add-issues-modal').length && gl.issueBoards.BoardsStore.state.currentBoard.milestone_id) {
+              return false;
+            }
+
+            return true;
+          },
+          clicked: function(options) {
+            const { $el, e } = options;
+            let selected = options.selectedObj;
+
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
             var data, isIssueIndex, isMRIndex, page, boardsStore;
             page = $('body').data('page');
             isIssueIndex = page === 'projects:issues:index';
diff --git a/app/assets/javascripts/namespace_select.js b/app/assets/javascripts/namespace_select.js
index b98e6121967a7..36bc1257cefd2 100644
--- a/app/assets/javascripts/namespace_select.js
+++ b/app/assets/javascripts/namespace_select.js
@@ -58,7 +58,8 @@
       });
     }
 
-    NamespaceSelect.prototype.onSelectItem = function(item, el, e) {
+    NamespaceSelect.prototype.onSelectItem = function(options) {
+      const { e } = options;
       return e.preventDefault();
     };
 
diff --git a/app/assets/javascripts/project.js b/app/assets/javascripts/project.js
index f944fcc5a58c4..738e710deb9ab 100644
--- a/app/assets/javascripts/project.js
+++ b/app/assets/javascripts/project.js
@@ -112,7 +112,8 @@ import Cookies from 'js-cookie';
           toggleLabel: function(obj, $el) {
             return $el.text().trim();
           },
-          clicked: function(selected, $el, e) {
+          clicked: function(options) {
+            const { e } = options;
             e.preventDefault();
             if ($('input[name="ref"]').length) {
               var $form = $dropdown.closest('form');
diff --git a/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js b/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
index e7fff57ff452c..1266d70f073aa 100644
--- a/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
+++ b/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
@@ -19,7 +19,10 @@
             return 'Select';
           }
         },
-        clicked(item, $el, e) {
+        clicked(opts) {
+          const { $el, e } = opts;
+          const item = opts.selectedObj;
+
           e.preventDefault();
           onSelect();
         }
diff --git a/app/assets/javascripts/protected_branches/protected_branch_dropdown.js b/app/assets/javascripts/protected_branches/protected_branch_dropdown.js
index 1d4bb8a13d677..bc6110fcd4e4f 100644
--- a/app/assets/javascripts/protected_branches/protected_branch_dropdown.js
+++ b/app/assets/javascripts/protected_branches/protected_branch_dropdown.js
@@ -35,7 +35,8 @@ class ProtectedBranchDropdown {
         return _.escape(protectedBranch.id);
       },
       onFilter: this.toggleCreateNewButton.bind(this),
-      clicked: (item, $el, e) => {
+      clicked: (options) => {
+        const { $el, e } = options;
         e.preventDefault();
         this.onSelect();
       }
diff --git a/app/assets/javascripts/sidebar/components/assignees/assignee_title.js b/app/assets/javascripts/sidebar/components/assignees/assignee_title.js
new file mode 100644
index 0000000000000..a9ad3708514b7
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/assignees/assignee_title.js
@@ -0,0 +1,41 @@
+export default {
+  name: 'AssigneeTitle',
+  props: {
+    loading: {
+      type: Boolean,
+      required: false,
+      default: false,
+    },
+    numberOfAssignees: {
+      type: Number,
+      required: true,
+    },
+    editable: {
+      type: Boolean,
+      required: true,
+    },
+  },
+  computed: {
+    assigneeTitle() {
+      const assignees = this.numberOfAssignees;
+      return assignees > 1 ? `${assignees} Assignees` : 'Assignee';
+    },
+  },
+  template: `
+    <div class="title hide-collapsed">
+      {{assigneeTitle}}
+      <i
+        v-if="loading"
+        aria-hidden="true"
+        class="fa fa-spinner fa-spin block-loading"
+      />
+      <a
+        v-if="editable"
+        class="edit-link pull-right"
+        href="#"
+      >
+        Edit
+      </a>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/assignees/assignees.js b/app/assets/javascripts/sidebar/components/assignees/assignees.js
new file mode 100644
index 0000000000000..88d7650f40ac8
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/assignees/assignees.js
@@ -0,0 +1,224 @@
+export default {
+  name: 'Assignees',
+  data() {
+    return {
+      defaultRenderCount: 5,
+      defaultMaxCounter: 99,
+      showLess: true,
+    };
+  },
+  props: {
+    rootPath: {
+      type: String,
+      required: true,
+    },
+    users: {
+      type: Array,
+      required: true,
+    },
+    editable: {
+      type: Boolean,
+      required: true,
+    },
+  },
+  computed: {
+    firstUser() {
+      return this.users[0];
+    },
+    hasMoreThanTwoAssignees() {
+      return this.users.length > 2;
+    },
+    hasMoreThanOneAssignee() {
+      return this.users.length > 1;
+    },
+    hasAssignees() {
+      return this.users.length > 0;
+    },
+    hasNoUsers() {
+      return !this.users.length;
+    },
+    hasOneUser() {
+      return this.users.length === 1;
+    },
+    renderShowMoreSection() {
+      return this.users.length > this.defaultRenderCount;
+    },
+    numberOfHiddenAssignees() {
+      return this.users.length - this.defaultRenderCount;
+    },
+    isHiddenAssignees() {
+      return this.numberOfHiddenAssignees > 0;
+    },
+    hiddenAssigneesLabel() {
+      return `+ ${this.numberOfHiddenAssignees} more`;
+    },
+    collapsedTooltipTitle() {
+      const maxRender = Math.min(this.defaultRenderCount, this.users.length);
+      const renderUsers = this.users.slice(0, maxRender);
+      const names = renderUsers.map(u => u.name);
+
+      if (this.users.length > maxRender) {
+        names.push(`+ ${this.users.length - maxRender} more`);
+      }
+
+      return names.join(', ');
+    },
+    sidebarAvatarCounter() {
+      let counter = `+${this.users.length - 1}`;
+
+      if (this.users.length > this.defaultMaxCounter) {
+        counter = `${this.defaultMaxCounter}+`;
+      }
+
+      return counter;
+    },
+  },
+  methods: {
+    assignSelf() {
+      this.$emit('assign-self');
+    },
+    toggleShowLess() {
+      this.showLess = !this.showLess;
+    },
+    renderAssignee(index) {
+      return !this.showLess || (index < this.defaultRenderCount && this.showLess);
+    },
+    avatarUrl(user) {
+      return user.avatarUrl || user.avatar_url;
+    },
+    assigneeUrl(user) {
+      return `${this.rootPath}${user.username}`;
+    },
+    assigneeAlt(user) {
+      return `${user.name}'s avatar`;
+    },
+    assigneeUsername(user) {
+      return `@${user.username}`;
+    },
+    shouldRenderCollapsedAssignee(index) {
+      const firstTwo = this.users.length <= 2 && index <= 2;
+
+      return index === 0 || firstTwo;
+    },
+  },
+  template: `
+    <div>
+      <div
+        class="sidebar-collapsed-icon sidebar-collapsed-user"
+        :class="{ 'multiple-users': hasMoreThanOneAssignee, 'has-tooltip': hasAssignees }"
+        data-container="body"
+        data-placement="left"
+        :title="collapsedTooltipTitle"
+      >
+        <i
+          v-if="hasNoUsers"
+          aria-label="No Assignee"
+          class="fa fa-user"
+        />
+        <button
+          type="button"
+          class="btn-link"
+          v-for="(user, index) in users"
+          v-if="shouldRenderCollapsedAssignee(index)"
+        >
+          <img
+            width="24"
+            class="avatar avatar-inline s24"
+            :alt="assigneeAlt(user)"
+            :src="avatarUrl(user)"
+          />
+          <span class="author">
+            {{ user.name }}
+          </span>
+        </button>
+        <button
+          v-if="hasMoreThanTwoAssignees"
+          class="btn-link"
+          type="button"
+        >
+          <span
+            class="avatar-counter sidebar-avatar-counter"
+          >
+            {{ sidebarAvatarCounter }}
+          </span>
+        </button>
+      </div>
+      <div class="value hide-collapsed">
+        <template v-if="hasNoUsers">
+          <span class="assign-yourself no-value">
+            No assignee
+            <template v-if="editable">
+             -
+              <button
+                type="button"
+                class="btn-link"
+                @click="assignSelf"
+              >
+                assign yourself
+              </button>
+            </template>
+          </span>
+        </template>
+        <template v-else-if="hasOneUser">
+          <a
+            class="author_link bold"
+            :href="assigneeUrl(firstUser)"
+          >
+            <img
+              width="32"
+              class="avatar avatar-inline s32"
+              :alt="assigneeAlt(firstUser)"
+              :src="avatarUrl(firstUser)"
+            />
+            <span class="author">
+              {{ firstUser.name }}
+            </span>
+            <span class="username">
+              {{ assigneeUsername(firstUser) }}
+            </span>
+          </a>
+        </template>
+        <template v-else>
+          <div class="user-list">
+            <div
+              class="user-item"
+              v-for="(user, index) in users"
+              v-if="renderAssignee(index)"
+            >
+              <a
+                class="user-link has-tooltip"
+                data-placement="bottom"
+                :href="assigneeUrl(user)"
+                :data-title="user.name"
+              >
+                <img
+                  width="32"
+                  class="avatar avatar-inline s32"
+                  :alt="assigneeAlt(user)"
+                  :src="avatarUrl(user)"
+                />
+              </a>
+            </div>
+          </div>
+          <div
+            v-if="renderShowMoreSection"
+            class="user-list-more"
+          >
+            <button
+              type="button"
+              class="btn-link"
+              @click="toggleShowLess"
+            >
+              <template v-if="showLess">
+                {{ hiddenAssigneesLabel }}
+              </template>
+              <template v-else>
+                - show less
+              </template>
+            </button>
+          </div>
+        </template>
+      </div>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/assignees/sidebar_assignees.js b/app/assets/javascripts/sidebar/components/assignees/sidebar_assignees.js
new file mode 100644
index 0000000000000..1488a66c695ad
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/assignees/sidebar_assignees.js
@@ -0,0 +1,84 @@
+/* global Flash */
+
+import AssigneeTitle from './assignee_title';
+import Assignees from './assignees';
+
+import Store from '../../stores/sidebar_store';
+import Mediator from '../../sidebar_mediator';
+
+import eventHub from '../../event_hub';
+
+export default {
+  name: 'SidebarAssignees',
+  data() {
+    return {
+      mediator: new Mediator(),
+      store: new Store(),
+      loading: false,
+      field: '',
+    };
+  },
+  components: {
+    'assignee-title': AssigneeTitle,
+    assignees: Assignees,
+  },
+  methods: {
+    assignSelf() {
+      // Notify gl dropdown that we are now assigning to current user
+      this.$el.parentElement.dispatchEvent(new Event('assignYourself'));
+
+      this.mediator.assignYourself();
+      this.saveAssignees();
+    },
+    saveAssignees() {
+      this.loading = true;
+
+      function setLoadingFalse() {
+        this.loading = false;
+      }
+
+      this.mediator.saveAssignees(this.field)
+        .then(setLoadingFalse.bind(this))
+        .catch(() => {
+          setLoadingFalse();
+          return new Flash('Error occurred when saving assignees');
+        });
+    },
+  },
+  created() {
+    this.removeAssignee = this.store.removeAssignee.bind(this.store);
+    this.addAssignee = this.store.addAssignee.bind(this.store);
+    this.removeAllAssignees = this.store.removeAllAssignees.bind(this.store);
+
+    // Get events from glDropdown
+    eventHub.$on('sidebar.removeAssignee', this.removeAssignee);
+    eventHub.$on('sidebar.addAssignee', this.addAssignee);
+    eventHub.$on('sidebar.removeAllAssignees', this.removeAllAssignees);
+    eventHub.$on('sidebar.saveAssignees', this.saveAssignees);
+  },
+  beforeDestroy() {
+    eventHub.$off('sidebar.removeAssignee', this.removeAssignee);
+    eventHub.$off('sidebar.addAssignee', this.addAssignee);
+    eventHub.$off('sidebar.removeAllAssignees', this.removeAllAssignees);
+    eventHub.$off('sidebar.saveAssignees', this.saveAssignees);
+  },
+  beforeMount() {
+    this.field = this.$el.dataset.field;
+  },
+  template: `
+    <div>
+      <assignee-title
+        :number-of-assignees="store.assignees.length"
+        :loading="loading"
+        :editable="store.editable"
+      />
+      <assignees
+        class="value"
+        :root-path="store.rootPath"
+        :users="store.assignees"
+        :editable="store.editable"
+        @assign-self="assignSelf"
+      />
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/collapsed_state.js b/app/assets/javascripts/sidebar/components/time_tracking/collapsed_state.js
new file mode 100644
index 0000000000000..0da265053bd0e
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/collapsed_state.js
@@ -0,0 +1,97 @@
+import stopwatchSvg from 'icons/_icon_stopwatch.svg';
+
+import '../../../lib/utils/pretty_time';
+
+export default {
+  name: 'time-tracking-collapsed-state',
+  props: {
+    showComparisonState: {
+      type: Boolean,
+      required: true,
+    },
+    showSpentOnlyState: {
+      type: Boolean,
+      required: true,
+    },
+    showEstimateOnlyState: {
+      type: Boolean,
+      required: true,
+    },
+    showNoTimeTrackingState: {
+      type: Boolean,
+      required: true,
+    },
+    timeSpentHumanReadable: {
+      type: String,
+      required: false,
+      default: '',
+    },
+    timeEstimateHumanReadable: {
+      type: String,
+      required: false,
+      default: '',
+    },
+  },
+  computed: {
+    timeSpent() {
+      return this.abbreviateTime(this.timeSpentHumanReadable);
+    },
+    timeEstimate() {
+      return this.abbreviateTime(this.timeEstimateHumanReadable);
+    },
+    divClass() {
+      if (this.showComparisonState) {
+        return 'compare';
+      } else if (this.showEstimateOnlyState) {
+        return 'estimate-only';
+      } else if (this.showSpentOnlyState) {
+        return 'spend-only';
+      } else if (this.showNoTimeTrackingState) {
+        return 'no-tracking';
+      }
+
+      return '';
+    },
+    spanClass() {
+      if (this.showComparisonState) {
+        return '';
+      } else if (this.showEstimateOnlyState || this.showSpentOnlyState) {
+        return 'bold';
+      } else if (this.showNoTimeTrackingState) {
+        return 'no-value';
+      }
+
+      return '';
+    },
+    text() {
+      if (this.showComparisonState) {
+        return `${this.timeSpent} / ${this.timeEstimate}`;
+      } else if (this.showEstimateOnlyState) {
+        return `-- / ${this.timeEstimate}`;
+      } else if (this.showSpentOnlyState) {
+        return `${this.timeSpent} / --`;
+      } else if (this.showNoTimeTrackingState) {
+        return 'None';
+      }
+
+      return '';
+    },
+  },
+  methods: {
+    abbreviateTime(timeStr) {
+      return gl.utils.prettyTime.abbreviateTime(timeStr);
+    },
+  },
+  template: `
+    <div class="sidebar-collapsed-icon">
+      ${stopwatchSvg}
+      <div class="time-tracking-collapsed-summary">
+        <div :class="divClass">
+          <span :class="spanClass">
+            {{ text }}
+          </span>
+        </div>
+      </div>
+    </div>
+    `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/comparison_pane.js b/app/assets/javascripts/sidebar/components/time_tracking/comparison_pane.js
new file mode 100644
index 0000000000000..40f5c89c5bbcf
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/comparison_pane.js
@@ -0,0 +1,98 @@
+import '../../../lib/utils/pretty_time';
+
+const prettyTime = gl.utils.prettyTime;
+
+export default {
+  name: 'time-tracking-comparison-pane',
+  props: {
+    timeSpent: {
+      type: Number,
+      required: true,
+    },
+    timeEstimate: {
+      type: Number,
+      required: true,
+    },
+    timeSpentHumanReadable: {
+      type: String,
+      required: true,
+    },
+    timeEstimateHumanReadable: {
+      type: String,
+      required: true,
+    },
+  },
+  computed: {
+    parsedRemaining() {
+      const diffSeconds = this.timeEstimate - this.timeSpent;
+      return prettyTime.parseSeconds(diffSeconds);
+    },
+    timeRemainingHumanReadable() {
+      return prettyTime.stringifyTime(this.parsedRemaining);
+    },
+    timeRemainingTooltip() {
+      const prefix = this.timeRemainingMinutes < 0 ? 'Over by' : 'Time remaining:';
+      return `${prefix} ${this.timeRemainingHumanReadable}`;
+    },
+    /* Diff values for comparison meter */
+    timeRemainingMinutes() {
+      return this.timeEstimate - this.timeSpent;
+    },
+    timeRemainingPercent() {
+      return `${Math.floor((this.timeSpent / this.timeEstimate) * 100)}%`;
+    },
+    timeRemainingStatusClass() {
+      return this.timeEstimate >= this.timeSpent ? 'within_estimate' : 'over_estimate';
+    },
+    /* Parsed time values */
+    parsedEstimate() {
+      return prettyTime.parseSeconds(this.timeEstimate);
+    },
+    parsedSpent() {
+      return prettyTime.parseSeconds(this.timeSpent);
+    },
+  },
+  template: `
+    <div class="time-tracking-comparison-pane">
+      <div
+        class="compare-meter"
+        data-toggle="tooltip"
+        data-placement="top"
+        role="timeRemainingDisplay"
+        :aria-valuenow="timeRemainingTooltip"
+        :title="timeRemainingTooltip"
+        :data-original-title="timeRemainingTooltip"
+        :class="timeRemainingStatusClass"
+      >
+        <div
+          class="meter-container"
+          role="timeSpentPercent"
+          :aria-valuenow="timeRemainingPercent"
+        >
+          <div
+            :style="{ width: timeRemainingPercent }"
+            class="meter-fill"
+          />
+        </div>
+        <div class="compare-display-container">
+          <div class="compare-display pull-left">
+            <span class="compare-label">
+              Spent
+            </span>
+            <span class="compare-value spent">
+              {{ timeSpentHumanReadable }}
+            </span>
+          </div>
+          <div class="compare-display estimated pull-right">
+            <span class="compare-label">
+              Est
+            </span>
+            <span class="compare-value">
+              {{ timeEstimateHumanReadable }}
+            </span>
+          </div>
+        </div>
+      </div>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/estimate_only_pane.js b/app/assets/javascripts/sidebar/components/time_tracking/estimate_only_pane.js
new file mode 100644
index 0000000000000..ad1b9179db017
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/estimate_only_pane.js
@@ -0,0 +1,17 @@
+export default {
+  name: 'time-tracking-estimate-only-pane',
+  props: {
+    timeEstimateHumanReadable: {
+      type: String,
+      required: true,
+    },
+  },
+  template: `
+    <div class="time-tracking-estimate-only-pane">
+      <span class="bold">
+        Estimated:
+      </span>
+      {{ timeEstimateHumanReadable }}
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/help_state.js b/app/assets/javascripts/sidebar/components/time_tracking/help_state.js
new file mode 100644
index 0000000000000..b2a77462fe0d4
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/help_state.js
@@ -0,0 +1,44 @@
+export default {
+  name: 'time-tracking-help-state',
+  props: {
+    rootPath: {
+      type: String,
+      required: true,
+    },
+  },
+  computed: {
+    href() {
+      return `${this.rootPath}help/workflow/time_tracking.md`;
+    },
+  },
+  template: `
+    <div class="time-tracking-help-state">
+      <div class="time-tracking-info">
+        <h4>
+          Track time with slash commands
+        </h4>
+        <p>
+          Slash commands can be used in the issues description and comment boxes.
+        </p>
+        <p>
+          <code>
+            /estimate
+          </code>
+          will update the estimated time with the latest command.
+        </p>
+        <p>
+          <code>
+            /spend
+          </code>
+          will update the sum of the time spent.
+        </p>
+        <a
+          class="btn btn-default learn-more-button"
+          :href="href"
+        >
+          Learn more
+        </a>
+      </div>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/no_tracking_pane.js b/app/assets/javascripts/sidebar/components/time_tracking/no_tracking_pane.js
new file mode 100644
index 0000000000000..d1dd1dcdd277d
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/no_tracking_pane.js
@@ -0,0 +1,10 @@
+export default {
+  name: 'time-tracking-no-tracking-pane',
+  template: `
+    <div class="time-tracking-no-tracking-pane">
+      <span class="no-value">
+        No estimate or time spent
+      </span>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js b/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
new file mode 100644
index 0000000000000..e2dba1fb0c2e7
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
@@ -0,0 +1,45 @@
+import '~/smart_interval';
+
+import timeTracker from './time_tracker';
+
+import Store from '../../stores/sidebar_store';
+import Mediator from '../../sidebar_mediator';
+
+export default {
+  data() {
+    return {
+      mediator: new Mediator(),
+      store: new Store(),
+    };
+  },
+  components: {
+    'issuable-time-tracker': timeTracker,
+  },
+  methods: {
+    listenForSlashCommands() {
+      $(document).on('ajax:success', '.gfm-form', (e, data) => {
+        const subscribedCommands = ['spend_time', 'time_estimate'];
+        const changedCommands = data.commands_changes
+          ? Object.keys(data.commands_changes)
+          : [];
+        if (changedCommands && _.intersection(subscribedCommands, changedCommands).length) {
+          this.mediator.fetch();
+        }
+      });
+    },
+  },
+  mounted() {
+    this.listenForSlashCommands();
+  },
+  template: `
+    <div class="block">
+      <issuable-time-tracker
+        :time_estimate="store.timeEstimate"
+        :time_spent="store.totalTimeSpent"
+        :human_time_estimate="store.humanTimeEstimate"
+        :human_time_spent="store.humanTotalTimeSpent"
+        :rootPath="store.rootPath"
+      />
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/spent_only_pane.js b/app/assets/javascripts/sidebar/components/time_tracking/spent_only_pane.js
new file mode 100644
index 0000000000000..bf9875626475e
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/spent_only_pane.js
@@ -0,0 +1,15 @@
+export default {
+  name: 'time-tracking-spent-only-pane',
+  props: {
+    timeSpentHumanReadable: {
+      type: String,
+      required: true,
+    },
+  },
+  template: `
+    <div class="time-tracking-spend-only-pane">
+      <span class="bold">Spent:</span>
+      {{ timeSpentHumanReadable }}
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/time_tracker.js b/app/assets/javascripts/sidebar/components/time_tracking/time_tracker.js
new file mode 100644
index 0000000000000..ed0d71a4f797d
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/time_tracker.js
@@ -0,0 +1,163 @@
+import timeTrackingHelpState from './help_state';
+import timeTrackingCollapsedState from './collapsed_state';
+import timeTrackingSpentOnlyPane from './spent_only_pane';
+import timeTrackingNoTrackingPane from './no_tracking_pane';
+import timeTrackingEstimateOnlyPane from './estimate_only_pane';
+import timeTrackingComparisonPane from './comparison_pane';
+
+import eventHub from '../../event_hub';
+
+export default {
+  name: 'issuable-time-tracker',
+  props: {
+    time_estimate: {
+      type: Number,
+      required: true,
+    },
+    time_spent: {
+      type: Number,
+      required: true,
+    },
+    human_time_estimate: {
+      type: String,
+      required: false,
+      default: '',
+    },
+    human_time_spent: {
+      type: String,
+      required: false,
+      default: '',
+    },
+    rootPath: {
+      type: String,
+      required: true,
+    },
+  },
+  data() {
+    return {
+      showHelp: false,
+    };
+  },
+  components: {
+    'time-tracking-collapsed-state': timeTrackingCollapsedState,
+    'time-tracking-estimate-only-pane': timeTrackingEstimateOnlyPane,
+    'time-tracking-spent-only-pane': timeTrackingSpentOnlyPane,
+    'time-tracking-no-tracking-pane': timeTrackingNoTrackingPane,
+    'time-tracking-comparison-pane': timeTrackingComparisonPane,
+    'time-tracking-help-state': timeTrackingHelpState,
+  },
+  computed: {
+    timeSpent() {
+      return this.time_spent;
+    },
+    timeEstimate() {
+      return this.time_estimate;
+    },
+    timeEstimateHumanReadable() {
+      return this.human_time_estimate;
+    },
+    timeSpentHumanReadable() {
+      return this.human_time_spent;
+    },
+    hasTimeSpent() {
+      return !!this.timeSpent;
+    },
+    hasTimeEstimate() {
+      return !!this.timeEstimate;
+    },
+    showComparisonState() {
+      return this.hasTimeEstimate && this.hasTimeSpent;
+    },
+    showEstimateOnlyState() {
+      return this.hasTimeEstimate && !this.hasTimeSpent;
+    },
+    showSpentOnlyState() {
+      return this.hasTimeSpent && !this.hasTimeEstimate;
+    },
+    showNoTimeTrackingState() {
+      return !this.hasTimeEstimate && !this.hasTimeSpent;
+    },
+    showHelpState() {
+      return !!this.showHelp;
+    },
+  },
+  methods: {
+    toggleHelpState(show) {
+      this.showHelp = show;
+    },
+    update(data) {
+      this.time_estimate = data.time_estimate;
+      this.time_spent = data.time_spent;
+      this.human_time_estimate = data.human_time_estimate;
+      this.human_time_spent = data.human_time_spent;
+    },
+  },
+  created() {
+    eventHub.$on('timeTracker:updateData', this.update);
+  },
+  template: `
+    <div
+      class="time_tracker time-tracking-component-wrap"
+      v-cloak
+    >
+      <time-tracking-collapsed-state
+        :show-comparison-state="showComparisonState"
+        :show-no-time-tracking-state="showNoTimeTrackingState"
+        :show-help-state="showHelpState"
+        :show-spent-only-state="showSpentOnlyState"
+        :show-estimate-only-state="showEstimateOnlyState"
+        :time-spent-human-readable="timeSpentHumanReadable"
+        :time-estimate-human-readable="timeEstimateHumanReadable"
+      />
+      <div class="title hide-collapsed">
+        Time tracking
+        <div
+          class="help-button pull-right"
+          v-if="!showHelpState"
+          @click="toggleHelpState(true)"
+        >
+            <i
+              class="fa fa-question-circle"
+              aria-hidden="true"
+            />
+        </div>
+        <div
+          class="close-help-button pull-right"
+          v-if="showHelpState"
+          @click="toggleHelpState(false)"
+        >
+          <i
+            class="fa fa-close"
+            aria-hidden="true"
+          />
+        </div>
+      </div>
+      <div class="time-tracking-content hide-collapsed">
+        <time-tracking-estimate-only-pane
+          v-if="showEstimateOnlyState"
+          :time-estimate-human-readable="timeEstimateHumanReadable"
+        />
+        <time-tracking-spent-only-pane
+          v-if="showSpentOnlyState"
+          :time-spent-human-readable="timeSpentHumanReadable"
+        />
+        <time-tracking-no-tracking-pane
+          v-if="showNoTimeTrackingState"
+        />
+        <time-tracking-comparison-pane
+          v-if="showComparisonState"
+          :time-estimate="timeEstimate"
+          :time-spent="timeSpent"
+          :time-spent-human-readable="timeSpentHumanReadable"
+          :time-estimate-human-readable="timeEstimateHumanReadable"
+        />
+        <transition name="help-state-toggle">
+          <time-tracking-help-state
+            v-if="showHelpState"
+            :rootPath="rootPath"
+          />
+        </transition>
+      </div>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/event_hub.js b/app/assets/javascripts/sidebar/event_hub.js
new file mode 100644
index 0000000000000..0948c2e53524a
--- /dev/null
+++ b/app/assets/javascripts/sidebar/event_hub.js
@@ -0,0 +1,3 @@
+import Vue from 'vue';
+
+export default new Vue();
diff --git a/app/assets/javascripts/sidebar/services/sidebar_service.js b/app/assets/javascripts/sidebar/services/sidebar_service.js
new file mode 100644
index 0000000000000..5a82d01dc41e7
--- /dev/null
+++ b/app/assets/javascripts/sidebar/services/sidebar_service.js
@@ -0,0 +1,28 @@
+import Vue from 'vue';
+import VueResource from 'vue-resource';
+
+Vue.use(VueResource);
+
+export default class SidebarService {
+  constructor(endpoint) {
+    if (!SidebarService.singleton) {
+      this.endpoint = endpoint;
+
+      SidebarService.singleton = this;
+    }
+
+    return SidebarService.singleton;
+  }
+
+  get() {
+    return Vue.http.get(this.endpoint);
+  }
+
+  update(key, data) {
+    return Vue.http.put(this.endpoint, {
+      [key]: data,
+    }, {
+      emulateJSON: true,
+    });
+  }
+}
diff --git a/app/assets/javascripts/sidebar/sidebar_bundle.js b/app/assets/javascripts/sidebar/sidebar_bundle.js
new file mode 100644
index 0000000000000..2ce53c2ed30d5
--- /dev/null
+++ b/app/assets/javascripts/sidebar/sidebar_bundle.js
@@ -0,0 +1,21 @@
+import Vue from 'vue';
+import sidebarTimeTracking from './components/time_tracking/sidebar_time_tracking';
+import sidebarAssignees from './components/assignees/sidebar_assignees';
+
+import Mediator from './sidebar_mediator';
+
+document.addEventListener('DOMContentLoaded', () => {
+  const mediator = new Mediator(gl.sidebarOptions);
+  mediator.fetch();
+
+  const sidebarAssigneesEl = document.querySelector('#js-vue-sidebar-assignees');
+
+  // Only create the sidebarAssignees vue app if it is found in the DOM
+  // We currently do not use sidebarAssignees for the MR page
+  if (sidebarAssigneesEl) {
+    new Vue(sidebarAssignees).$mount(sidebarAssigneesEl);
+  }
+
+  new Vue(sidebarTimeTracking).$mount('#issuable-time-tracker');
+});
+
diff --git a/app/assets/javascripts/sidebar/sidebar_mediator.js b/app/assets/javascripts/sidebar/sidebar_mediator.js
new file mode 100644
index 0000000000000..c13f3391f0d0e
--- /dev/null
+++ b/app/assets/javascripts/sidebar/sidebar_mediator.js
@@ -0,0 +1,38 @@
+/* global Flash */
+
+import Service from './services/sidebar_service';
+import Store from './stores/sidebar_store';
+
+export default class SidebarMediator {
+  constructor(options) {
+    if (!SidebarMediator.singleton) {
+      this.store = new Store(options);
+      this.service = new Service(options.endpoint);
+      SidebarMediator.singleton = this;
+    }
+
+    return SidebarMediator.singleton;
+  }
+
+  assignYourself() {
+    this.store.addAssignee(this.store.currentUser);
+  }
+
+  saveAssignees(field) {
+    const selected = this.store.assignees.map(u => u.id);
+
+    // If there are no ids, that means we have to unassign (which is id = 0)
+    // And it only accepts an array, hence [0]
+    return this.service.update(field, selected.length === 0 ? [0] : selected);
+  }
+
+  fetch() {
+    this.service.get()
+      .then((response) => {
+        const data = response.json();
+        this.store.processAssigneeData(data);
+        this.store.processTimeTrackingData(data);
+      })
+      .catch(() => new Flash('Error occured when fetching sidebar data'));
+  }
+}
diff --git a/app/assets/javascripts/sidebar/stores/sidebar_store.js b/app/assets/javascripts/sidebar/stores/sidebar_store.js
new file mode 100644
index 0000000000000..94408c4d71547
--- /dev/null
+++ b/app/assets/javascripts/sidebar/stores/sidebar_store.js
@@ -0,0 +1,52 @@
+export default class SidebarStore {
+  constructor(store) {
+    if (!SidebarStore.singleton) {
+      const { currentUser, rootPath, editable } = store;
+      this.currentUser = currentUser;
+      this.rootPath = rootPath;
+      this.editable = editable;
+      this.timeEstimate = 0;
+      this.totalTimeSpent = 0;
+      this.humanTimeEstimate = '';
+      this.humanTimeSpent = '';
+      this.assignees = [];
+
+      SidebarStore.singleton = this;
+    }
+
+    return SidebarStore.singleton;
+  }
+
+  processAssigneeData(data) {
+    if (data.assignees) {
+      this.assignees = data.assignees;
+    }
+  }
+
+  processTimeTrackingData(data) {
+    this.timeEstimate = data.time_estimate;
+    this.totalTimeSpent = data.total_time_spent;
+    this.humanTimeEstimate = data.human_time_estimate;
+    this.humanTotalTimeSpent = data.human_total_time_spent;
+  }
+
+  addAssignee(assignee) {
+    if (!this.findAssignee(assignee)) {
+      this.assignees.push(assignee);
+    }
+  }
+
+  findAssignee(findAssignee) {
+    return this.assignees.filter(assignee => assignee.id === findAssignee.id)[0];
+  }
+
+  removeAssignee(removeAssignee) {
+    if (removeAssignee) {
+      this.assignees = this.assignees.filter(assignee => assignee.id !== removeAssignee.id);
+    }
+  }
+
+  removeAllAssignees() {
+    this.assignees = [];
+  }
+}
diff --git a/app/assets/javascripts/subbable_resource.js b/app/assets/javascripts/subbable_resource.js
deleted file mode 100644
index d819160512849..0000000000000
--- a/app/assets/javascripts/subbable_resource.js
+++ /dev/null
@@ -1,51 +0,0 @@
-(() => {
-/*
-*   SubbableResource can be extended to provide a pubsub-style service for one-off REST
-*   calls. Subscribe by passing a callback or render method you will use to handle responses.
- *
-* */
-
-  class SubbableResource {
-    constructor(resourcePath) {
-      this.endpoint = resourcePath;
-
-      // TODO: Switch to axios.create
-      this.resource = $.ajax;
-      this.subscribers = [];
-    }
-
-    subscribe(callback) {
-      this.subscribers.push(callback);
-    }
-
-    publish(newResponse) {
-      const responseCopy = _.extend({}, newResponse);
-      this.subscribers.forEach((fn) => {
-        fn(responseCopy);
-      });
-      return newResponse;
-    }
-
-    get(payload) {
-      return this.resource(payload)
-        .then(data => this.publish(data));
-    }
-
-    post(payload) {
-      return this.resource(payload)
-        .then(data => this.publish(data));
-    }
-
-    put(payload) {
-      return this.resource(payload)
-        .then(data => this.publish(data));
-    }
-
-    delete(payload) {
-      return this.resource(payload)
-        .then(data => this.publish(data));
-    }
-  }
-
-  gl.SubbableResource = SubbableResource;
-})(window.gl || (window.gl = {}));
diff --git a/app/assets/javascripts/subscription_select.js b/app/assets/javascripts/subscription_select.js
index 8b25f43ffc745..0cd591c732086 100644
--- a/app/assets/javascripts/subscription_select.js
+++ b/app/assets/javascripts/subscription_select.js
@@ -19,8 +19,8 @@
               return label;
             };
           })(this),
-          clicked: function(item, $el, e) {
-            return e.preventDefault();
+          clicked: function(options) {
+            return options.e.preventDefault();
           },
           id: function(obj, el) {
             return $(el).data("id");
diff --git a/app/assets/javascripts/users_select.js b/app/assets/javascripts/users_select.js
index 68cf9ced3efbf..7828bf86f6028 100644
--- a/app/assets/javascripts/users_select.js
+++ b/app/assets/javascripts/users_select.js
@@ -1,6 +1,7 @@
 /* eslint-disable func-names, space-before-function-paren, one-var, no-var, prefer-rest-params, wrap-iife, quotes, max-len, one-var-declaration-per-line, vars-on-top, prefer-arrow-callback, consistent-return, comma-dangle, object-shorthand, no-shadow, no-unused-vars, no-else-return, no-self-compare, prefer-template, no-unused-expressions, no-lonely-if, yoda, prefer-spread, no-void, camelcase, no-param-reassign */
 /* global Issuable */
-/* global ListUser */
+
+import eventHub from './sidebar/event_hub';
 
 (function() {
   var bind = function(fn, me) { return function() { return fn.apply(me, arguments); }; },
@@ -54,42 +55,92 @@
           selectedIdDefault = (defaultNullUser && showNullUser) ? 0 : null;
           selectedId = $dropdown.data('selected') || selectedIdDefault;
 
-          var updateIssueBoardsIssue = function () {
-            $loading.removeClass('hidden').fadeIn();
-            gl.issueBoards.BoardsStore.detail.issue.update($dropdown.attr('data-issue-update'))
-              .then(function () {
-                $loading.fadeOut();
-              })
-              .catch(function () {
-                $loading.fadeOut();
-              });
+          const assignYourself = function () {
+            const unassignedSelected = $dropdown.closest('.selectbox')
+              .find(`input[name='${$dropdown.data('field-name')}'][value=0]`);
+
+            if (unassignedSelected) {
+              unassignedSelected.remove();
+            }
+
+            // Save current selected user to the DOM
+            const input = document.createElement('input');
+            input.type = 'hidden';
+            input.name = $dropdown.data('field-name');
+
+            const currentUserInfo = $dropdown.data('currentUserInfo');
+
+            if (currentUserInfo) {
+              input.value = currentUserInfo.id;
+              input.dataset.meta = currentUserInfo.name;
+            } else if (_this.currentUser) {
+              input.value = _this.currentUser.id;
+            }
+
+            $dropdown.before(input);
+          };
+
+          if ($block[0]) {
+            $block[0].addEventListener('assignYourself', assignYourself);
+          }
+
+          const getSelectedUserInputs = function() {
+            return $selectbox
+              .find(`input[name="${$dropdown.data('field-name')}"]`);
+          };
+
+          const getSelected = function() {
+            return getSelectedUserInputs()
+              .map((index, input) => parseInt(input.value, 10))
+              .get();
+          };
+
+          const getMultiSelectDropdownTitle = function(selectedUser, isSelected) {
+            const selectedUsers = getSelected()
+              .filter(u => u !== 0);
+
+            const firstUser = getSelectedUserInputs()
+              .map((index, input) => ({
+                name: input.dataset.meta,
+                value: parseInt(input.value, 10),
+              }))
+              .filter(u => u.id !== 0)
+              .get(0);
+
+            if (selectedUsers.length === 0) {
+              return 'Unassigned';
+            } else if (selectedUsers.length === 1) {
+              return firstUser.name;
+            } else if (isSelected) {
+              const otherSelected = selectedUsers.filter(s => s !== selectedUser.id);
+              return `${selectedUser.name} + ${otherSelected.length} more`;
+            } else {
+              return `${firstUser.name} + ${selectedUsers.length - 1} more`;
+            }
           };
 
           $('.assign-to-me-link').on('click', (e) => {
             e.preventDefault();
             $(e.currentTarget).hide();
-            const $input = $(`input[name="${$dropdown.data('field-name')}"]`);
-            $input.val(gon.current_user_id);
-            selectedId = $input.val();
-            $dropdown.find('.dropdown-toggle-text').text(gon.current_user_fullname).removeClass('is-default');
-          });
-
-          $block.on('click', '.js-assign-yourself', function(e) {
-            e.preventDefault();
 
-            if ($dropdown.hasClass('js-issue-board-sidebar')) {
-              gl.issueBoards.boardStoreIssueSet('assignee', new ListUser({
-                id: _this.currentUser.id,
-                username: _this.currentUser.username,
-                name: _this.currentUser.name,
-                avatar_url: _this.currentUser.avatar_url
-              }));
+            if ($dropdown.data('multiSelect')) {
+              assignYourself();
 
-              updateIssueBoardsIssue();
+              const currentUserInfo = $dropdown.data('currentUserInfo');
+              $dropdown.find('.dropdown-toggle-text').text(getMultiSelectDropdownTitle(currentUserInfo)).removeClass('is-default');
             } else {
-              return assignTo(_this.currentUser.id);
+              const $input = $(`input[name="${$dropdown.data('field-name')}"]`);
+              $input.val(gon.current_user_id);
+              selectedId = $input.val();
+              $dropdown.find('.dropdown-toggle-text').text(gon.current_user_fullname).removeClass('is-default');
             }
           });
+
+          $block.on('click', '.js-assign-yourself', (e) => {
+            e.preventDefault();
+            return assignTo(_this.currentUser.id);
+          });
+
           assignTo = function(selected) {
             var data;
             data = {};
@@ -97,6 +148,7 @@
             data[abilityName].assignee_id = selected != null ? selected : null;
             $loading.removeClass('hidden').fadeIn();
             $dropdown.trigger('loading.gl.dropdown');
+
             return $.ajax({
               type: 'PUT',
               dataType: 'json',
@@ -106,7 +158,6 @@
               var user;
               $dropdown.trigger('loaded.gl.dropdown');
               $loading.fadeOut();
-              $selectbox.hide();
               if (data.assignee) {
                 user = {
                   name: data.assignee.name,
@@ -133,51 +184,90 @@
               var isAuthorFilter;
               isAuthorFilter = $('.js-author-search');
               return _this.users(term, options, function(users) {
-                var anyUser, index, j, len, name, obj, showDivider;
-                if (term.length === 0) {
-                  showDivider = 0;
-                  if (firstUser) {
-                    // Move current user to the front of the list
-                    for (index = j = 0, len = users.length; j < len; index = (j += 1)) {
-                      obj = users[index];
-                      if (obj.username === firstUser) {
-                        users.splice(index, 1);
-                        users.unshift(obj);
-                        break;
-                      }
+                // GitLabDropdownFilter returns this.instance
+                // GitLabDropdownRemote returns this.options.instance
+                const glDropdown = this.instance || this.options.instance;
+                glDropdown.options.processData(term, users, callback);
+              }.bind(this));
+            },
+            processData: function(term, users, callback) {
+              let anyUser;
+              let index;
+              let j;
+              let len;
+              let name;
+              let obj;
+              let showDivider;
+              if (term.length === 0) {
+                showDivider = 0;
+                if (firstUser) {
+                  // Move current user to the front of the list
+                  for (index = j = 0, len = users.length; j < len; index = (j += 1)) {
+                    obj = users[index];
+                    if (obj.username === firstUser) {
+                      users.splice(index, 1);
+                      users.unshift(obj);
+                      break;
                     }
                   }
-                  if (showNullUser) {
-                    showDivider += 1;
-                    users.unshift({
-                      beforeDivider: true,
-                      name: 'Unassigned',
-                      id: 0
-                    });
-                  }
-                  if (showAnyUser) {
-                    showDivider += 1;
-                    name = showAnyUser;
-                    if (name === true) {
-                      name = 'Any User';
-                    }
-                    anyUser = {
-                      beforeDivider: true,
-                      name: name,
-                      id: null
-                    };
-                    users.unshift(anyUser);
+                }
+                if (showNullUser) {
+                  showDivider += 1;
+                  users.unshift({
+                    beforeDivider: true,
+                    name: 'Unassigned',
+                    id: 0
+                  });
+                }
+                if (showAnyUser) {
+                  showDivider += 1;
+                  name = showAnyUser;
+                  if (name === true) {
+                    name = 'Any User';
                   }
+                  anyUser = {
+                    beforeDivider: true,
+                    name: name,
+                    id: null
+                  };
+                  users.unshift(anyUser);
                 }
+
                 if (showDivider) {
-                  users.splice(showDivider, 0, "divider");
+                  users.splice(showDivider, 0, 'divider');
                 }
 
-                callback(users);
-                if (showMenuAbove) {
-                  $dropdown.data('glDropdown').positionMenuAbove();
+                if ($dropdown.hasClass('js-multiselect')) {
+                  const selected = getSelected().filter(i => i !== 0);
+
+                  if (selected.length > 0) {
+                    if ($dropdown.data('dropdown-header')) {
+                      showDivider += 1;
+                      users.splice(showDivider, 0, {
+                        header: $dropdown.data('dropdown-header'),
+                      });
+                    }
+
+                    const selectedUsers = users
+                      .filter(u => selected.indexOf(u.id) !== -1)
+                      .sort((a, b) => a.name > b.name);
+
+                    users = users.filter(u => selected.indexOf(u.id) === -1);
+
+                    selectedUsers.forEach((selectedUser) => {
+                      showDivider += 1;
+                      users.splice(showDivider, 0, selectedUser);
+                    });
+
+                    users.splice(showDivider + 1, 0, 'divider');
+                  }
                 }
-              });
+              }
+
+              callback(users);
+              if (showMenuAbove) {
+                $dropdown.data('glDropdown').positionMenuAbove();
+              }
             },
             filterable: true,
             filterRemote: true,
@@ -186,7 +276,22 @@
             },
             selectable: true,
             fieldName: $dropdown.data('field-name'),
-            toggleLabel: function(selected, el) {
+            toggleLabel: function(selected, el, glDropdown) {
+              const inputValue = glDropdown.filterInput.val();
+
+              if (this.multiSelect && inputValue === '') {
+                // Remove non-users from the fullData array
+                const users = glDropdown.filteredFullData();
+                const callback = glDropdown.parseData.bind(glDropdown);
+
+                // Update the data model
+                this.processData(inputValue, users, callback);
+              }
+
+              if (this.multiSelect) {
+                return getMultiSelectDropdownTitle(selected, $(el).hasClass('is-active'));
+              }
+
               if (selected && 'id' in selected && $(el).hasClass('is-active')) {
                 $dropdown.find('.dropdown-toggle-text').removeClass('is-default');
                 if (selected.text) {
@@ -200,15 +305,92 @@
               }
             },
             defaultLabel: defaultLabel,
-            inputId: 'issue_assignee_id',
             hidden: function(e) {
-              $selectbox.hide();
-              // display:block overrides the hide-collapse rule
-              return $value.css('display', '');
+              if ($dropdown.hasClass('js-multiselect')) {
+                eventHub.$emit('sidebar.saveAssignees');
+              }
+
+              if (!$dropdown.data('always-show-selectbox')) {
+                $selectbox.hide();
+
+                // Recalculate where .value is because vue might have changed it
+                $block = $selectbox.closest('.block');
+                $value = $block.find('.value');
+                // display:block overrides the hide-collapse rule
+                $value.css('display', '');
+              }
             },
+<<<<<<< HEAD
             vue: $dropdown.hasClass('js-issue-board-sidebar'),
             clicked: function(user, $el, e) {
               var isIssueIndex, isMRIndex, page, selected, isSelecting;
+=======
+            multiSelect: $dropdown.hasClass('js-multiselect'),
+            inputMeta: $dropdown.data('input-meta'),
+            clicked: function(options) {
+              const { $el, e, isMarking } = options;
+              const user = options.selectedObj;
+
+              if ($dropdown.hasClass('js-multiselect')) {
+                const isActive = $el.hasClass('is-active');
+                const previouslySelected = $dropdown.closest('.selectbox')
+                    .find("input[name='" + ($dropdown.data('field-name')) + "'][value!=0]");
+
+                // Enables support for limiting the number of users selected
+                // Automatically removes the first on the list if more users are selected
+                const maxSelect = $dropdown.data('max-select');
+                if (maxSelect) {
+                  const selected = getSelected();
+
+                  if (selected.length > maxSelect) {
+                    const firstSelectedId = selected[0];
+                    const firstSelected = $dropdown.closest('.selectbox')
+                      .find(`input[name='${$dropdown.data('field-name')}'][value=${firstSelectedId}]`);
+
+                    firstSelected.remove();
+                    eventHub.$emit('sidebar.removeAssignee', {
+                      id: firstSelectedId,
+                    });
+                  }
+                }
+
+                if (user.beforeDivider && user.name.toLowerCase() === 'unassigned') {
+                  // Unassigned selected
+                  previouslySelected.each((index, element) => {
+                    const id = parseInt(element.value, 10);
+                    element.remove();
+                  });
+                  eventHub.$emit('sidebar.removeAllAssignees');
+                } else if (isActive) {
+                  // user selected
+                  eventHub.$emit('sidebar.addAssignee', user);
+
+                  // Remove unassigned selection (if it was previously selected)
+                  const unassignedSelected = $dropdown.closest('.selectbox')
+                    .find("input[name='" + ($dropdown.data('field-name')) + "'][value=0]");
+
+                  if (unassignedSelected) {
+                    unassignedSelected.remove();
+                  }
+                } else {
+                  if (previouslySelected.length === 0) {
+                  // Select unassigned because there is no more selected users
+                    this.addInput($dropdown.data('field-name'), 0, {});
+                  }
+
+                  // User unselected
+                  eventHub.$emit('sidebar.removeAssignee', user);
+                }
+
+                if (getSelected().find(u => u === gon.current_user_id)) {
+                  $('.assign-to-me-link').hide();
+                } else {
+                  $('.assign-to-me-link').show();
+                }
+              }
+
+              var isIssueIndex, isMRIndex, page, selected;
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
               page = $('body').data('page');
               isIssueIndex = page === 'projects:issues:index';
               isMRIndex = (page === page && page === 'projects:merge_requests:index');
@@ -229,6 +411,7 @@
                 return Issuable.filterResults($dropdown.closest('form'));
               } else if ($dropdown.hasClass('js-filter-submit')) {
                 return $dropdown.closest('form').submit();
+<<<<<<< HEAD
               } else if ($dropdown.hasClass('js-issue-board-sidebar')) {
                 if (user.id && isSelecting) {
                   gl.issueBoards.boardStoreIssueSet('assignee', new ListUser({
@@ -243,6 +426,9 @@
 
                 updateIssueBoardsIssue();
               } else {
+=======
+              } else if (!$dropdown.hasClass('js-multiselect')) {
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
                 selected = $dropdown.closest('.selectbox').find("input[name='" + ($dropdown.data('field-name')) + "']").val();
                 return assignTo(selected);
               }
@@ -256,29 +442,54 @@
                 selectedId = parseInt($dropdown[0].dataset.selected, 10) || selectedIdDefault;
               }
               $el.find('.is-active').removeClass('is-active');
-              $el.find(`li[data-user-id="${selectedId}"] .dropdown-menu-user-link`).addClass('is-active');
+
+              function highlightSelected(id) {
+                $el.find(`li[data-user-id="${id}"] .dropdown-menu-user-link`).addClass('is-active');
+              }
+
+              if ($selectbox[0]) {
+                getSelected().forEach(selectedId => highlightSelected(selectedId));
+              } else {
+                highlightSelected(selectedId);
+              }
             },
+            updateLabel: $dropdown.data('dropdown-title'),
             renderRow: function(user) {
-              var avatar, img, listClosingTags, listWithName, listWithUserName, selected, username;
+              var avatar, img, listClosingTags, listWithName, listWithUserName, username;
               username = user.username ? "@" + user.username : "";
               avatar = user.avatar_url ? user.avatar_url : false;
-              selected = user.id === parseInt(selectedId, 10) ? "is-active" : "";
+
+              let selected = user.id === parseInt(selectedId, 10);
+
+              if (this.multiSelect) {
+                const fieldName = this.fieldName;
+                const field = $dropdown.closest('.selectbox').find("input[name='" + fieldName + "'][value='" + user.id + "']");
+
+                if (field.length) {
+                  selected = true;
+                }
+              }
+
               img = "";
               if (user.beforeDivider != null) {
-                "<li> <a href='#' class='" + selected + "'> " + user.name + " </a> </li>";
+                `<li><a href='#' class='${selected === true ? 'is-active' : ''}'>${user.name}</a></li>`;
               } else {
                 if (avatar) {
-                  img = "<img src='" + avatar + "' class='avatar avatar-inline' width='30' />";
+                  img = "<img src='" + avatar + "' class='avatar avatar-inline' width='32' />";
                 }
               }
-              // split into three parts so we can remove the username section if nessesary
-              listWithName = "<li data-user-id=" + user.id + "> <a href='#' class='dropdown-menu-user-link " + selected + "'> " + img + " <strong class='dropdown-menu-user-full-name'> " + user.name + " </strong>";
-              listWithUserName = "<span class='dropdown-menu-user-username'> " + username + " </span>";
-              listClosingTags = "</a> </li>";
-              if (username === '') {
-                listWithUserName = '';
-              }
-              return listWithName + listWithUserName + listClosingTags;
+
+              return `
+                <li data-user-id=${user.id}>
+                  <a href='#' class='dropdown-menu-user-link ${selected === true ? 'is-active' : ''}'>
+                    ${img}
+                    <strong class='dropdown-menu-user-full-name'>
+                      ${user.name}
+                    </strong>
+                    ${username ? `<span class='dropdown-menu-user-username'>${username}</span>` : ''}
+                  </a>
+                </li>
+              `;
             }
           });
         };
diff --git a/app/assets/stylesheets/framework/avatar.scss b/app/assets/stylesheets/framework/avatar.scss
index 3f5b78ed44536..91c1ebd5a7de3 100644
--- a/app/assets/stylesheets/framework/avatar.scss
+++ b/app/assets/stylesheets/framework/avatar.scss
@@ -93,3 +93,14 @@
     align-self: center;
   }
 }
+
+.avatar-counter {
+  background-color: $gray-darkest;
+  color: $white-light;
+  border: 1px solid $border-color;
+  border-radius: 1em;
+  font-family: $regular_font;
+  font-size: 9px;
+  line-height: 16px;
+  text-align: center;
+}
diff --git a/app/assets/stylesheets/framework/dropdowns.scss b/app/assets/stylesheets/framework/dropdowns.scss
index 73ded9f30d470..856989bccf11f 100644
--- a/app/assets/stylesheets/framework/dropdowns.scss
+++ b/app/assets/stylesheets/framework/dropdowns.scss
@@ -251,11 +251,9 @@
   }
 
   .dropdown-header {
-    color: $gl-text-color;
+    color: $gl-text-color-secondary;
     font-size: 13px;
-    font-weight: 600;
     line-height: 22px;
-    text-transform: capitalize;
     padding: 0 16px;
   }
 
@@ -337,8 +335,8 @@
 .dropdown-menu-user {
   .avatar {
     float: left;
-    width: 30px;
-    height: 30px;
+    width: 2 * $gl-padding;
+    height: 2 * $gl-padding;
     margin: 0 10px 0 0;
   }
 }
@@ -381,6 +379,7 @@
 .dropdown-menu-selectable {
   a {
     padding-left: 26px;
+    position: relative;
 
     &.is-indeterminate,
     &.is-active {
@@ -406,6 +405,9 @@
 
     &.is-active::before {
       content: "\f00c";
+      position: absolute;
+      top: 50%;
+      transform: translateY(-50%);
     }
   }
 }
diff --git a/app/assets/stylesheets/framework/lists.scss b/app/assets/stylesheets/framework/lists.scss
index 15dc0aa6a5288..c9a25946ffd5f 100644
--- a/app/assets/stylesheets/framework/lists.scss
+++ b/app/assets/stylesheets/framework/lists.scss
@@ -255,6 +255,7 @@ ul.controls {
       .avatar-inline {
         margin-left: 0;
         margin-right: 0;
+        margin-bottom: 0;
       }
     }
   }
diff --git a/app/assets/stylesheets/pages/boards.scss b/app/assets/stylesheets/pages/boards.scss
index 0be1c21595923..68d7ab4bf8411 100644
--- a/app/assets/stylesheets/pages/boards.scss
+++ b/app/assets/stylesheets/pages/boards.scss
@@ -207,8 +207,13 @@
     margin-bottom: 5px;
   }
 
-  &.is-active {
+  &.is-active,
+  &.is-active .card-assignee:hover a {
     background-color: $row-hover;
+
+    &:first-child:not(:only-child) {
+      box-shadow: -10px 0 10px 1px $row-hover;
+    }
   }
 
   .label {
@@ -224,7 +229,7 @@
 }
 
 .card-title {
-  margin: 0;
+  margin: 0 30px 0 0;
   font-size: 1em;
   line-height: inherit;
 
@@ -240,10 +245,69 @@
   min-height: 20px;
 
   .card-assignee {
-    margin-left: auto;
-    margin-right: 5px;
-    padding-left: 10px;
+    display: flex;
+    justify-content: flex-end;
+    position: absolute;
+    right: 15px;
     height: 20px;
+    width: 20px;
+
+    .avatar-counter {
+      display: none;
+      vertical-align: middle;
+      min-width: 20px;
+      line-height: 19px;
+      height: 20px;
+      padding-left: 2px;
+      padding-right: 2px;
+      border-radius: 2em;
+    }
+
+    img {
+      vertical-align: top;
+    }
+
+    a {
+      position: relative;
+      margin-left: -15px;
+    }
+
+    a:nth-child(1) {
+      z-index: 3;
+    }
+
+    a:nth-child(2) {
+      z-index: 2;
+    }
+
+    a:nth-child(3) {
+      z-index: 1;
+    }
+
+    a:nth-child(4) {
+      display: none;
+    }
+
+    &:hover {
+      .avatar-counter {
+        display: inline-block;
+      }
+
+      a {
+        position: static;
+        background-color: $white-light;
+        transition: background-color 0s;
+        margin-left: auto;
+
+        &:nth-child(4) {
+          display: block;
+        }
+
+        &:first-child:not(:only-child) {
+          box-shadow: -10px 0 10px 1px $white-light;
+        }
+      }
+    }
   }
 
   .avatar {
diff --git a/app/assets/stylesheets/pages/diff.scss b/app/assets/stylesheets/pages/diff.scss
index feefaad8a15c6..77f2638683a1d 100644
--- a/app/assets/stylesheets/pages/diff.scss
+++ b/app/assets/stylesheets/pages/diff.scss
@@ -570,14 +570,7 @@
 
 .diff-comments-more-count,
 .diff-notes-collapse {
-  background-color: $gray-darkest;
-  color: $white-light;
-  border: 1px solid $white-light;
-  border-radius: 1em;
-  font-family: $regular_font;
-  font-size: 9px;
-  line-height: 17px;
-  text-align: center;
+  @extend .avatar-counter;
 }
 
 .diff-notes-collapse {
diff --git a/app/assets/stylesheets/pages/issuable.scss b/app/assets/stylesheets/pages/issuable.scss
index ad6eb9f6fe040..485ea369f3d17 100644
--- a/app/assets/stylesheets/pages/issuable.scss
+++ b/app/assets/stylesheets/pages/issuable.scss
@@ -95,10 +95,15 @@
 }
 
 .right-sidebar {
-  a {
+  a,
+  .btn-link {
     color: inherit;
   }
 
+  .btn-link {
+    outline: none;
+  }
+
   .issuable-header-text {
     margin-top: 7px;
   }
@@ -215,6 +220,10 @@
       }
     }
 
+    .assign-yourself .btn-link {
+      padding-left: 0;
+    }
+
     .light {
       font-weight: normal;
     }
@@ -239,6 +248,10 @@
       margin-left: 0;
     }
 
+    .assignee .user-list .avatar {
+      margin: 0;
+    }
+
     .username {
       display: block;
       margin-top: 4px;
@@ -301,6 +314,10 @@
         margin-top: 0;
       }
 
+      .sidebar-avatar-counter {
+        padding-top: 2px;
+      }
+
       .todo-undone {
         color: $gl-link-color;
       }
@@ -309,10 +326,15 @@
         display: none;
       }
 
-      .avatar:hover {
+      .avatar:hover,
+      .avatar-counter:hover {
         border-color: $issuable-sidebar-color;
       }
 
+      .avatar-counter:hover {
+        color: $issuable-sidebar-color;
+      }
+
       .btn-clipboard {
         border: none;
         color: $issuable-sidebar-color;
@@ -322,6 +344,17 @@
           color: $gl-text-color;
         }
       }
+
+      &.multiple-users {
+        display: flex;
+        justify-content: center;
+      }
+    }
+
+    .sidebar-avatar-counter {
+      width: 24px;
+      height: 24px;
+      border-radius: 12px;
     }
 
     .sidebar-collapsed-user {
@@ -332,6 +365,37 @@
     .issuable-header-btn {
       display: none;
     }
+
+    .multiple-users {
+      height: 24px;
+      margin-bottom: 17px;
+      margin-top: 4px;
+      padding-bottom: 4px;
+
+      .btn-link {
+        padding: 0;
+        border: 0;
+
+        .avatar {
+          margin: 0;
+        }
+      }
+
+      .btn-link:first-child {
+        position: absolute;
+        left: 10px;
+        z-index: 1;
+      }
+
+      .btn-link:last-child {
+        position: absolute;
+        right: 10px;
+
+        &:hover {
+          text-decoration: none;
+        }
+      }
+    }
   }
 
   a {
@@ -380,17 +444,21 @@
 }
 
 .participants-list {
+  display: flex;
+  flex-wrap: wrap;
+  justify-content: space-between;
   margin: -5px;
 }
 
+.user-list {
+  display: flex;
+  flex-wrap: wrap;
+}
+
 .participants-author {
-  display: inline-block;
+  flex-basis: 14%;
   padding: 5px;
 
-  &:nth-of-type(7n) {
-    padding-right: 0;
-  }
-
   .author_link {
     display: block;
   }
@@ -400,13 +468,39 @@
   }
 }
 
-.participants-more {
+.user-item {
+  display: inline-block;
+  padding: 5px;
+  flex-basis: 20%;
+
+  .user-link {
+    display: inline-block;
+  }
+}
+
+.participants-more,
+.user-list-more {
   margin-top: 5px;
   margin-left: 5px;
 
-  a {
+  a,
+  .btn-link {
     color: $gl-text-color-secondary;
   }
+
+  .btn-link {
+    outline: none;
+    padding: 0;
+  }
+
+  .btn-link:hover {
+    @extend a:hover;
+    text-decoration: none;
+  }
+
+  .btn-link:focus {
+    text-decoration: none;
+  }
 }
 
 .issuable-form-padding-top {
@@ -499,6 +593,19 @@
   }
 }
 
+.issuable-list li,
+.issue-info-container .controls {
+  .avatar-counter {
+    display: inline-block;
+    vertical-align: middle;
+    min-width: 16px;
+    line-height: 14px;
+    height: 16px;
+    padding-left: 2px;
+    padding-right: 2px;
+  }
+}
+
 .time_tracker {
   padding-bottom: 0;
   border-bottom: 0;
diff --git a/app/controllers/concerns/issuable_actions.rb b/app/controllers/concerns/issuable_actions.rb
index 3ccf2a9ce33e1..b199f18da1e6d 100644
--- a/app/controllers/concerns/issuable_actions.rb
+++ b/app/controllers/concerns/issuable_actions.rb
@@ -66,6 +66,7 @@ def bulk_update_params
       :milestone_id,
       :state_event,
       :subscription_event,
+      assignee_ids: [],
       label_ids: [],
       add_label_ids: [],
       remove_label_ids: []
diff --git a/app/controllers/concerns/issuable_collections.rb b/app/controllers/concerns/issuable_collections.rb
index c8a501d73195a..6df2c06874566 100644
--- a/app/controllers/concerns/issuable_collections.rb
+++ b/app/controllers/concerns/issuable_collections.rb
@@ -43,7 +43,7 @@ def issuable_meta_data(issuable_collection, collection_type)
   end
 
   def issues_collection
-    issues_finder.execute.preload(:project, :author, :assignee, :labels, :milestone, project: :namespace)
+    issues_finder.execute.preload(:project, :author, :assignees, :labels, :milestone, project: :namespace)
   end
 
   def merge_requests_collection
diff --git a/app/controllers/projects/boards/issues_controller.rb b/app/controllers/projects/boards/issues_controller.rb
index 28c9646910d2c..da9b789d6171f 100644
--- a/app/controllers/projects/boards/issues_controller.rb
+++ b/app/controllers/projects/boards/issues_controller.rb
@@ -82,7 +82,7 @@ def serialize_as_json(resource)
           labels: true,
           only: [:id, :iid, :title, :confidential, :due_date, :relative_position],
           include: {
-            assignee: { only: [:id, :name, :username], methods: [:avatar_url] },
+            assignees: { only: [:id, :name, :username], methods: [:avatar_url] },
             milestone: { only: [:id, :title] }
           },
           user: current_user
diff --git a/app/controllers/projects/issues_controller.rb b/app/controllers/projects/issues_controller.rb
index af9157bfbb598..2cb38fd953dfc 100644
--- a/app/controllers/projects/issues_controller.rb
+++ b/app/controllers/projects/issues_controller.rb
@@ -67,7 +67,7 @@ def index
 
   def new
     params[:issue] ||= ActionController::Parameters.new(
-      assignee_id: ""
+      assignee_ids: ""
     )
     build_params = issue_params.merge(
       merge_request_to_resolve_discussions_of: params[:merge_request_to_resolve_discussions_of],
@@ -150,7 +150,7 @@ def update
         if @issue.valid?
           render json: @issue.to_json(methods: [:task_status, :task_status_short],
                                       include: { milestone: {},
-                                                 assignee: { only: [:name, :username], methods: [:avatar_url] },
+                                                 assignees: { only: [:id, :name, :username], methods: [:avatar_url] },
                                                  labels: { methods: :text_color } })
         else
           render json: { errors: @issue.errors.full_messages }, status: :unprocessable_entity
@@ -275,7 +275,7 @@ def redirect_old
   def issue_params
     params.require(:issue).permit(
       :title, :assignee_id, :position, :description, :confidential,
-      :milestone_id, :due_date, :state_event, :task_num, :lock_version, label_ids: []
+      :milestone_id, :due_date, :state_event, :task_num, :lock_version, label_ids: [], assignee_ids: [],
     )
   end
 
diff --git a/app/finders/issuable_finder.rb b/app/finders/issuable_finder.rb
index 4cc42b88a2a6a..957ad87585837 100644
--- a/app/finders/issuable_finder.rb
+++ b/app/finders/issuable_finder.rb
@@ -231,7 +231,7 @@ def by_scope(items)
     when 'created-by-me', 'authored'
       items.where(author_id: current_user.id)
     when 'assigned-to-me'
-      items.where(assignee_id: current_user.id)
+      items.assigned_to(current_user)
     else
       items
     end
diff --git a/app/finders/issues_finder.rb b/app/finders/issues_finder.rb
index 76715e5970dac..b4c074bc69c94 100644
--- a/app/finders/issues_finder.rb
+++ b/app/finders/issues_finder.rb
@@ -26,17 +26,28 @@ def init_collection
     IssuesFinder.not_restricted_by_confidentiality(current_user)
   end
 
+  def by_assignee(items)
+    if assignee
+      items.assigned_to(assignee)
+    elsif no_assignee?
+      items.unassigned
+    elsif assignee_id? || assignee_username? # assignee not found
+      items.none
+    else
+      items
+    end
+  end
+
   def self.not_restricted_by_confidentiality(user)
-    return Issue.where('issues.confidential IS NULL OR issues.confidential IS FALSE') if user.blank?
+    return Issue.where('issues.confidential IS NOT TRUE') if user.blank?
 
     return Issue.all if user.admin?
 
     Issue.where('
-      issues.confidential IS NULL
-      OR issues.confidential IS FALSE
+      issues.confidential IS NOT TRUE
       OR (issues.confidential = TRUE
         AND (issues.author_id = :user_id
-          OR issues.assignee_id = :user_id
+          OR EXISTS (SELECT TRUE FROM issue_assignees WHERE user_id = :user_id AND issue_id = issues.id)
           OR issues.project_id IN(:project_ids)))',
       user_id: user.id,
       project_ids: user.authorized_projects(Gitlab::Access::REPORTER).select(:id))
diff --git a/app/helpers/form_helper.rb b/app/helpers/form_helper.rb
index 1182939f65695..639a720b024f7 100644
--- a/app/helpers/form_helper.rb
+++ b/app/helpers/form_helper.rb
@@ -15,4 +15,37 @@ def form_errors(model)
         end
     end
   end
+
+  def issue_dropdown_options(issuable, has_multiple_assignees = true)
+    options = {
+      toggle_class: 'js-user-search js-assignee-search',
+      title: 'Select assignee',
+      filter: true,
+      dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-assignee',
+      placeholder: 'Search users',
+      data: {
+        first_user: current_user&.username,
+        null_user: true,
+        current_user: true,
+        project_id: issuable.project.try(:id),
+        field_name: "#{issuable.class.model_name.param_key}[assignee_ids][]",
+        default_label: 'Assignee',
+        'max-select': 1,
+        'dropdown-header': 'Assignee',
+      }
+    }
+
+    if has_multiple_assignees
+      options[:toggle_class] += ' js-multiselect js-save-user-data'
+      options[:title] = 'Select assignee(s)'
+      options[:data][:multi_select] = true
+      options[:data][:'input-meta'] = 'name'
+      options[:data][:'always-show-selectbox'] = true
+      options[:data][:current_user_info] = current_user.to_json(only: [:id, :name])
+      options[:data][:'dropdown-header'] = 'Assignee(s)'
+      options[:data].delete(:'max-select')
+    end
+
+    options
+  end
 end
diff --git a/app/helpers/issuables_helper.rb b/app/helpers/issuables_helper.rb
index 0b13dbf5f8d26..7656929efe7ad 100644
--- a/app/helpers/issuables_helper.rb
+++ b/app/helpers/issuables_helper.rb
@@ -63,6 +63,16 @@ def template_dropdown_tag(issuable, &block)
     end
   end
 
+  def users_dropdown_label(selected_users)
+    if selected_users.length == 0
+      "Unassigned"
+    elsif selected_users.length == 1
+      selected_users[0].name
+    else
+      "#{selected_users[0].name} + #{selected_users.length - 1} more"
+    end
+  end
+
   def user_dropdown_label(user_id, default_label)
     return default_label if user_id.nil?
     return "Unassigned" if user_id == "0"
diff --git a/app/mailers/emails/issues.rb b/app/mailers/emails/issues.rb
index d64e48f774b92..0f84784129511 100644
--- a/app/mailers/emails/issues.rb
+++ b/app/mailers/emails/issues.rb
@@ -11,10 +11,12 @@ def new_mention_in_issue_email(recipient_id, issue_id, updated_by_user_id)
       mail_answer_thread(@issue, issue_thread_options(updated_by_user_id, recipient_id))
     end
 
-    def reassigned_issue_email(recipient_id, issue_id, previous_assignee_id, updated_by_user_id)
+    def reassigned_issue_email(recipient_id, issue_id, previous_assignee_ids, updated_by_user_id)
       setup_issue_mail(issue_id, recipient_id)
 
-      @previous_assignee = User.find_by(id: previous_assignee_id) if previous_assignee_id
+      @previous_assignees = []
+      @previous_assignees = User.where(id: previous_assignee_ids) if previous_assignee_ids.any?
+
       mail_answer_thread(@issue, issue_thread_options(updated_by_user_id, recipient_id))
     end
 
diff --git a/app/models/concerns/issuable.rb b/app/models/concerns/issuable.rb
index 26dbf4d95708f..edf4e9e5d780f 100644
--- a/app/models/concerns/issuable.rb
+++ b/app/models/concerns/issuable.rb
@@ -26,7 +26,6 @@ module Issuable
     cache_markdown_field :description, issuable_state_filter_enabled: true
 
     belongs_to :author, class_name: "User"
-    belongs_to :assignee, class_name: "User"
     belongs_to :updated_by, class_name: "User"
     belongs_to :milestone
     has_many :notes, as: :noteable, inverse_of: :noteable, dependent: :destroy do
@@ -65,11 +64,8 @@ def award_emojis_loaded?
     validates :title, presence: true, length: { maximum: 255 }
 
     scope :authored, ->(user) { where(author_id: user) }
-    scope :assigned_to, ->(u) { where(assignee_id: u.id)}
     scope :recent, -> { reorder(id: :desc) }
     scope :order_position_asc, -> { reorder(position: :asc) }
-    scope :assigned, -> { where("assignee_id IS NOT NULL") }
-    scope :unassigned, -> { where("assignee_id IS NULL") }
     scope :of_projects, ->(ids) { where(project_id: ids) }
     scope :of_milestones, ->(ids) { where(milestone_id: ids) }
     scope :with_milestone, ->(title) { left_joins_milestones.where(milestones: { title: title }) }
@@ -92,7 +88,6 @@ def award_emojis_loaded?
     attr_mentionable :description
 
     participant :author
-    participant :assignee
     participant :notes_with_associations
 
     strip_attributes :title
@@ -102,13 +97,6 @@ def award_emojis_loaded?
     after_save :update_assignee_cache_counts, if: :assignee_id_changed?
     after_save :record_metrics, unless: :imported?
 
-    def update_assignee_cache_counts
-      # make sure we flush the cache for both the old *and* new assignees(if they exist)
-      previous_assignee = User.find_by_id(assignee_id_was) if assignee_id_was
-      previous_assignee&.update_cache_counts
-      assignee&.update_cache_counts
-    end
-
     # We want to use optimistic lock for cases when only title or description are involved
     # http://api.rubyonrails.org/classes/ActiveRecord/Locking/Optimistic.html
     def locking_enabled?
@@ -237,10 +225,6 @@ def new?
     today? && created_at == updated_at
   end
 
-  def is_being_reassigned?
-    assignee_id_changed?
-  end
-
   def open?
     opened? || reopened?
   end
@@ -269,7 +253,11 @@ def to_hook_data(user)
       # DEPRECATED
       repository: project.hook_attrs.slice(:name, :url, :description, :homepage)
     }
-    hook_data[:assignee] = assignee.hook_attrs if assignee
+    if self.is_a?(Issue)
+      hook_data[:assignees] = assignees.map(&:hook_attrs) if assignees.any?
+    else
+      hook_data[:assignee] = assignee.hook_attrs if assignee
+    end
 
     hook_data
   end
@@ -331,11 +319,6 @@ def can_move?(*)
     false
   end
 
-  def assignee_or_author?(user)
-    # We're comparing IDs here so we don't need to load any associations.
-    author_id == user.id || assignee_id == user.id
-  end
-
   def record_metrics
     metrics = self.metrics || create_metrics
     metrics.record!
diff --git a/app/models/concerns/milestoneish.rb b/app/models/concerns/milestoneish.rb
index f449229864d44..a3472af5c55c9 100644
--- a/app/models/concerns/milestoneish.rb
+++ b/app/models/concerns/milestoneish.rb
@@ -40,7 +40,7 @@ def elapsed_days
   def issues_visible_to_user(user)
     memoize_per_user(user, :issues_visible_to_user) do
       IssuesFinder.new(user, issues_finder_params)
-        .execute.where(milestone_id: milestoneish_ids)
+        .execute.includes(:assignees).where(milestone_id: milestoneish_ids)
     end
   end
 
diff --git a/app/models/global_milestone.rb b/app/models/global_milestone.rb
index 0afbca2cb325e..538615130a762 100644
--- a/app/models/global_milestone.rb
+++ b/app/models/global_milestone.rb
@@ -36,7 +36,7 @@ def self.states_count(projects)
     closed = count_by_state(milestones_by_state_and_title, 'closed')
     all = milestones_by_state_and_title.map { |(_, title), _| title }.uniq.count
 
-    { 
+    {
       opened: opened,
       closed: closed,
       all: all
@@ -86,7 +86,7 @@ def closed?
   end
 
   def issues
-    @issues ||= Issue.of_milestones(milestoneish_ids).includes(:project, :assignee, :labels)
+    @issues ||= Issue.of_milestones(milestoneish_ids).includes(:project, :assignees, :labels)
   end
 
   def merge_requests
@@ -94,7 +94,7 @@ def merge_requests
   end
 
   def participants
-    @participants ||= milestones.includes(:participants).map(&:participants).flatten.compact.uniq
+    @participants ||= milestones.map(&:participants).flatten.uniq
   end
 
   def labels
diff --git a/app/models/issue.rb b/app/models/issue.rb
index 78bde6820da97..27e3ed9bc7f54 100644
--- a/app/models/issue.rb
+++ b/app/models/issue.rb
@@ -24,10 +24,17 @@ class Issue < ActiveRecord::Base
 
   has_many :merge_requests_closing_issues, class_name: 'MergeRequestsClosingIssues', dependent: :delete_all
 
+  has_many :issue_assignees
+  has_many :assignees, class_name: "User", through: :issue_assignees
+
   validates :project, presence: true
 
   scope :in_projects, ->(project_ids) { where(project_id: project_ids) }
 
+  scope :assigned, -> { where('EXISTS (SELECT TRUE FROM issue_assignees WHERE issue_id = issues.id)') }
+  scope :unassigned, -> { where('NOT EXISTS (SELECT TRUE FROM issue_assignees WHERE issue_id = issues.id)') }
+  scope :assigned_to, ->(u) { where('EXISTS (SELECT TRUE FROM issue_assignees WHERE user_id = ? AND issue_id = issues.id)', u.id)}
+
   scope :without_due_date, -> { where(due_date: nil) }
   scope :due_before, ->(date) { where('issues.due_date < ?', date) }
   scope :due_between, ->(from_date, to_date) { where('issues.due_date >= ?', from_date).where('issues.due_date <= ?', to_date) }
@@ -37,13 +44,15 @@ class Issue < ActiveRecord::Base
 
   scope :created_after, -> (datetime) { where("created_at >= ?", datetime) }
 
-  scope :include_associations, -> { includes(:assignee, :labels, project: :namespace) }
+  scope :include_associations, -> { includes(:labels, project: :namespace) }
 
   after_save :expire_etag_cache
 
   attr_spammable :title, spam_title: true
   attr_spammable :description, spam_description: true
 
+  participant :assignees
+
   state_machine :state, initial: :opened do
     event :close do
       transition [:reopened, :opened] => :closed
@@ -63,10 +72,14 @@ class Issue < ActiveRecord::Base
   end
 
   def hook_attrs
+    assignee_ids = self.assignee_ids
+
     attrs = {
       total_time_spent: total_time_spent,
       human_total_time_spent: human_total_time_spent,
-      human_time_estimate: human_time_estimate
+      human_time_estimate: human_time_estimate,
+      assignee_ids: assignee_ids,
+      assignee_id: assignee_ids.first # This key is deprecated
     }
 
     attributes.merge!(attrs)
@@ -114,6 +127,22 @@ def self.order_by_position_and_priority
               "id DESC")
   end
 
+  # Returns a Hash of attributes to be used for Twitter card metadata
+  def card_attributes
+    {
+      'Author'   => author.try(:name),
+      'Assignee' => assignee_list
+    }
+  end
+
+  def assignee_or_author?(user)
+    author_id == user.id || assignees.exists?(user.id)
+  end
+
+  def assignee_list
+    assignees.map(&:name).to_sentence
+  end
+
   # `from` argument can be a Namespace or Project.
   def to_reference(from = nil, full: false)
     reference = "#{self.class.reference_prefix}#{iid}"
@@ -248,7 +277,7 @@ def readable_by?(user)
       true
     elsif confidential?
       author == user ||
-        assignee == user ||
+        assignees.include?(user) ||
         project.team.member?(user, Gitlab::Access::REPORTER)
     else
       project.public? ||
diff --git a/app/models/issue_assignee.rb b/app/models/issue_assignee.rb
new file mode 100644
index 0000000000000..b94b55bb1d1ec
--- /dev/null
+++ b/app/models/issue_assignee.rb
@@ -0,0 +1,29 @@
+class IssueAssignee < ActiveRecord::Base
+  extend Gitlab::CurrentSettings
+
+  belongs_to :issue
+  belongs_to :assignee, class_name: "User", foreign_key: :user_id
+
+  after_create :update_assignee_cache_counts
+  after_destroy :update_assignee_cache_counts
+
+  # EE-specific
+  after_create :update_elasticsearch_index
+  after_destroy :update_elasticsearch_index
+  # EE-specific
+
+  def update_assignee_cache_counts
+    assignee&.update_cache_counts
+  end
+
+  def update_elasticsearch_index
+    if current_application_settings.elasticsearch_indexing?
+      ElasticIndexerWorker.perform_async(
+        :update,
+        'Issue',
+        issue.id,
+        changed_fields: ['assignee_ids']
+      )
+    end
+  end
+end
diff --git a/app/models/merge_request.rb b/app/models/merge_request.rb
index 12c5481cd6d24..35231bab12ee1 100644
--- a/app/models/merge_request.rb
+++ b/app/models/merge_request.rb
@@ -17,6 +17,8 @@ class MergeRequest < ActiveRecord::Base
 
   has_many :merge_requests_closing_issues, class_name: 'MergeRequestsClosingIssues', dependent: :delete_all
 
+  belongs_to :assignee, class_name: "User"
+
   serialize :merge_params, Hash
 
   after_create :ensure_merge_request_diff, unless: :importing?
@@ -114,8 +116,14 @@ class MergeRequest < ActiveRecord::Base
 
   scope :join_project, -> { joins(:target_project) }
   scope :references_project, -> { references(:target_project) }
+  scope :assigned, -> { where("assignee_id IS NOT NULL") }
+  scope :unassigned, -> { where("assignee_id IS NULL") }
+  scope :assigned_to, ->(u) { where(assignee_id: u.id)}
+
+  participant :assignee
 
   after_save :keep_around_commit
+  after_save :update_assignee_cache_counts, if: :assignee_id_changed?
 
   def self.reference_prefix
     '!'
@@ -177,6 +185,30 @@ def self.wip_title(title)
     work_in_progress?(title) ? title : "WIP: #{title}"
   end
 
+  def update_assignee_cache_counts
+    # make sure we flush the cache for both the old *and* new assignees(if they exist)
+    previous_assignee = User.find_by_id(assignee_id_was) if assignee_id_was
+    previous_assignee&.update_cache_counts
+    assignee&.update_cache_counts
+  end
+
+  # Returns a Hash of attributes to be used for Twitter card metadata
+  def card_attributes
+    {
+      'Author'   => author.try(:name),
+      'Assignee' => assignee.try(:name)
+    }
+  end
+
+  # This method is needed for compatibility with issues to not mess view and other code
+  def assignees
+    Array(assignee)
+  end
+
+  def assignee_or_author?(user)
+    author_id == user.id || assignee_id == user.id
+  end
+
   # `from` argument can be a Namespace or Project.
   def to_reference(from = nil, full: false)
     reference = "#{self.class.reference_prefix}#{iid}"
diff --git a/app/models/milestone.rb b/app/models/milestone.rb
index 652b15519285a..c06bfe0ccdd77 100644
--- a/app/models/milestone.rb
+++ b/app/models/milestone.rb
@@ -21,7 +21,6 @@ class Milestone < ActiveRecord::Base
   has_many :issues
   has_many :labels, -> { distinct.reorder('labels.title') },  through: :issues
   has_many :merge_requests
-  has_many :participants, -> { distinct.reorder('users.name') }, through: :issues, source: :assignee
   has_many :events, as: :target, dependent: :destroy
 
   scope :active, -> { with_state(:active) }
@@ -107,6 +106,10 @@ def self.upcoming_ids_by_projects(projects)
     end
   end
 
+  def participants
+    User.joins(assigned_issues: :milestone).where("milestones.id = ?", id)
+  end
+
   def self.sort(method)
     case method.to_s
     when 'due_date_asc'
diff --git a/app/models/user.rb b/app/models/user.rb
index 2b7ebe6c1a71f..a3126cbb644a4 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -89,6 +89,7 @@ class User < ActiveRecord::Base
   has_many :subscriptions,            dependent: :destroy
   has_many :recent_events, -> { order "id DESC" }, foreign_key: :author_id,   class_name: "Event"
   has_many :oauth_applications, class_name: 'Doorkeeper::Application', as: :owner, dependent: :destroy
+
   has_one  :abuse_report,             dependent: :destroy, foreign_key: :user_id
   has_many :reported_abuse_reports,   dependent: :destroy, foreign_key: :reporter_id, class_name: "AbuseReport"
   has_many :spam_logs,                dependent: :destroy
@@ -99,6 +100,10 @@ class User < ActiveRecord::Base
   has_many :award_emoji,              dependent: :destroy
   has_many :triggers,                 dependent: :destroy, class_name: 'Ci::Trigger', foreign_key: :owner_id
 
+  has_many :issue_assignees
+  has_many :assigned_issues, class_name: "Issue", through: :issue_assignees, source: :issue
+  has_many :assigned_merge_requests,  dependent: :nullify, foreign_key: :assignee_id, class_name: "MergeRequest"
+
   # Issues that a user owns are expected to be moved to the "ghost" user before
   # the user is destroyed. If the user owns any issues during deletion, this
   # should be treated as an exceptional condition.
diff --git a/app/serializers/issuable_entity.rb b/app/serializers/issuable_entity.rb
index 29aecb50849fc..65b204d4dd27b 100644
--- a/app/serializers/issuable_entity.rb
+++ b/app/serializers/issuable_entity.rb
@@ -1,7 +1,6 @@
 class IssuableEntity < Grape::Entity
   expose :id
   expose :iid
-  expose :assignee_id
   expose :author_id
   expose :description
   expose :lock_version
diff --git a/app/serializers/issue_entity.rb b/app/serializers/issue_entity.rb
index 6429159ebe152..bc4f68710b20d 100644
--- a/app/serializers/issue_entity.rb
+++ b/app/serializers/issue_entity.rb
@@ -1,6 +1,7 @@
 class IssueEntity < IssuableEntity
   expose :branch_name
   expose :confidential
+  expose :assignees, using: API::Entities::UserBasic
   expose :due_date
   expose :moved_to_id
   expose :project_id
diff --git a/app/serializers/merge_request_entity.rb b/app/serializers/merge_request_entity.rb
index 5f80ab397a9ba..453ba52b892a0 100644
--- a/app/serializers/merge_request_entity.rb
+++ b/app/serializers/merge_request_entity.rb
@@ -1,4 +1,5 @@
 class MergeRequestEntity < IssuableEntity
+  expose :assignee_id
   expose :in_progress_merge_commit_sha
   expose :locked_at
   expose :merge_commit_sha
diff --git a/app/services/issuable/bulk_update_service.rb b/app/services/issuable/bulk_update_service.rb
index 60891cbb255fb..40ff9b8b8679a 100644
--- a/app/services/issuable/bulk_update_service.rb
+++ b/app/services/issuable/bulk_update_service.rb
@@ -7,10 +7,14 @@ def execute(type)
       ids = params.delete(:issuable_ids).split(",")
       items = model_class.where(id: ids)
 
-      %i(state_event milestone_id assignee_id add_label_ids remove_label_ids subscription_event).each do |key|
+      %i(state_event milestone_id assignee_id assignee_ids add_label_ids remove_label_ids subscription_event).each do |key|
         params.delete(key) unless params[key].present?
       end
 
+      if params[:assignee_ids] == [IssuableFinder::NONE.to_s]
+        params[:assignee_ids] = []
+      end
+
       items.each do |issuable|
         next unless can?(current_user, :"update_#{type}", issuable)
 
diff --git a/app/services/issuable_base_service.rb b/app/services/issuable_base_service.rb
index b071a3984811e..6c2777a8d2d2e 100644
--- a/app/services/issuable_base_service.rb
+++ b/app/services/issuable_base_service.rb
@@ -1,11 +1,6 @@
 class IssuableBaseService < BaseService
   private
 
-  def create_assignee_note(issuable)
-    SystemNoteService.change_assignee(
-      issuable, issuable.project, current_user, issuable.assignee)
-  end
-
   def create_milestone_note(issuable)
     SystemNoteService.change_milestone(
       issuable, issuable.project, current_user, issuable.milestone)
@@ -53,6 +48,7 @@ def filter_params(issuable)
       params.delete(:add_label_ids)
       params.delete(:remove_label_ids)
       params.delete(:label_ids)
+      params.delete(:assignee_ids)
       params.delete(:assignee_id)
       params.delete(:due_date)
     end
@@ -77,7 +73,7 @@ def filter_assignee(issuable)
   def assignee_can_read?(issuable, assignee_id)
     new_assignee = User.find_by_id(assignee_id)
 
-    return false unless new_assignee.present?
+    return false unless new_assignee
 
     ability_name = :"read_#{issuable.to_ability_name}"
     resource     = issuable.persisted? ? issuable : project
@@ -207,6 +203,7 @@ def update(issuable)
     filter_params(issuable)
     old_labels = issuable.labels.to_a
     old_mentioned_users = issuable.mentioned_users.to_a
+    old_assignees = issuable.assignees.to_a
 
     label_ids = process_label_ids(params, existing_label_ids: issuable.label_ids)
     params[:label_ids] = label_ids if labels_changing?(issuable.label_ids, label_ids)
@@ -222,7 +219,13 @@ def update(issuable)
           handle_common_system_notes(issuable, old_labels: old_labels)
         end
 
-        handle_changes(issuable, old_labels: old_labels, old_mentioned_users: old_mentioned_users)
+        handle_changes(
+          issuable,
+          old_labels: old_labels,
+          old_mentioned_users: old_mentioned_users,
+          old_assignees: old_assignees
+        )
+
         after_update(issuable)
         issuable.create_new_cross_references!(current_user)
         execute_hooks(issuable, 'update')
@@ -272,7 +275,7 @@ def toggle_award(issuable)
     end
   end
 
-  def has_changes?(issuable, old_labels: [])
+  def has_changes?(issuable, old_labels: [], old_assignees: [])
     valid_attrs = [:title, :description, :assignee_id, :milestone_id, :target_branch]
 
     attrs_changed = valid_attrs.any? do |attr|
@@ -281,7 +284,9 @@ def has_changes?(issuable, old_labels: [])
 
     labels_changed = issuable.labels != old_labels
 
-    attrs_changed || labels_changed
+    assignees_changed = issuable.assignees != old_assignees
+
+    attrs_changed || labels_changed || assignees_changed
   end
 
   def handle_common_system_notes(issuable, old_labels: [])
diff --git a/app/services/issues/base_service.rb b/app/services/issues/base_service.rb
index ee1b40db71891..eedbfa724ff66 100644
--- a/app/services/issues/base_service.rb
+++ b/app/services/issues/base_service.rb
@@ -9,11 +9,30 @@ def hook_data(issue, action)
 
     private
 
+    def create_assignee_note(issue, old_assignees)
+      SystemNoteService.change_issue_assignees(
+        issue, issue.project, current_user, old_assignees)
+    end
+
     def execute_hooks(issue, action = 'open')
       issue_data  = hook_data(issue, action)
       hooks_scope = issue.confidential? ? :confidential_issue_hooks : :issue_hooks
       issue.project.execute_hooks(issue_data, hooks_scope)
       issue.project.execute_services(issue_data, hooks_scope)
     end
+
+    def filter_assignee(issuable)
+      return if params[:assignee_ids].blank?
+
+      assignee_ids = params[:assignee_ids].select { |assignee_id| assignee_can_read?(issuable, assignee_id) }
+
+      if params[:assignee_ids].map(&:to_s) == [IssuableFinder::NONE]
+        params[:assignee_ids] = []
+      elsif assignee_ids.any?
+        params[:assignee_ids] = assignee_ids
+      else
+        params.delete(:assignee_ids)
+      end
+    end
   end
 end
diff --git a/app/services/issues/update_service.rb b/app/services/issues/update_service.rb
index b7fe5cb168b97..cd9f9a4a16e02 100644
--- a/app/services/issues/update_service.rb
+++ b/app/services/issues/update_service.rb
@@ -12,8 +12,12 @@ def before_update(issue)
       spam_check(issue, current_user)
     end
 
-    def handle_changes(issue, old_labels: [], old_mentioned_users: [])
-      if has_changes?(issue, old_labels: old_labels)
+    def handle_changes(issue, options)
+      old_labels = options[:old_labels] || []
+      old_mentioned_users = options[:old_mentioned_users] || []
+      old_assignees = options[:old_assignees] || []
+
+      if has_changes?(issue, old_labels: old_labels, old_assignees: old_assignees)
         todo_service.mark_pending_todos_as_done(issue, current_user)
       end
 
@@ -26,9 +30,9 @@ def handle_changes(issue, old_labels: [], old_mentioned_users: [])
         create_milestone_note(issue)
       end
 
-      if issue.previous_changes.include?('assignee_id')
-        create_assignee_note(issue)
-        notification_service.reassigned_issue(issue, current_user)
+      if issue.assignees != old_assignees
+        create_assignee_note(issue, old_assignees)
+        notification_service.reassigned_issue(issue, current_user, old_assignees)
         todo_service.reassigned_issue(issue, current_user)
       end
 
diff --git a/app/services/merge_requests/assign_issues_service.rb b/app/services/merge_requests/assign_issues_service.rb
index 066efa1acc3dd..8c6c484102039 100644
--- a/app/services/merge_requests/assign_issues_service.rb
+++ b/app/services/merge_requests/assign_issues_service.rb
@@ -4,7 +4,7 @@ def assignable_issues
       @assignable_issues ||= begin
         if current_user == merge_request.author
           closes_issues.select do |issue|
-            !issue.is_a?(ExternalIssue) && !issue.assignee_id? && can?(current_user, :admin_issue, issue)
+            !issue.is_a?(ExternalIssue) && !issue.assignees.present? && can?(current_user, :admin_issue, issue)
           end
         else
           []
@@ -14,7 +14,7 @@ def assignable_issues
 
     def execute
       assignable_issues.each do |issue|
-        Issues::UpdateService.new(issue.project, current_user, assignee_id: current_user.id).execute(issue)
+        Issues::UpdateService.new(issue.project, current_user, assignee_ids: [current_user.id]).execute(issue)
       end
 
       {
diff --git a/app/services/merge_requests/base_service.rb b/app/services/merge_requests/base_service.rb
index 582d5c47b6603..3542a41ac831b 100644
--- a/app/services/merge_requests/base_service.rb
+++ b/app/services/merge_requests/base_service.rb
@@ -38,6 +38,11 @@ def execute_hooks(merge_request, action = 'open', oldrev = nil)
 
     private
 
+    def create_assignee_note(merge_request)
+      SystemNoteService.change_assignee(
+        merge_request, merge_request.project, current_user, merge_request.assignee)
+    end
+
     # Returns all origin and fork merge requests from `@project` satisfying passed arguments.
     def merge_requests_for(source_branch, mr_states: [:opened, :reopened])
       MergeRequest
diff --git a/app/services/merge_requests/update_service.rb b/app/services/merge_requests/update_service.rb
index ab7fcf3b6e2b7..5c843a258fb3b 100644
--- a/app/services/merge_requests/update_service.rb
+++ b/app/services/merge_requests/update_service.rb
@@ -21,7 +21,10 @@ def execute(merge_request)
       update(merge_request)
     end
 
-    def handle_changes(merge_request, old_labels: [], old_mentioned_users: [])
+    def handle_changes(merge_request, options)
+      old_labels = options[:old_labels] || []
+      old_mentioned_users = options[:old_mentioned_users] || []
+
       if has_changes?(merge_request, old_labels: old_labels)
         todo_service.mark_pending_todos_as_done(merge_request, current_user)
       end
diff --git a/app/services/notification_recipient_service.rb b/app/services/notification_recipient_service.rb
index 8bb995158de7e..988bd0a7cdbe9 100644
--- a/app/services/notification_recipient_service.rb
+++ b/app/services/notification_recipient_service.rb
@@ -19,9 +19,14 @@ def build_recipients(target, current_user, action: nil, previous_assignee: nil,
     # Re-assign is considered as a mention of the new assignee so we add the
     # new assignee to the list of recipients after we rejected users with
     # the "on mention" notification level
-    if [:reassign_merge_request, :reassign_issue].include?(custom_action)
+    case custom_action
+    when :reassign_merge_request
       recipients << previous_assignee if previous_assignee
       recipients << target.assignee
+    when :reassign_issue
+      previous_assignees = Array(previous_assignee)
+      recipients.concat(previous_assignees)
+      recipients.concat(target.assignees)
     end
 
     recipients = reject_muted_users(recipients)
diff --git a/app/services/notification_service.rb b/app/services/notification_service.rb
index 6b186263bd103..fe9f5ae2b3304 100644
--- a/app/services/notification_service.rb
+++ b/app/services/notification_service.rb
@@ -66,8 +66,23 @@ def close_issue(issue, current_user)
   #  * issue new assignee if their notification level is not Disabled
   #  * users with custom level checked with "reassign issue"
   #
-  def reassigned_issue(issue, current_user)
-    reassign_resource_email(issue, issue.project, current_user, :reassigned_issue_email)
+  def reassigned_issue(issue, current_user, previous_assignees = [])
+    recipients = NotificationRecipientService.new(issue.project).build_recipients(
+      issue,
+      current_user,
+      action: "reassign",
+      previous_assignee: previous_assignees
+    )
+
+    recipients.each do |recipient|
+      mailer.send(
+        :reassigned_issue_email,
+        recipient.id,
+        issue.id,
+        previous_assignees.map(&:id),
+        current_user.id
+      ).deliver_later
+    end
   end
 
   # When we add labels to an issue we should send an email to:
@@ -367,10 +382,10 @@ def mailer
   end
 
   def previous_record(object, attribute)
-    if object && attribute
-      if object.previous_changes.include?(attribute)
-        object.previous_changes[attribute].first
-      end
+    return unless object && attribute
+
+    if object.previous_changes.include?(attribute)
+      object.previous_changes[attribute].first
     end
   end
 end
diff --git a/app/services/slash_commands/interpret_service.rb b/app/services/slash_commands/interpret_service.rb
index 6aeebc26685ae..dc2ee695384e0 100644
--- a/app/services/slash_commands/interpret_service.rb
+++ b/app/services/slash_commands/interpret_service.rb
@@ -88,20 +88,33 @@ def extractor
       current_user.can?(:"admin_#{issuable.to_ability_name}", project)
     end
     command :assign do |assignee_param|
-      user = extract_references(assignee_param, :user).first
-      user ||= User.find_by(username: assignee_param)
+      user_ids = extract_references(assignee_param, :user).map(&:id)
 
-      @updates[:assignee_id] = user.id if user
+      if user_ids.empty?
+        user_ids = User.where(username: assignee_param.split(' ').map(&:strip)).pluck(:id)
+      end
+
+      next if user_ids.empty?
+
+      if issuable.is_a?(Issue)
+        @updates[:assignee_ids] = user_ids
+      else
+        @updates[:assignee_id] = user_ids.last
+      end
     end
 
     desc 'Remove assignee'
     condition do
       issuable.persisted? &&
-        issuable.assignee_id? &&
+        issuable.assignees.any? &&
         current_user.can?(:"admin_#{issuable.to_ability_name}", project)
     end
     command :unassign do
-      @updates[:assignee_id] = nil
+      if issuable.is_a?(Issue)
+        @updates[:assignee_ids] = []
+      else
+        @updates[:assignee_id] = nil
+      end
     end
 
     desc 'Set milestone'
diff --git a/app/services/system_note_service.rb b/app/services/system_note_service.rb
index c9e25c7aaa20e..898c69b3f8c2c 100644
--- a/app/services/system_note_service.rb
+++ b/app/services/system_note_service.rb
@@ -49,6 +49,44 @@ def change_assignee(noteable, project, author, assignee)
     create_note(NoteSummary.new(noteable, project, author, body, action: 'assignee'))
   end
 
+  # Called when the assignees of an Issue is changed or removed
+  #
+  # issue - Issue object
+  # project  - Project owning noteable
+  # author   - User performing the change
+  # assignees - Users being assigned, or nil
+  #
+  # Example Note text:
+  #
+  #   "removed all assignees"
+  #
+  #   "assigned to @user1 additionally to @user2"
+  #
+  #   "assigned to @user1, @user2 and @user3 and unassigned from @user4 and @user5"
+  #
+  #   "assigned to @user1 and @user2"
+  #
+  # Returns the created Note object
+  def change_issue_assignees(issue, project, author, old_assignees)
+    body =
+      if issue.assignees.any? && old_assignees.any?
+        unassigned_users = old_assignees - issue.assignees
+        added_users = issue.assignees.to_a - old_assignees
+
+        text_parts = []
+        text_parts << "assigned to #{added_users.map(&:to_reference).to_sentence}" if added_users.any?
+        text_parts << "unassigned #{unassigned_users.map(&:to_reference).to_sentence}" if unassigned_users.any?
+
+        text_parts.join(' and ')
+      elsif old_assignees.any?
+        "removed all assignees"
+      elsif issue.assignees.any?
+        "assigned to #{issue.assignees.map(&:to_reference).to_sentence}"
+      end
+
+    create_note(noteable: issue, project: project, author: author, note: body)
+  end
+
   # Called when one or more labels on a Noteable are added and/or removed
   #
   # noteable       - Noteable object
diff --git a/app/services/todo_service.rb b/app/services/todo_service.rb
index 8ae61694b503e..322c62863655a 100644
--- a/app/services/todo_service.rb
+++ b/app/services/todo_service.rb
@@ -251,9 +251,9 @@ def handle_note(note, author, skip_users = [])
   end
 
   def create_assignment_todo(issuable, author)
-    if issuable.assignee
+    if issuable.assignees.any?
       attributes = attributes_for_todo(issuable.project, issuable, author, Todo::ASSIGNED)
-      create_todos(issuable.assignee, attributes)
+      create_todos(issuable.assignees, attributes)
     end
   end
 
diff --git a/app/views/issues/_issue.atom.builder b/app/views/issues/_issue.atom.builder
index 23a884480552a..9ec765c45f927 100644
--- a/app/views/issues/_issue.atom.builder
+++ b/app/views/issues/_issue.atom.builder
@@ -23,10 +23,12 @@ xml.entry do
     end
   end
 
-  if issue.assignee
-    xml.assignee do
-      xml.name issue.assignee.name
-      xml.email issue.assignee_public_email
+  if issue.assignees.any?
+    xml.assignees do
+      issue.assignees.each do |assignee|
+        xml.name assignee.name
+        xml.email assignee.public_email
+      end
     end
   end
 end
diff --git a/app/views/notify/_reassigned_issuable_email.text.erb b/app/views/notify/_reassigned_issuable_email.text.erb
deleted file mode 100644
index daf20a226dd7b..0000000000000
--- a/app/views/notify/_reassigned_issuable_email.text.erb
+++ /dev/null
@@ -1,6 +0,0 @@
-Reassigned <%= issuable.class.model_name.human.titleize %> <%= issuable.iid %>
-
-<%= url_for([issuable.project.namespace.becomes(Namespace), issuable.project, issuable, { only_path: false }]) %>
-
-Assignee changed <%= "from #{@previous_assignee.name}" if @previous_assignee -%>
- to <%= "#{issuable.assignee_id ? issuable.assignee_name : 'Unassigned'}" %>
diff --git a/app/views/notify/new_issue_email.html.haml b/app/views/notify/new_issue_email.html.haml
index c762578971a5f..eb5157ccac920 100644
--- a/app/views/notify/new_issue_email.html.haml
+++ b/app/views/notify/new_issue_email.html.haml
@@ -2,9 +2,9 @@
   %p.details
     #{link_to @issue.author_name, user_url(@issue.author)} created an issue:
 
-- if @issue.assignee_id.present?
+- if @issue.assignees.any?
   %p
-    Assignee: #{@issue.assignee_name}
+    Assignee: #{@issue.assignee_list}
 
 - if @issue.description
   %div
diff --git a/app/views/notify/new_issue_email.text.erb b/app/views/notify/new_issue_email.text.erb
index ca5c2f2688c0c..13f1ac08e9454 100644
--- a/app/views/notify/new_issue_email.text.erb
+++ b/app/views/notify/new_issue_email.text.erb
@@ -2,6 +2,6 @@ New Issue was created.
 
 Issue <%= @issue.iid %>: <%= url_for(namespace_project_issue_url(@issue.project.namespace, @issue.project, @issue)) %>
 Author:    <%= @issue.author_name %>
-Assignee:  <%= @issue.assignee_name %>
+Assignee:  <%= @issue.assignee_list %>
 
 <%= @issue.description %>
diff --git a/app/views/notify/new_mention_in_issue_email.text.erb b/app/views/notify/new_mention_in_issue_email.text.erb
index 457e94b48001b..f19ac3adfc7aa 100644
--- a/app/views/notify/new_mention_in_issue_email.text.erb
+++ b/app/views/notify/new_mention_in_issue_email.text.erb
@@ -2,6 +2,6 @@ You have been mentioned in an issue.
 
 Issue <%= @issue.iid %>: <%= url_for(namespace_project_issue_url(@issue.project.namespace, @issue.project, @issue)) %>
 Author:    <%= @issue.author_name %>
-Assignee:  <%= @issue.assignee_name %>
+Assignee:  <%= @issue.assignee_list %>
 
 <%= @issue.description %>
diff --git a/app/views/notify/reassigned_issue_email.html.haml b/app/views/notify/reassigned_issue_email.html.haml
index 498ba8b83651b..ee2f40e1683a4 100644
--- a/app/views/notify/reassigned_issue_email.html.haml
+++ b/app/views/notify/reassigned_issue_email.html.haml
@@ -1 +1,10 @@
-= render 'reassigned_issuable_email', issuable: @issue
+%p
+  Assignee changed
+  - if @previous_assignees.any?
+    from
+    %strong= @previous_assignees.map(&:name).to_sentence
+  to
+  - if @issue.assignees.any?
+    %strong= @issue.assignee_list
+  - else
+    %strong Unassigned
diff --git a/app/views/notify/reassigned_issue_email.text.erb b/app/views/notify/reassigned_issue_email.text.erb
index 710253be98429..6c357f1074a48 100644
--- a/app/views/notify/reassigned_issue_email.text.erb
+++ b/app/views/notify/reassigned_issue_email.text.erb
@@ -1 +1,6 @@
-<%= render 'reassigned_issuable_email', issuable: @issue %>
+Reassigned Issue <%= @issue.iid %>
+
+<%= url_for([@issue.project.namespace.becomes(Namespace), @issue.project, @issue, { only_path: false }]) %>
+
+Assignee changed <%= "from #{@previous_assignees.map(&:name).to_sentence}" if @previous_assignees.any? -%>
+ to <%= "#{@issue.assignees.any? ? @issue.assignee_list : 'Unassigned'}" %>
diff --git a/app/views/notify/reassigned_merge_request_email.html.haml b/app/views/notify/reassigned_merge_request_email.html.haml
index 2a650130f59ce..841df872857fc 100644
--- a/app/views/notify/reassigned_merge_request_email.html.haml
+++ b/app/views/notify/reassigned_merge_request_email.html.haml
@@ -1 +1,9 @@
-= render 'reassigned_issuable_email', issuable: @merge_request
+Reassigned Merge Request #{ @merge_request.iid }
+
+= url_for([@merge_request.project.namespace.becomes(Namespace), @merge_request.project, @merge_request, { only_path: false }])
+
+Assignee changed
+- if @previous_assignee
+  from #{@previous_assignee.name}
+to
+= @merge_request.assignee_id ? @merge_request.assignee_name : 'Unassigned'
diff --git a/app/views/notify/reassigned_merge_request_email.text.erb b/app/views/notify/reassigned_merge_request_email.text.erb
index b5b4f1ff99a6b..998a40fefdeb3 100644
--- a/app/views/notify/reassigned_merge_request_email.text.erb
+++ b/app/views/notify/reassigned_merge_request_email.text.erb
@@ -1 +1,6 @@
-<%= render 'reassigned_issuable_email', issuable: @merge_request %>
+Reassigned Merge Request <%= @merge_request.iid %>
+
+<%= url_for([@merge_request.project.namespace.becomes(Namespace), @merge_request.project, @merge_request, { only_path: false }]) %>
+
+Assignee changed <%= "from #{@previous_assignee.name}" if @previous_assignee -%>
+ to <%= "#{@merge_request.assignee_id ? @merge_request.assignee_name : 'Unassigned'}" %>
diff --git a/app/views/projects/boards/components/sidebar/_assignee.html.haml b/app/views/projects/boards/components/sidebar/_assignee.html.haml
index 0f42433452134..96f7f12b1d74a 100644
--- a/app/views/projects/boards/components/sidebar/_assignee.html.haml
+++ b/app/views/projects/boards/components/sidebar/_assignee.html.haml
@@ -1,40 +1,32 @@
-.block.assignee
-  .title.hide-collapsed
-    Assignee
-    - if can?(current_user, :admin_issue, @project)
-      = icon("spinner spin", class: "block-loading")
-      = link_to "Edit", "#", class: "edit-link pull-right"
-  .value.hide-collapsed
-    %span.assign-yourself.no-value{ "v-if" => "!issue.assignee" }
-      No assignee
-      - if can?(current_user, :admin_issue, @project)
-        \-
-        %a.js-assign-yourself{ href: "#" }
-          assign yourself
-    %a.author_link.bold{ ":href" => "'#{root_url}' + issue.assignee.username",
-      "v-if" => "issue.assignee" }
-      %img.avatar.avatar-inline.s32{ ":src" => "issue.assignee.avatar",
-        width: "32", alt: "Avatar" }
-      %span.author
-        {{ issue.assignee.name }}
-      %span.username
-        = precede "@" do
-          {{ issue.assignee.username }}
+.block.assignee{ ref: "assigneeBlock" }
+  %template{ "v-if" => "issue.assignees" }
+    %assignee-title{ ":number-of-assignees" => "issue.assignees.length",
+      ":loading" => "loadingAssignees",
+      ":editable" => can?(current_user, :admin_issue, @project) }
+    %assignees.value{ "root-path" => "#{root_url}",
+      ":users" => "issue.assignees",
+      ":editable" => can?(current_user, :admin_issue, @project),
+      "@assign-self" => "assignSelf" }
+
   - if can?(current_user, :admin_issue, @project)
     .selectbox.hide-collapsed
       %input{ type: "hidden",
-        name: "issue[assignee_id]",
-        id: "issue_assignee_id",
-        ":value" => "issue.assignee.id",
-        "v-if" => "issue.assignee" }
+        name: "issue[assignee_ids][]",
+        ":value" => "assignee.id",
+        "v-if" => "issue.assignees",
+        "v-for" => "assignee in issue.assignees" }
       .dropdown
+<<<<<<< HEAD
         %button.dropdown-menu-toggle.js-user-search.js-author-search.js-issue-board-sidebar{ type: "button", data: { toggle: "dropdown", field_name: "issue[assignee_id]", first_user: (current_user.username if current_user), current_user: "true", project_id: @project.id, null_user: "true", null_user_default: "true" },
+=======
+        %button.dropdown-menu-toggle.js-user-search.js-author-search.js-multiselect.js-save-user-data.js-issue-board-sidebar{ type: "button", ref: "assigneeDropdown", data: { toggle: "dropdown", field_name: "issue[assignee_ids][]", first_user: (current_user.username if current_user), current_user: "true", project_id: @project.id, null_user: "true", multi_select: "true", dropdown: { header: 'Assignee(s)'} },
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
           ":data-issuable-id" => "issue.id",
           ":data-selected" => "assigneeId",
           ":data-issue-update" => "'#{namespace_project_issues_path(@project.namespace, @project)}/' + issue.id + '.json'" }
-          Select assignee
+          Select assignee(s)
           = icon("chevron-down")
-        .dropdown-menu.dropdown-menu-user.dropdown-menu-selectable.dropdown-menu-author
+        .dropdown-menu.dropdown-select.dropdown-menu-user.dropdown-menu-selectable.dropdown-menu-author
           = dropdown_title("Assign to")
           = dropdown_filter("Search users")
           = dropdown_content
diff --git a/app/views/projects/issues/_issue.html.haml b/app/views/projects/issues/_issue.html.haml
index 0e3902c066ad2..c184e0e00224b 100644
--- a/app/views/projects/issues/_issue.html.haml
+++ b/app/views/projects/issues/_issue.html.haml
@@ -13,9 +13,9 @@
             %li
               CLOSED
 
-          - if issue.assignee
+          - if issue.assignees.any?
             %li
-              = link_to_member(@project, issue.assignee, name: false, title: "Assigned to :name")
+              = render 'shared/issuable/assignees', project: @project, issue: issue
 
           = render 'shared/issuable_meta_data', issuable: issue
 
diff --git a/app/views/shared/issuable/_assignees.html.haml b/app/views/shared/issuable/_assignees.html.haml
new file mode 100644
index 0000000000000..36bbb1148d49b
--- /dev/null
+++ b/app/views/shared/issuable/_assignees.html.haml
@@ -0,0 +1,15 @@
+- max_render = 3
+- max = [max_render, issue.assignees.length].min
+
+- issue.assignees.each_with_index do |assignee, index|
+  - if index < max
+    = link_to_member(@project, assignee, name: false, title: "Assigned to :name")
+
+- if issue.assignees.length > max_render
+  - counter = issue.assignees.length - max_render
+
+  %span{ class: 'avatar-counter has-tooltip', data: { container: 'body', placement: 'bottom', 'line-type' => 'old', 'original-title' => "+#{counter} more assignees" } }
+    - if counter < 99
+      = "+#{counter}"
+    - else
+      99+
diff --git a/app/views/shared/issuable/_participants.html.haml b/app/views/shared/issuable/_participants.html.haml
index 171da89993736..db407363a0929 100644
--- a/app/views/shared/issuable/_participants.html.haml
+++ b/app/views/shared/issuable/_participants.html.haml
@@ -12,9 +12,9 @@
     - participants.each do |participant|
       .participants-author.js-participants-author
         = link_to_member(@project, participant, name: false, size: 24)
-    - if participants_extra > 0
-      .participants-more
-        %a.js-participants-more{ href: "#", data: { original_text: "+ #{participants_size - 7} more", less_text: "- show less" } }
-          + #{participants_extra} more
+  - if participants_extra > 0
+    .hide-collapsed.participants-more
+      %a.js-participants-more{ href: "#", data: { original_text: "+ #{participants_size - 7} more", less_text: "- show less" } }
+        + #{participants_extra} more
 :javascript
   IssuableContext.prototype.PARTICIPANTS_ROW_COUNT = #{participants_row};
diff --git a/app/views/shared/issuable/_search_bar.html.haml b/app/views/shared/issuable/_search_bar.html.haml
index b6fce5e3cd439..dd727f1abfb54 100644
--- a/app/views/shared/issuable/_search_bar.html.haml
+++ b/app/views/shared/issuable/_search_bar.html.haml
@@ -124,8 +124,17 @@
                 %li
                   %a{ href: "#", data: { id: "close" } } Closed
           .filter-item.inline
+            - if type == :issues
+              - field_name = "update[assignee_ids][]"
+            - else
+              - field_name = "update[assignee_id]"
+
             = dropdown_tag("Assignee", options: { toggle_class: "js-user-search js-update-assignee js-filter-submit js-filter-bulk-update", title: "Assign to", filter: true, dropdown_class: "dropdown-menu-user dropdown-menu-selectable",
+<<<<<<< HEAD
               placeholder: "Search authors", data: { first_user: (current_user.username if current_user), null_user: true, current_user: true, project_id: @project.id, field_name: "update[assignee_id]", default_label: "Assignee" } })
+=======
+              placeholder: "Search authors", data: { first_user: (current_user.username if current_user), null_user: true, current_user: true, project_id: @project.id, field_name: field_name } })
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
           .filter-item.inline
             = dropdown_tag("Milestone", options: { title: "Assign milestone", toggle_class: 'js-milestone-select js-extra-options js-filter-submit js-filter-bulk-update', filter: true, dropdown_class: "dropdown-menu-selectable dropdown-menu-milestone", placeholder: "Search milestones", data: { show_no: true, field_name: "update[milestone_id]", project_id: @project.id, milestones: namespace_project_milestones_path(@project.namespace, @project, :json), use_id: true, default_label: "Milestone" } })
           .filter-item.inline.labels-filter
diff --git a/app/views/shared/issuable/_sidebar.html.haml b/app/views/shared/issuable/_sidebar.html.haml
index bc638e994f3c4..9bf73c87c88aa 100644
--- a/app/views/shared/issuable/_sidebar.html.haml
+++ b/app/views/shared/issuable/_sidebar.html.haml
@@ -1,10 +1,10 @@
 - todo = issuable_todo(issuable)
 - content_for :page_specific_javascripts do
   = page_specific_javascript_bundle_tag('common_vue')
-  = page_specific_javascript_bundle_tag('issuable')
+  = page_specific_javascript_bundle_tag('sidebar')
 
 %aside.right-sidebar.js-right-sidebar{ data: { "offset-top" => "102", "spy" => "affix" }, class: sidebar_gutter_collapsed_class, 'aria-live' => 'polite' }
-  .issuable-sidebar
+  .issuable-sidebar{ data: { endpoint: "#{issuable_json_path(issuable)}" } }
     - can_edit_issuable = can?(current_user, :"admin_#{issuable.to_ability_name}", @project)
     .block.issuable-sidebar-header
       - if current_user
@@ -20,36 +20,59 @@
         .block.todo.hide-expanded
           = render "shared/issuable/sidebar_todo", todo: todo, issuable: issuable, is_collapsed: true
       .block.assignee
-        .sidebar-collapsed-icon.sidebar-collapsed-user{ data: { toggle: "tooltip", placement: "left", container: "body" }, title: (issuable.assignee.name if issuable.assignee) }
-          - if issuable.assignee
-            = link_to_member(@project, issuable.assignee, size: 24)
-          - else
-            = icon('user', 'aria-hidden': 'true')
-        .title.hide-collapsed
-          Assignee
-          = icon('spinner spin', class: 'hidden block-loading', 'aria-hidden': 'true')
-          - if can_edit_issuable
-            = link_to 'Edit', '#', class: 'edit-link pull-right'
-        .value.hide-collapsed
-          - if issuable.assignee
-            = link_to_member(@project, issuable.assignee, size: 32, extra_class: 'bold') do
-              - if issuable.instance_of?(MergeRequest) && !issuable.can_be_merged_by?(issuable.assignee)
-                %span.pull-right.cannot-be-merged{ data: { toggle: 'tooltip', placement: 'left' }, title: 'Not allowed to merge' }
-                  = icon('exclamation-triangle', 'aria-hidden': 'true')
-              %span.username
-                = issuable.assignee.to_reference
-          - else
-            %span.assign-yourself.no-value
-              No assignee
-              - if can_edit_issuable
-                \-
-                %a.js-assign-yourself{ href: '#' }
-                  assign yourself
+        - if issuable.instance_of?(Issue)
+          #js-vue-sidebar-assignees{ data: { field: "#{issuable.to_ability_name}[assignee_ids]" } }
+        - else
+          .sidebar-collapsed-icon.sidebar-collapsed-user{ data: { toggle: "tooltip", placement: "left", container: "body" }, title: (issuable.assignee.name if issuable.assignee) }
+            - if issuable.assignee
+              = link_to_member(@project, issuable.assignee, size: 24)
+            - else
+              = icon('user', 'aria-hidden': 'true')
+          .title.hide-collapsed
+            Assignee
+            = icon('spinner spin', class: 'hidden block-loading', 'aria-hidden': 'true')
+            - if can_edit_issuable
+              = link_to 'Edit', '#', class: 'edit-link pull-right'
+          .value.hide-collapsed
+            - if issuable.assignee
+              = link_to_member(@project, issuable.assignee, size: 32, extra_class: 'bold') do
+                - if issuable.instance_of?(MergeRequest) && !issuable.can_be_merged_by?(issuable.assignee)
+                  %span.pull-right.cannot-be-merged{ data: { toggle: 'tooltip', placement: 'left' }, title: 'Not allowed to merge' }
+                    = icon('exclamation-triangle', 'aria-hidden': 'true')
+                %span.username
+                  = issuable.assignee.to_reference
+            - else
+              %span.assign-yourself.no-value
+                No assignee
+                - if can_edit_issuable
+                  \-
+                  %a.js-assign-yourself{ href: '#' }
+                    assign yourself
 
         .selectbox.hide-collapsed
+<<<<<<< HEAD
           = f.hidden_field 'assignee_id', value: issuable.assignee_id, id: 'issue_assignee_id'
           = dropdown_tag('Select assignee', options: { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true, null_user_default: true, selected: issuable.assignee_id } })
+=======
+          - issuable.assignees.each do |assignee|
+            = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", assignee.id, id: nil
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
 
+          - options = { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true } }
+
+          - if issuable.instance_of?(Issue)
+            - if issuable.assignees.length == 0
+              = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", 0, id: nil
+            - title = 'Select assignee(s)'
+            - options[:toggle_class] += ' js-multiselect js-save-user-data'
+            - options[:data][:field_name] = "#{issuable.to_ability_name}[assignee_ids][]"
+            - options[:data][:multi_select] = true
+            - options[:data]['dropdown-title'] = title
+            - options[:data]['dropdown-header'] = 'Assignee(s)'
+          - else
+            - title = 'Select assignee'
+
+          = dropdown_tag(title, options: options)
       .block.milestone
         .sidebar-collapsed-icon
           = icon('clock-o', 'aria-hidden': 'true')
@@ -75,11 +98,10 @@
           = dropdown_tag('Milestone', options: { title: 'Assign milestone', toggle_class: 'js-milestone-select js-extra-options', filter: true, dropdown_class: 'dropdown-menu-selectable', placeholder: 'Search milestones', data: { show_no: true, field_name: "#{issuable.to_ability_name}[milestone_id]", project_id: @project.id, issuable_id: issuable.id, milestones: namespace_project_milestones_path(@project.namespace, @project, :json), ability_name: issuable.to_ability_name, issue_update: issuable_json_path(issuable), use_id: true }})
       - if issuable.has_attribute?(:time_estimate)
         #issuable-time-tracker.block
-          %issuable-time-tracker{ ':time_estimate' => 'issuable.time_estimate', ':time_spent' => 'issuable.total_time_spent', ':human_time_estimate' => 'issuable.human_time_estimate', ':human_time_spent' => 'issuable.human_total_time_spent', 'docs-url' => help_page_path('workflow/time_tracking.md') }
-            // Fallback while content is loading
-            .title.hide-collapsed
-              Time tracking
-              = icon('spinner spin', 'aria-hidden': 'true')
+          // Fallback while content is loading
+          .title.hide-collapsed
+            Time tracking
+            = icon('spinner spin', 'aria-hidden': 'true')
       - if issuable.has_attribute?(:due_date)
         .block.due_date
           .sidebar-collapsed-icon
@@ -169,8 +191,13 @@
           = clipboard_button(text: project_ref, title: "Copy reference to clipboard", placement: "left")
 
     :javascript
-      gl.IssuableResource = new gl.SubbableResource('#{issuable_json_path(issuable)}');
-      new gl.IssuableTimeTracking("#{escape_javascript(serialize_issuable(issuable))}");
+      gl.sidebarOptions = {
+        endpoint: "#{issuable_json_path(issuable)}",
+        editable: #{can_edit_issuable ? true : false},
+        currentUser: #{current_user.to_json(only: [:username, :id, :name], methods: :avatar_url)},
+        rootPath: "#{root_path}"
+      };
+
       new MilestoneSelect('{"full_path":"#{@project.full_path}"}');
       new LabelsSelect();
       new IssuableContext('#{escape_javascript(current_user.to_json(only: [:username, :id, :name]))}');
diff --git a/app/views/shared/issuable/form/_issue_assignee.html.haml b/app/views/shared/issuable/form/_issue_assignee.html.haml
new file mode 100644
index 0000000000000..c33474ac3b4e7
--- /dev/null
+++ b/app/views/shared/issuable/form/_issue_assignee.html.haml
@@ -0,0 +1,30 @@
+- issue = issuable
+.block.assignee
+  .sidebar-collapsed-icon.sidebar-collapsed-user{ data: { toggle: "tooltip", placement: "left", container: "body" }, title: (issuable.assignee_list) }
+    - if issue.assignees.any?
+      - issue.assignees.each do |assignee|
+        = link_to_member(@project, assignee, size: 24)
+    - else
+      = icon('user', 'aria-hidden': 'true')
+  .title.hide-collapsed
+    Assignee
+    = icon('spinner spin', class: 'hidden block-loading', 'aria-hidden': 'true')
+    - if can_edit_issuable
+      = link_to 'Edit', '#', class: 'edit-link pull-right'
+  .value.hide-collapsed
+    - if issue.assignees.any?
+      - issue.assignees.each do |assignee|
+        = link_to_member(@project, assignee, size: 32, extra_class: 'bold') do
+          %span.username
+            = assignee.to_reference
+    - else
+      %span.assign-yourself.no-value
+        No assignee
+        - if can_edit_issuable
+          \-
+          %a.js-assign-yourself{ href: '#' }
+            assign yourself
+
+  .selectbox.hide-collapsed
+    = f.hidden_field 'assignee_ids', value: issuable.assignee_ids, id: 'issue_assignee_ids'
+    = dropdown_tag('Select assignee', options: { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true } })
diff --git a/app/views/shared/issuable/form/_merge_request_assignee.html.haml b/app/views/shared/issuable/form/_merge_request_assignee.html.haml
new file mode 100644
index 0000000000000..18011d528a0be
--- /dev/null
+++ b/app/views/shared/issuable/form/_merge_request_assignee.html.haml
@@ -0,0 +1,31 @@
+- merge_request = issuable
+.block.assignee
+  .sidebar-collapsed-icon.sidebar-collapsed-user{ data: { toggle: "tooltip", placement: "left", container: "body" }, title: (merge_request.assignee.name if merge_request.assignee) }
+    - if merge_request.assignee
+      = link_to_member(@project, merge_request.assignee, size: 24)
+    - else
+      = icon('user', 'aria-hidden': 'true')
+  .title.hide-collapsed
+    Assignee
+    = icon('spinner spin', class: 'hidden block-loading', 'aria-hidden': 'true')
+    - if can_edit_issuable
+      = link_to 'Edit', '#', class: 'edit-link pull-right'
+  .value.hide-collapsed
+    - if merge_request.assignee
+      = link_to_member(@project, merge_request.assignee, size: 32, extra_class: 'bold') do
+        - unless merge_request.can_be_merged_by?(merge_request.assignee)
+          %span.pull-right.cannot-be-merged{ data: { toggle: 'tooltip', placement: 'left' }, title: 'Not allowed to merge' }
+            = icon('exclamation-triangle', 'aria-hidden': 'true')
+        %span.username
+          = merge_request.assignee.to_reference
+    - else
+      %span.assign-yourself.no-value
+        No assignee
+        - if can_edit_issuable
+          \-
+          %a.js-assign-yourself{ href: '#' }
+            assign yourself
+
+  .selectbox.hide-collapsed
+    = f.hidden_field 'assignee_id', value: merge_request.assignee_id, id: 'issue_assignee_id'
+    = dropdown_tag('Select assignee', options: { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: @project&.id, author_id: merge_request.author_id, field_name: 'merge_request[assignee_id]', issue_update: issuable_json_path(merge_request), ability_name: 'merge_request', null_user: true } })
diff --git a/app/views/shared/issuable/form/_metadata.html.haml b/app/views/shared/issuable/form/_metadata.html.haml
index 9dbfedb84f1a2..411cb717fc766 100644
--- a/app/views/shared/issuable/form/_metadata.html.haml
+++ b/app/views/shared/issuable/form/_metadata.html.haml
@@ -10,13 +10,24 @@
 .row
   %div{ class: (has_due_date ? "col-lg-6" : "col-sm-12") }
     .form-group.issue-assignee
-      = form.label :assignee_id, "Assignee", class: "control-label #{"col-lg-4" if has_due_date}"
-      .col-sm-10{ class: ("col-lg-8" if has_due_date) }
-        .issuable-form-select-holder
-          = form.hidden_field :assignee_id
-          = dropdown_tag(user_dropdown_label(issuable.assignee_id, "Assignee"), options: { toggle_class: "js-dropdown-keep-input js-user-search js-issuable-form-dropdown js-assignee-search", title: "Select assignee", filter: true, dropdown_class: "dropdown-menu-user dropdown-menu-selectable dropdown-menu-assignee js-filter-submit",
-            placeholder: "Search assignee", data: { first_user: current_user.try(:username), null_user: true, current_user: true, project_id: issuable.project.try(:id), selected: issuable.assignee_id, field_name: "#{issuable.class.model_name.param_key}[assignee_id]", default_label: "Assignee"} })
-        = link_to 'Assign to me', '#', class: "assign-to-me-link #{'hide' if issuable.assignee_id == current_user.id}"
+      - if issuable.is_a?(Issue)
+        = form.label :assignee_ids, "Assignee", class: "control-label #{"col-lg-4" if has_due_date}"
+        .col-sm-10{ class: ("col-lg-8" if has_due_date) }
+          .issuable-form-select-holder.selectbox
+            - issuable.assignees.each do |assignee|
+              = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", assignee.id, id: nil, data: { meta: assignee.name }
+
+            = dropdown_tag(users_dropdown_label(issuable.assignees), options: issue_dropdown_options(issuable, true))
+          = link_to 'Assign to me', '#', class: "assign-to-me-link #{'hide' if issuable.assignees.include?(current_user)}"
+      - else
+        = form.label :assignee_id, "Assignee", class: "control-label #{"col-lg-4" if has_due_date}"
+        .col-sm-10{ class: ("col-lg-8" if has_due_date) }
+          .issuable-form-select-holder
+            = form.hidden_field :assignee_id
+
+            = dropdown_tag(user_dropdown_label(issuable.assignee_id, "Assignee"), options: { toggle_class: "js-dropdown-keep-input js-user-search js-issuable-form-dropdown js-assignee-search", title: "Select assignee", filter: true, dropdown_class: "dropdown-menu-user dropdown-menu-selectable dropdown-menu-assignee js-filter-submit",
+              placeholder: "Search assignee", data: { first_user: current_user.try(:username), null_user: true, current_user: true, project_id: issuable.project.try(:id), selected: issuable.assignee_id, field_name: "#{issuable.class.model_name.param_key}[assignee_id]", default_label: "Assignee"} })
+          = link_to 'Assign to me', '#', class: "assign-to-me-link #{'hide' if issuable.assignee_id == current_user.id}"
     .form-group.issue-milestone
       = form.label :milestone_id, "Milestone", class: "control-label #{"col-lg-4" if has_due_date}"
       .col-sm-10{ class: ("col-lg-8" if has_due_date) }
diff --git a/app/views/shared/milestones/_issuable.html.haml b/app/views/shared/milestones/_issuable.html.haml
index 5247d6a51e642..22547a30cdfb1 100644
--- a/app/views/shared/milestones/_issuable.html.haml
+++ b/app/views/shared/milestones/_issuable.html.haml
@@ -1,7 +1,7 @@
 -# @project is present when viewing Project's milestone
 - project = @project || issuable.project
 - namespace = @project_namespace || project.namespace.becomes(Namespace)
-- assignee = issuable.assignee
+- assignees = issuable.assignees
 - issuable_type = issuable.class.table_name
 - base_url_args = [namespace, project]
 - issuable_type_args = base_url_args + [issuable_type]
@@ -26,7 +26,7 @@
         - render_colored_label(label)
 
     %span.assignee-icon
-      - if assignee
-        = link_to polymorphic_path(issuable_type_args, { milestone_title: @milestone.title, assignee_id: issuable.assignee_id, state: 'all' }),
+      - assignees.each do |assignee|
+        = link_to polymorphic_path(base_url_args, { milestone_title: @milestone.title, assignee_id: assignee.id, state: 'all' }),
                   class: 'has-tooltip', title: "Assigned to #{assignee.name}", data: { container: 'body' } do
-          - image_tag(avatar_icon(issuable.assignee, 16), class: "avatar s16", alt: '')
+          - image_tag(avatar_icon(assignee, 16), class: "avatar s16", alt: '')
diff --git a/changelogs/unreleased/update-issue-board-cards-design.yml b/changelogs/unreleased/update-issue-board-cards-design.yml
new file mode 100644
index 0000000000000..5ef94a74e8a92
--- /dev/null
+++ b/changelogs/unreleased/update-issue-board-cards-design.yml
@@ -0,0 +1,4 @@
+---
+title: Update issue board cards design
+merge_request: 10353
+author:
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 0ec9e48845edc..8cacafd611f64 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -33,7 +33,11 @@ var config = {
     graphs:               './graphs/graphs_bundle.js',
     group:                './group.js',
     groups_list:          './groups_list.js',
+<<<<<<< HEAD
     issuable:             './issuable/issuable_bundle.js',
+=======
+    issues:               './issues/issues_bundle.js',
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     issue_show:           './issue_show/index.js',
     main:                 './main.js',
     merge_conflicts:      './merge_conflicts/merge_conflicts_bundle.js',
@@ -46,6 +50,7 @@ var config = {
     profile:              './profile/profile_bundle.js',
     protected_branches:   './protected_branches/protected_branches_bundle.js',
     protected_tags:       './protected_tags',
+    sidebar:              './sidebar/sidebar_bundle.js',
     snippet:              './snippet/snippet_bundle.js',
     sketch_viewer:        './blob/sketch_viewer.js',
     stl_viewer:           './blob/stl_viewer.js',
@@ -125,7 +130,7 @@ var config = {
         'diff_notes',
         'environments',
         'environments_folder',
-        'issuable',
+        'sidebar',
         'issue_show',
         'merge_conflicts',
         'notebook_viewer',
diff --git a/db/fixtures/development/09_issues.rb b/db/fixtures/development/09_issues.rb
index d93d133d15767..0b32a461d56b1 100644
--- a/db/fixtures/development/09_issues.rb
+++ b/db/fixtures/development/09_issues.rb
@@ -8,7 +8,7 @@
         description: FFaker::Lorem.sentence,
         state: ['opened', 'closed'].sample,
         milestone: project.milestones.sample,
-        assignee: project.team.users.sample
+        assignees: [project.team.users.sample]
       }
 
       Issues::CreateService.new(project, project.team.users.sample, issue_params).execute
diff --git a/db/migrate/20170320171632_create_issue_assignees_table.rb b/db/migrate/20170320171632_create_issue_assignees_table.rb
new file mode 100644
index 0000000000000..72b70baa8d916
--- /dev/null
+++ b/db/migrate/20170320171632_create_issue_assignees_table.rb
@@ -0,0 +1,40 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class CreateIssueAssigneesTable < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  INDEX_NAME = 'index_issue_assignees_on_issue_id_and_user_id'
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using the methods "add_concurrent_index" or "add_column_with_default"
+  # you must disable the use of transactions as these methods can not run in an
+  # existing transaction. When using "add_concurrent_index" make sure that this
+  # method is the _only_ method called in the migration, any other changes
+  # should go in a separate migration. This ensures that upon failure _only_ the
+  # index creation fails and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  # disable_ddl_transaction!
+
+  def up
+    create_table :issue_assignees, id: false do |t|
+      t.references :user, foreign_key: { on_delete: :cascade }, index: true, null: false
+      t.references :issue, foreign_key: { on_delete: :cascade }, null: false
+    end
+
+    add_index :issue_assignees, [:issue_id, :user_id], unique: true, name: INDEX_NAME
+  end
+
+  def down
+    drop_table :issue_assignees
+  end
+end
diff --git a/db/migrate/20170320173259_migrate_assignees.rb b/db/migrate/20170320173259_migrate_assignees.rb
new file mode 100644
index 0000000000000..ba8edbd7d32e2
--- /dev/null
+++ b/db/migrate/20170320173259_migrate_assignees.rb
@@ -0,0 +1,52 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class MigrateAssignees < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using the methods "add_concurrent_index" or "add_column_with_default"
+  # you must disable the use of transactions as these methods can not run in an
+  # existing transaction. When using "add_concurrent_index" make sure that this
+  # method is the _only_ method called in the migration, any other changes
+  # should go in a separate migration. This ensures that upon failure _only_ the
+  # index creation fails and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  disable_ddl_transaction!
+
+  def up
+    # Optimisation: this accounts for most of the invalid assignee IDs on GitLab.com
+    update_column_in_batches(:issues, :assignee_id, nil) do |table, query|
+      query.where(table[:assignee_id].eq(0))
+    end
+
+    users = Arel::Table.new(:users)
+
+    update_column_in_batches(:issues, :assignee_id, nil) do |table, query|
+      query.where(table[:assignee_id].not_eq(nil)\
+        .and(
+          users.project("true").where(users[:id].eq(table[:assignee_id])).exists.not
+        ))
+    end
+
+    execute <<-EOF
+      INSERT INTO issue_assignees(issue_id, user_id)
+      SELECT id, assignee_id FROM issues WHERE assignee_id IS NOT NULL
+    EOF
+  end
+
+  def down
+    execute <<-EOF
+      DELETE FROM issue_assignees
+    EOF
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 01c0f00c92496..340d4064c0245 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -452,6 +452,14 @@
 
   add_index "identities", ["user_id"], name: "index_identities_on_user_id", using: :btree
 
+  create_table "issue_assignees", id: false, force: :cascade do |t|
+    t.integer "user_id", null: false
+    t.integer "issue_id", null: false
+  end
+
+  add_index "issue_assignees", ["issue_id", "user_id"], name: "index_issue_assignees_on_issue_id_and_user_id", unique: true, using: :btree
+  add_index "issue_assignees", ["user_id"], name: "index_issue_assignees_on_user_id", using: :btree
+
   create_table "issue_metrics", force: :cascade do |t|
     t.integer "issue_id", null: false
     t.datetime "first_mentioned_in_commit_at"
@@ -1383,6 +1391,8 @@
   add_foreign_key "ci_trigger_requests", "ci_triggers", column: "trigger_id", name: "fk_b8ec8b7245", on_delete: :cascade
   add_foreign_key "ci_trigger_schedules", "ci_triggers", column: "trigger_id", name: "fk_90a406cc94", on_delete: :cascade
   add_foreign_key "ci_triggers", "users", column: "owner_id", name: "fk_e8e10d1964", on_delete: :cascade
+  add_foreign_key "issue_assignees", "issues", on_delete: :cascade
+  add_foreign_key "issue_assignees", "users", on_delete: :cascade
   add_foreign_key "container_repositories", "projects"
   add_foreign_key "issue_metrics", "issues", on_delete: :cascade
   add_foreign_key "label_priorities", "labels", on_delete: :cascade
diff --git a/doc/api/issues.md b/doc/api/issues.md
index 6c10b5ab0e71a..1d43b1298b9b5 100644
--- a/doc/api/issues.md
+++ b/doc/api/issues.md
@@ -70,6 +70,14 @@ Example response:
          "updated_at" : "2016-01-04T15:31:39.996Z"
       },
       "project_id" : 1,
+      "assignees" : [{
+         "state" : "active",
+         "id" : 1,
+         "name" : "Administrator",
+         "web_url" : "https://gitlab.example.com/root",
+         "avatar_url" : null,
+         "username" : "root"
+      }],
       "assignee" : {
          "state" : "active",
          "id" : 1,
@@ -92,6 +100,8 @@ Example response:
 ]
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## List group issues
 
 Get a list of a group's issues.
@@ -153,6 +163,14 @@ Example response:
       "description" : "Omnis vero earum sunt corporis dolor et placeat.",
       "state" : "closed",
       "iid" : 1,
+      "assignees" : [{
+         "avatar_url" : null,
+         "web_url" : "https://gitlab.example.com/lennie",
+         "state" : "active",
+         "username" : "lennie",
+         "id" : 9,
+         "name" : "Dr. Luella Kovacek"
+      }],
       "assignee" : {
          "avatar_url" : null,
          "web_url" : "https://gitlab.example.com/lennie",
@@ -174,6 +192,8 @@ Example response:
 ]
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## List project issues
 
 Get a list of a project's issues.
@@ -235,6 +255,14 @@ Example response:
       "description" : "Omnis vero earum sunt corporis dolor et placeat.",
       "state" : "closed",
       "iid" : 1,
+      "assignees" : [{
+         "avatar_url" : null,
+         "web_url" : "https://gitlab.example.com/lennie",
+         "state" : "active",
+         "username" : "lennie",
+         "id" : 9,
+         "name" : "Dr. Luella Kovacek"
+      }],
       "assignee" : {
          "avatar_url" : null,
          "web_url" : "https://gitlab.example.com/lennie",
@@ -256,6 +284,8 @@ Example response:
 ]
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Single issue
 
 Get a single project issue.
@@ -300,6 +330,14 @@ Example response:
    "description" : "Omnis vero earum sunt corporis dolor et placeat.",
    "state" : "closed",
    "iid" : 1,
+   "assignees" : [{
+      "avatar_url" : null,
+      "web_url" : "https://gitlab.example.com/lennie",
+      "state" : "active",
+      "username" : "lennie",
+      "id" : 9,
+      "name" : "Dr. Luella Kovacek"
+   }],
    "assignee" : {
       "avatar_url" : null,
       "web_url" : "https://gitlab.example.com/lennie",
@@ -321,6 +359,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## New issue
 
 Creates a new project issue.
@@ -329,13 +369,13 @@ Creates a new project issue.
 POST /projects/:id/issues
 ```
 
-| Attribute                                 | Type    | Required | Description  |
-|-------------------------------------------|---------|----------|--------------|
+| Attribute                                 | Type           | Required | Description  |
+|-------------------------------------------|----------------|----------|--------------|
 | `id`                                      | integer/string | yes      | The ID or [URL-encoded path of the project](README.md#namespaced-path-encoding) owned by the authenticated user |
 | `title`                                   | string  | yes      | The title of an issue |
 | `description`                             | string  | no       | The description of an issue  |
 | `confidential`                            | boolean | no       | Set an issue to be confidential. Default is `false`.  |
-| `assignee_id`                             | integer | no       | The ID of a user to assign issue |
+| `assignee_ids`                            | Array[integer] | no       | The ID of a user to assign issue |
 | `milestone_id`                            | integer | no       | The ID of a milestone to assign issue  |
 | `labels`                                  | string  | no       | Comma-separated label names for an issue  |
 | `created_at`                              | string  | no       | Date time string, ISO 8601 formatted, e.g. `2016-03-11T03:45:40Z` (requires admin or project owner rights) |
@@ -357,6 +397,7 @@ Example response:
    "iid" : 14,
    "title" : "Issues with auth",
    "state" : "opened",
+   "assignees" : [],
    "assignee" : null,
    "labels" : [
       "bug"
@@ -380,6 +421,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Edit issue
 
 Updates an existing project issue. This call is also used to mark an issue as
@@ -396,7 +439,7 @@ PUT /projects/:id/issues/:issue_iid
 | `title`        | string  | no       | The title of an issue                                                                                      |
 | `description`  | string  | no       | The description of an issue                                                                                |
 | `confidential` | boolean | no       | Updates an issue to be confidential                                                                        |
-| `assignee_id`  | integer | no       | The ID of a user to assign the issue to                                                                    |
+| `assignee_ids`  | Array[integer] | no       | The ID of a user to assign the issue to                                                                    |
 | `milestone_id` | integer | no       | The ID of a milestone to assign the issue to                                                               |
 | `labels`       | string  | no       | Comma-separated label names for an issue                                                                   |
 | `state_event`  | string  | no       | The state event of an issue. Set `close` to close the issue and `reopen` to reopen it                      |
@@ -430,6 +473,7 @@ Example response:
       "bug"
    ],
    "id" : 85,
+   "assignees" : [],
    "assignee" : null,
    "milestone" : null,
    "subscribed" : true,
@@ -440,6 +484,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Delete an issue
 
 Only for admins and project owners. Soft deletes the issue in question.
@@ -494,6 +540,14 @@ Example response:
   "updated_at": "2016-04-07T12:20:17.596Z",
   "labels": [],
   "milestone": null,
+  "assignees": [{
+    "name": "Miss Monserrate Beier",
+    "username": "axel.block",
+    "id": 12,
+    "state": "active",
+    "avatar_url": "http://www.gravatar.com/avatar/46f6f7dc858ada7be1853f7fb96e81da?s=80&d=identicon",
+    "web_url": "https://gitlab.example.com/axel.block"
+  }],
   "assignee": {
     "name": "Miss Monserrate Beier",
     "username": "axel.block",
@@ -516,6 +570,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Subscribe to an issue
 
 Subscribes the authenticated user to an issue to receive notifications.
@@ -549,6 +605,14 @@ Example response:
   "updated_at": "2016-04-07T12:20:17.596Z",
   "labels": [],
   "milestone": null,
+  "assignees": [{
+    "name": "Miss Monserrate Beier",
+    "username": "axel.block",
+    "id": 12,
+    "state": "active",
+    "avatar_url": "http://www.gravatar.com/avatar/46f6f7dc858ada7be1853f7fb96e81da?s=80&d=identicon",
+    "web_url": "https://gitlab.example.com/axel.block"
+  }],
   "assignee": {
     "name": "Miss Monserrate Beier",
     "username": "axel.block",
@@ -571,6 +635,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Unsubscribe from an issue
 
 Unsubscribes the authenticated user from the issue to not receive notifications
@@ -652,6 +718,14 @@ Example response:
       "updated_at": "2016-06-17T07:47:33.832Z",
       "due_date": null
     },
+    "assignees": [{
+      "name": "Jarret O'Keefe",
+      "username": "francisca",
+      "id": 14,
+      "state": "active",
+      "avatar_url": "http://www.gravatar.com/avatar/a7fa515d53450023c83d62986d0658a8?s=80&d=identicon",
+      "web_url": "https://gitlab.example.com/francisca"
+    }],
     "assignee": {
       "name": "Jarret O'Keefe",
       "username": "francisca",
@@ -683,6 +757,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Set a time estimate for an issue
 
 Sets an estimated time of work for this issue.
diff --git a/doc/user/project/integrations/webhooks.md b/doc/user/project/integrations/webhooks.md
index dbdc93a77a8eb..e15daa2feae1b 100644
--- a/doc/user/project/integrations/webhooks.md
+++ b/doc/user/project/integrations/webhooks.md
@@ -232,6 +232,7 @@ X-Gitlab-Event: Issue Hook
   "object_attributes": {
     "id": 301,
     "title": "New API: create/update/delete file",
+    "assignee_ids": [51],
     "assignee_id": 51,
     "author_id": 51,
     "project_id": 14,
@@ -246,6 +247,11 @@ X-Gitlab-Event: Issue Hook
     "url": "http://example.com/diaspora/issues/23",
     "action": "open"
   },
+  "assignees": [{
+    "name": "User1",
+    "username": "user1",
+    "avatar_url": "http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=40\u0026d=identicon"
+  }],
   "assignee": {
     "name": "User1",
     "username": "user1",
@@ -265,6 +271,9 @@ X-Gitlab-Event: Issue Hook
   }]
 }
 ```
+
+**Note**: `assignee` and `assignee_id` keys are deprecated and now show the first assignee only.
+
 ### Comment events
 
 Triggered when a new comment is made on commits, merge requests, issues, and code snippets.
@@ -544,6 +553,7 @@ X-Gitlab-Event: Note Hook
   "issue": {
     "id": 92,
     "title": "test",
+    "assignee_ids": [],
     "assignee_id": null,
     "author_id": 1,
     "project_id": 5,
@@ -559,6 +569,8 @@ X-Gitlab-Event: Note Hook
 }
 ```
 
+**Note**: `assignee_id` field is deprecated and now shows the first assignee only.
+
 #### Comment on code snippet
 
 **Request header**:
diff --git a/features/steps/dashboard/dashboard.rb b/features/steps/dashboard/dashboard.rb
index c715c85c43c01..bf09d7b7114be 100644
--- a/features/steps/dashboard/dashboard.rb
+++ b/features/steps/dashboard/dashboard.rb
@@ -77,7 +77,7 @@ class Spinach::Features::Dashboard < Spinach::FeatureSteps
 
   step 'project "Shop" has issue "Bugfix1" with label "feature"' do
     project = Project.find_by(name: "Shop")
-    issue = create(:issue, title: "Bugfix1", project: project, assignee: current_user)
+    issue = create(:issue, title: "Bugfix1", project: project, assignees: [current_user])
     issue.labels << project.labels.find_by(title: 'feature')
   end
 end
diff --git a/features/steps/dashboard/todos.rb b/features/steps/dashboard/todos.rb
index 3225e19995bbe..b56558ba0d2ae 100644
--- a/features/steps/dashboard/todos.rb
+++ b/features/steps/dashboard/todos.rb
@@ -182,7 +182,7 @@ def enterprise
   end
 
   def issue
-    @issue ||= create(:issue, assignee: current_user, project: project)
+    @issue ||= create(:issue, assignees: [current_user], project: project)
   end
 
   def merge_request
diff --git a/features/steps/group/milestones.rb b/features/steps/group/milestones.rb
index 49fcd6f120143..0b0983f0d0653 100644
--- a/features/steps/group/milestones.rb
+++ b/features/steps/group/milestones.rb
@@ -113,7 +113,7 @@ def group_milestone
 
       create :issue,
         project: project,
-        assignee: current_user,
+        assignees: [current_user],
         author: current_user,
         milestone: milestone
 
@@ -125,7 +125,7 @@ def group_milestone
 
       issue = create :issue,
         project: project,
-        assignee: current_user,
+        assignees: [current_user],
         author: current_user,
         milestone: milestone
 
diff --git a/features/steps/groups.rb b/features/steps/groups.rb
index 4dc87dc4d9c29..83d8abbab1fae 100644
--- a/features/steps/groups.rb
+++ b/features/steps/groups.rb
@@ -61,7 +61,7 @@ class Spinach::Features::Groups < Spinach::FeatureSteps
   step 'project from group "Owned" has issues assigned to me' do
     create :issue,
       project: project,
-      assignee: current_user,
+      assignees: [current_user],
       author: current_user
   end
 
@@ -123,7 +123,7 @@ class Spinach::Features::Groups < Spinach::FeatureSteps
   step 'the archived project have some issues' do
     create :issue,
       project: @archived_project,
-      assignee: current_user,
+      assignees: [current_user],
       author: current_user
   end
 
diff --git a/lib/api/api.rb b/lib/api/api.rb
index 1bf20f76ad6a4..842c7301a6611 100644
--- a/lib/api/api.rb
+++ b/lib/api/api.rb
@@ -6,6 +6,7 @@ class API < Grape::API
 
     version 'v3', using: :path do
       helpers ::API::V3::Helpers
+      helpers ::API::Helpers::CommonHelpers
 
       mount ::API::V3::AwardEmoji
       mount ::API::V3::Boards
@@ -77,6 +78,7 @@ class API < Grape::API
     # Ensure the namespace is right, otherwise we might load Grape::API::Helpers
     helpers ::SentryHelper
     helpers ::API::Helpers
+    helpers ::API::Helpers::CommonHelpers
 
     # Keep in alphabetical order
     mount ::API::AccessRequests
diff --git a/lib/api/entities.rb b/lib/api/entities.rb
index 6d6ccefe8776c..f8f5548d23da9 100644
--- a/lib/api/entities.rb
+++ b/lib/api/entities.rb
@@ -256,7 +256,11 @@ class Milestone < ProjectEntity
     class IssueBasic < ProjectEntity
       expose :label_names, as: :labels
       expose :milestone, using: Entities::Milestone
-      expose :assignee, :author, using: Entities::UserBasic
+      expose :assignees, :author, using: Entities::UserBasic
+
+      expose :assignee, using: ::API::Entities::UserBasic do |issue, options|
+        issue.assignees.first
+      end
 
       expose :user_notes_count
       expose :upvotes, :downvotes
diff --git a/lib/api/helpers/common_helpers.rb b/lib/api/helpers/common_helpers.rb
new file mode 100644
index 0000000000000..6236fdd43ca1e
--- /dev/null
+++ b/lib/api/helpers/common_helpers.rb
@@ -0,0 +1,13 @@
+module API
+  module Helpers
+    module CommonHelpers
+      def convert_parameters_from_legacy_format(params)
+        if params[:assignee_id].present?
+          params[:assignee_ids] = [params.delete(:assignee_id)]
+        end
+
+        params
+      end
+    end
+  end
+end
diff --git a/lib/api/issues.rb b/lib/api/issues.rb
index 522f0f3be9291..78db960ae28fa 100644
--- a/lib/api/issues.rb
+++ b/lib/api/issues.rb
@@ -32,7 +32,8 @@ def find_issues(args = {})
 
       params :issue_params_ce do
         optional :description, type: String, desc: 'The description of an issue'
-        optional :assignee_id, type: Integer, desc: 'The ID of a user to assign issue'
+        optional :assignee_ids, type: Array[Integer], desc: 'The array of user IDs to assign issue'
+        optional :assignee_id,  type: Integer, desc: '[Deprecated] The ID of a user to assign issue'
         optional :milestone_id, type: Integer, desc: 'The ID of a milestone to assign issue'
         optional :labels, type: String, desc: 'Comma-separated list of label names'
         optional :due_date, type: String, desc: 'Date string in the format YEAR-MONTH-DAY'
@@ -135,6 +136,8 @@ def find_issues(args = {})
 
         issue_params = declared_params(include_missing: false)
 
+        issue_params = convert_parameters_from_legacy_format(issue_params)
+
         issue = ::Issues::CreateService.new(user_project,
                                             current_user,
                                             issue_params.merge(request: request, api: true)).execute
@@ -159,7 +162,7 @@ def find_issues(args = {})
                               desc: 'Date time when the issue was updated. Available only for admins and project owners.'
         optional :state_event, type: String, values: %w[reopen close], desc: 'State of the issue'
         use :issue_params
-        at_least_one_of :title, :description, :assignee_id, :milestone_id,
+        at_least_one_of :title, :description, :assignee_ids, :assignee_id, :milestone_id,
                         :labels, :created_at, :due_date, :confidential, :state_event
       end
       put ':id/issues/:issue_iid' do
@@ -173,6 +176,8 @@ def find_issues(args = {})
 
         update_params = declared_params(include_missing: false).merge(request: request, api: true)
 
+        update_params = convert_parameters_from_legacy_format(update_params)
+
         issue = ::Issues::UpdateService.new(user_project,
                                             current_user,
                                             update_params).execute(issue)
diff --git a/lib/api/v3/entities.rb b/lib/api/v3/entities.rb
index 832b4bdeb4fef..7c8be7e51db56 100644
--- a/lib/api/v3/entities.rb
+++ b/lib/api/v3/entities.rb
@@ -248,6 +248,13 @@ class ProjectHook < ::API::Entities::Hook
         expose :project_id, :issues_events, :merge_requests_events
         expose :note_events, :build_events, :pipeline_events, :wiki_page_events
       end
+
+      class Issue < ::API::Entities::Issue
+        unexpose :assignees
+        expose :assignee do |issue, options|
+          ::API::Entities::UserBasic.represent(issue.assignees.first, options)
+        end
+      end
     end
   end
 end
diff --git a/lib/api/v3/issues.rb b/lib/api/v3/issues.rb
index 715083fc4f80f..cedbeb0ded0f7 100644
--- a/lib/api/v3/issues.rb
+++ b/lib/api/v3/issues.rb
@@ -8,6 +8,7 @@ class Issues < Grape::API
       helpers do
         def find_issues(args = {})
           args = params.merge(args)
+          args = convert_parameters_from_legacy_format(args)
 
           args.delete(:id)
           args[:milestone_title] = args.delete(:milestone)
@@ -51,7 +52,7 @@ def find_issues(args = {})
 
       resource :issues do
         desc "Get currently authenticated user's issues" do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           optional :state, type: String, values: %w[opened closed all], default: 'all',
@@ -60,8 +61,12 @@ def find_issues(args = {})
         end
         get do
           issues = find_issues(scope: 'authored')
+<<<<<<< HEAD
 
           present paginate(issues), with: ::API::Entities::Issue, current_user: current_user
+=======
+          present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
         end
       end
 
@@ -70,7 +75,7 @@ def find_issues(args = {})
       end
       resource :groups, requirements: { id: %r{[^/]+} } do
         desc 'Get a list of group issues' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           optional :state, type: String, values: %w[opened closed all], default: 'all',
@@ -82,7 +87,7 @@ def find_issues(args = {})
 
           issues = find_issues(group_id: group.id, match_all_labels: true)
 
-          present paginate(issues), with: ::API::Entities::Issue, current_user: current_user
+          present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user
         end
       end
 
@@ -94,7 +99,7 @@ def find_issues(args = {})
 
         desc 'Get a list of project issues' do
           detail 'iid filter is deprecated have been removed on V4'
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           optional :state, type: String, values: %w[opened closed all], default: 'all',
@@ -107,22 +112,22 @@ def find_issues(args = {})
 
           issues = find_issues(project_id: project.id)
 
-          present paginate(issues), with: ::API::Entities::Issue, current_user: current_user, project: user_project
+          present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
         end
 
         desc 'Get a single project issue' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :issue_id, type: Integer, desc: 'The ID of a project issue'
         end
         get ":id/issues/:issue_id" do
           issue = find_project_issue(params[:issue_id])
-          present issue, with: ::API::Entities::Issue, current_user: current_user, project: user_project
+          present issue, with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
         end
 
         desc 'Create a new project issue' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :title, type: String, desc: 'The title of an issue'
@@ -140,6 +145,7 @@ def find_issues(args = {})
 
           issue_params = declared_params(include_missing: false)
           issue_params = issue_params.merge(merge_request_to_resolve_discussions_of: issue_params.delete(:merge_request_for_resolving_discussions))
+          issue_params = convert_parameters_from_legacy_format(issue_params)
 
           issue = ::Issues::CreateService.new(user_project,
                                               current_user,
@@ -147,14 +153,14 @@ def find_issues(args = {})
           render_spam_error! if issue.spam?
 
           if issue.valid?
-            present issue, with: ::API::Entities::Issue, current_user: current_user, project: user_project
+            present issue, with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
           else
             render_validation_error!(issue)
           end
         end
 
         desc 'Update an existing issue' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :issue_id, type: Integer, desc: 'The ID of a project issue'
@@ -176,6 +182,7 @@ def find_issues(args = {})
           end
 
           update_params = declared_params(include_missing: false).merge(request: request, api: true)
+          update_params = convert_parameters_from_legacy_format(update_params)
 
           issue = ::Issues::UpdateService.new(user_project,
                                               current_user,
@@ -184,14 +191,14 @@ def find_issues(args = {})
           render_spam_error! if issue.spam?
 
           if issue.valid?
-            present issue, with: ::API::Entities::Issue, current_user: current_user, project: user_project
+            present issue, with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
           else
             render_validation_error!(issue)
           end
         end
 
         desc 'Move an existing issue' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :issue_id, type: Integer, desc: 'The ID of a project issue'
@@ -206,7 +213,7 @@ def find_issues(args = {})
 
           begin
             issue = ::Issues::MoveService.new(user_project, current_user).execute(issue, new_project)
-            present issue, with: ::API::Entities::Issue, current_user: current_user, project: user_project
+            present issue, with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
           rescue ::Issues::MoveService::MoveError => error
             render_api_error!(error.message, 400)
           end
diff --git a/lib/api/v3/merge_requests.rb b/lib/api/v3/merge_requests.rb
index 1616142a619db..b6b7254ae2956 100644
--- a/lib/api/v3/merge_requests.rb
+++ b/lib/api/v3/merge_requests.rb
@@ -34,7 +34,7 @@ def issue_entity(project)
             if project.has_external_issue_tracker?
               ::API::Entities::ExternalIssue
             else
-              ::API::Entities::Issue
+              ::API::V3::Entities::Issue
             end
           end
 
diff --git a/lib/api/v3/milestones.rb b/lib/api/v3/milestones.rb
index be90cec4afcbc..4c7061d493948 100644
--- a/lib/api/v3/milestones.rb
+++ b/lib/api/v3/milestones.rb
@@ -39,7 +39,7 @@ def filter_milestones_state(milestones, state)
         end
 
         desc 'Get all issues for a single project milestone' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :milestone_id, type: Integer, desc: 'The ID of a project milestone'
@@ -56,7 +56,7 @@ def filter_milestones_state(milestones, state)
           }
 
           issues = IssuesFinder.new(current_user, finder_params).execute
-          present paginate(issues), with: ::API::Entities::Issue, current_user: current_user, project: user_project
+          present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
         end
       end
     end
diff --git a/lib/banzai/reference_parser/issue_parser.rb b/lib/banzai/reference_parser/issue_parser.rb
index e02b360924ae2..89ec715ddf6d9 100644
--- a/lib/banzai/reference_parser/issue_parser.rb
+++ b/lib/banzai/reference_parser/issue_parser.rb
@@ -28,7 +28,7 @@ def issues_for_nodes(nodes)
           nodes,
           Issue.all.includes(
             :author,
-            :assignee,
+            :assignees,
             {
               # These associations are primarily used for checking permissions.
               # Eager loading these ensures we don't end up running dozens of
diff --git a/lib/gitlab/chat_commands/presenters/issue_base.rb b/lib/gitlab/chat_commands/presenters/issue_base.rb
index 054f7f4be0ce6..25bc82994baa5 100644
--- a/lib/gitlab/chat_commands/presenters/issue_base.rb
+++ b/lib/gitlab/chat_commands/presenters/issue_base.rb
@@ -22,7 +22,7 @@ def fields
           [
             {
               title: "Assignee",
-              value: @resource.assignee ? @resource.assignee.name : "_None_",
+              value: @resource.assignees.any? ? @resource.assignees.first.name : "_None_",
               short: true
             },
             {
diff --git a/lib/gitlab/fogbugz_import/importer.rb b/lib/gitlab/fogbugz_import/importer.rb
index 222bcdcbf9c8a..3dcee681c72f4 100644
--- a/lib/gitlab/fogbugz_import/importer.rb
+++ b/lib/gitlab/fogbugz_import/importer.rb
@@ -122,15 +122,15 @@ def import_cases
           author_id = user_info(bug['ixPersonOpenedBy'])[:gitlab_id] || project.creator_id
 
           issue = Issue.create!(
-            iid:         bug['ixBug'],
-            project_id:  project.id,
-            title:       bug['sTitle'],
-            description: body,
-            author_id:   author_id,
-            assignee_id: assignee_id,
-            state:       bug['fOpen'] == 'true' ? 'opened' : 'closed',
-            created_at:  date,
-            updated_at:  DateTime.parse(bug['dtLastUpdated'])
+            iid:          bug['ixBug'],
+            project_id:   project.id,
+            title:        bug['sTitle'],
+            description:  body,
+            author_id:    author_id,
+            assignee_ids: [assignee_id],
+            state:        bug['fOpen'] == 'true' ? 'opened' : 'closed',
+            created_at:   date,
+            updated_at:   DateTime.parse(bug['dtLastUpdated'])
           )
 
           issue_labels = ::LabelsFinder.new(nil, project_id: project.id, title: labels).execute(skip_authorization: true)
diff --git a/lib/gitlab/github_import/issue_formatter.rb b/lib/gitlab/github_import/issue_formatter.rb
index 6f5ac4dac0d38..977cd0423ba53 100644
--- a/lib/gitlab/github_import/issue_formatter.rb
+++ b/lib/gitlab/github_import/issue_formatter.rb
@@ -10,7 +10,7 @@ def attributes
           description: description,
           state: state,
           author_id: author_id,
-          assignee_id: assignee_id,
+          assignee_ids: Array(assignee_id),
           created_at: raw_data.created_at,
           updated_at: raw_data.updated_at
         }
diff --git a/lib/gitlab/google_code_import/importer.rb b/lib/gitlab/google_code_import/importer.rb
index 5ca3e6a95cab8..1b43440673c11 100644
--- a/lib/gitlab/google_code_import/importer.rb
+++ b/lib/gitlab/google_code_import/importer.rb
@@ -108,13 +108,13 @@ def import_issues
           end
 
           issue = Issue.create!(
-            iid:         raw_issue['id'],
-            project_id:  project.id,
-            title:       raw_issue['title'],
-            description: body,
-            author_id:   project.creator_id,
-            assignee_id: assignee_id,
-            state:       raw_issue['state'] == 'closed' ? 'closed' : 'opened'
+            iid:          raw_issue['id'],
+            project_id:   project.id,
+            title:        raw_issue['title'],
+            description:  body,
+            author_id:    project.creator_id,
+            assignee_ids: [assignee_id],
+            state:        raw_issue['state'] == 'closed' ? 'closed' : 'opened'
           )
 
           issue_labels = ::LabelsFinder.new(nil, project_id: project.id, title: labels).execute(skip_authorization: true)
diff --git a/spec/controllers/dashboard/todos_controller_spec.rb b/spec/controllers/dashboard/todos_controller_spec.rb
index 762e90f4a1672..085f3fd8543d5 100644
--- a/spec/controllers/dashboard/todos_controller_spec.rb
+++ b/spec/controllers/dashboard/todos_controller_spec.rb
@@ -14,7 +14,7 @@
   describe 'GET #index' do
     context 'when using pagination' do
       let(:last_page) { user.todos.page.total_pages }
-      let!(:issues) { create_list(:issue, 2, project: project, assignee: user) }
+      let!(:issues) { create_list(:issue, 2, project: project, assignees: [user]) }
 
       before do
         issues.each { |issue| todo_service.new_issue(issue, user) }
diff --git a/spec/controllers/projects/boards/issues_controller_spec.rb b/spec/controllers/projects/boards/issues_controller_spec.rb
index 15667e8d4b116..dc3b72c6de4e2 100644
--- a/spec/controllers/projects/boards/issues_controller_spec.rb
+++ b/spec/controllers/projects/boards/issues_controller_spec.rb
@@ -34,7 +34,7 @@
           issue = create(:labeled_issue, project: project, labels: [planning])
           create(:labeled_issue, project: project, labels: [planning])
           create(:labeled_issue, project: project, labels: [development], due_date: Date.tomorrow)
-          create(:labeled_issue, project: project, labels: [development], assignee: johndoe)
+          create(:labeled_issue, project: project, labels: [development], assignees: [johndoe])
           issue.subscribe(johndoe, project)
 
           list_issues user: user, board: board, list: list2
diff --git a/spec/controllers/projects/issues_controller_spec.rb b/spec/controllers/projects/issues_controller_spec.rb
index 5f1f892821a05..1f79e72495ac7 100644
--- a/spec/controllers/projects/issues_controller_spec.rb
+++ b/spec/controllers/projects/issues_controller_spec.rb
@@ -173,12 +173,12 @@
           namespace_id: project.namespace.to_param,
           project_id: project,
           id: issue.iid,
-          issue: { assignee_id: assignee.id },
+          issue: { assignee_ids: [assignee.id] },
           format: :json
         body = JSON.parse(response.body)
 
-        expect(body['assignee'].keys)
-          .to match_array(%w(name username avatar_url))
+        expect(body['assignees'].first.keys)
+          .to match_array(%w(id name username avatar_url))
       end
     end
 
@@ -348,7 +348,7 @@ def move_issue
     let(:admin) { create(:admin) }
     let!(:issue) { create(:issue, project: project) }
     let!(:unescaped_parameter_value) { create(:issue, :confidential, project: project, author: author) }
-    let!(:request_forgery_timing_attack) { create(:issue, :confidential, project: project, assignee: assignee) }
+    let!(:request_forgery_timing_attack) { create(:issue, :confidential, project: project, assignees: [assignee]) }
 
     describe 'GET #index' do
       it 'does not list confidential issues for guests' do
diff --git a/spec/controllers/projects/merge_requests_controller_spec.rb b/spec/controllers/projects/merge_requests_controller_spec.rb
index a793da4162a8d..0483c6b787954 100644
--- a/spec/controllers/projects/merge_requests_controller_spec.rb
+++ b/spec/controllers/projects/merge_requests_controller_spec.rb
@@ -1067,7 +1067,7 @@ def post_assign_issues
     end
 
     it 'correctly pluralizes flash message on success' do
-      issue2.update!(assignee: user)
+      issue2.assignees = [user]
 
       post_assign_issues
 
diff --git a/spec/features/atom/dashboard_issues_spec.rb b/spec/features/atom/dashboard_issues_spec.rb
index 58b14e09740fa..9ea325ab41b05 100644
--- a/spec/features/atom/dashboard_issues_spec.rb
+++ b/spec/features/atom/dashboard_issues_spec.rb
@@ -32,7 +32,7 @@
       end
 
       context "issue with basic fields" do
-        let!(:issue2) { create(:issue, author: user, assignee: assignee, project: project2, description: 'test desc') }
+        let!(:issue2) { create(:issue, author: user, assignees: [assignee], project: project2, description: 'test desc') }
 
         it "renders issue fields" do
           visit issues_dashboard_path(:atom, private_token: user.private_token)
@@ -41,7 +41,7 @@
 
           expect(entry).to be_present
           expect(entry).to have_selector('author email', text: issue2.author_public_email)
-          expect(entry).to have_selector('assignee email', text: issue2.assignee_public_email)
+          expect(entry).to have_selector('assignees email', text: assignee.public_email)
           expect(entry).not_to have_selector('labels')
           expect(entry).not_to have_selector('milestone')
           expect(entry).to have_selector('description', text: issue2.description)
@@ -51,7 +51,7 @@
       context "issue with label and milestone" do
         let!(:milestone1) { create(:milestone, project: project1, title: 'v1') }
         let!(:label1)     { create(:label, project: project1, title: 'label1') }
-        let!(:issue1)     { create(:issue, author: user, assignee: assignee, project: project1, milestone: milestone1) }
+        let!(:issue1)     { create(:issue, author: user, assignees: [assignee], project: project1, milestone: milestone1) }
 
         before do
           issue1.labels << label1
@@ -64,7 +64,7 @@
 
           expect(entry).to be_present
           expect(entry).to have_selector('author email', text: issue1.author_public_email)
-          expect(entry).to have_selector('assignee email', text: issue1.assignee_public_email)
+          expect(entry).to have_selector('assignees email', text: assignee.public_email)
           expect(entry).to have_selector('labels label', text: label1.title)
           expect(entry).to have_selector('milestone', text: milestone1.title)
           expect(entry).not_to have_selector('description')
diff --git a/spec/features/atom/issues_spec.rb b/spec/features/atom/issues_spec.rb
index b3903ec2faf92..78f8f46a04eb2 100644
--- a/spec/features/atom/issues_spec.rb
+++ b/spec/features/atom/issues_spec.rb
@@ -6,7 +6,7 @@
     let!(:assignee) { create(:user, email: 'private2@example.com', public_email: 'public2@example.com') }
     let!(:group)    { create(:group) }
     let!(:project)  { create(:project) }
-    let!(:issue)    { create(:issue, author: user, assignee: assignee, project: project) }
+    let!(:issue)    { create(:issue, author: user, assignees: [assignee], project: project) }
 
     before do
       project.team << [user, :developer]
@@ -22,7 +22,7 @@
           to have_content('application/atom+xml')
         expect(body).to have_selector('title', text: "#{project.name} issues")
         expect(body).to have_selector('author email', text: issue.author_public_email)
-        expect(body).to have_selector('assignee email', text: issue.author_public_email)
+        expect(body).to have_selector('assignees email', text: issue.author_public_email)
         expect(body).to have_selector('entry summary', text: issue.title)
       end
     end
@@ -36,7 +36,7 @@
           to have_content('application/atom+xml')
         expect(body).to have_selector('title', text: "#{project.name} issues")
         expect(body).to have_selector('author email', text: issue.author_public_email)
-        expect(body).to have_selector('assignee email', text: issue.author_public_email)
+        expect(body).to have_selector('assignees email', text: issue.author_public_email)
         expect(body).to have_selector('entry summary', text: issue.title)
       end
     end
diff --git a/spec/features/boards/boards_spec.rb b/spec/features/boards/boards_spec.rb
index a172ce1e8c022..18585488e2660 100644
--- a/spec/features/boards/boards_spec.rb
+++ b/spec/features/boards/boards_spec.rb
@@ -71,7 +71,7 @@
     let!(:list2) { create(:list, board: board, label: development, position: 1) }
 
     let!(:confidential_issue) { create(:labeled_issue, :confidential, project: project, author: user, labels: [planning], relative_position: 9) }
-    let!(:issue1) { create(:labeled_issue, project: project, assignee: user, labels: [planning], relative_position: 8) }
+    let!(:issue1) { create(:labeled_issue, project: project, assignees: [user], labels: [planning], relative_position: 8) }
     let!(:issue2) { create(:labeled_issue, project: project, author: user2, labels: [planning], relative_position: 7) }
     let!(:issue3) { create(:labeled_issue, project: project, labels: [planning], relative_position: 6) }
     let!(:issue4) { create(:labeled_issue, project: project, labels: [planning], relative_position: 5) }
diff --git a/spec/features/boards/modal_filter_spec.rb b/spec/features/boards/modal_filter_spec.rb
index 4a4c13e79c8a9..e1367c675e58b 100644
--- a/spec/features/boards/modal_filter_spec.rb
+++ b/spec/features/boards/modal_filter_spec.rb
@@ -98,7 +98,7 @@
   end
 
   context 'assignee' do
-    let!(:issue) { create(:issue, project: project, assignee: user2) }
+    let!(:issue) { create(:issue, project: project, assignees: [user2]) }
 
     before do
       project.team << [user2, :developer]
diff --git a/spec/features/boards/sidebar_spec.rb b/spec/features/boards/sidebar_spec.rb
index bafa4f0593710..02b6b5dc88865 100644
--- a/spec/features/boards/sidebar_spec.rb
+++ b/spec/features/boards/sidebar_spec.rb
@@ -4,13 +4,14 @@
   include WaitForVueResource
 
   let(:user)         { create(:user) }
+  let(:user2)        { create(:user) }
   let(:project)      { create(:empty_project, :public) }
   let!(:milestone)   { create(:milestone, project: project) }
   let!(:development) { create(:label, project: project, name: 'Development') }
   let!(:bug)         { create(:label, project: project, name: 'Bug') }
   let!(:regression)  { create(:label, project: project, name: 'Regression') }
   let!(:stretch)     { create(:label, project: project, name: 'Stretch') }
-  let!(:issue1)      { create(:labeled_issue, project: project, assignee: user, milestone: milestone, labels: [development], relative_position: 2) }
+  let!(:issue1)      { create(:labeled_issue, project: project, assignees: [user], milestone: milestone, labels: [development], relative_position: 2) }
   let!(:issue2)      { create(:labeled_issue, project: project, labels: [development, stretch], relative_position: 1) }
   let(:board)        { create(:board, project: project) }
   let!(:list)        { create(:list, board: board, label: development, position: 0) }
@@ -20,6 +21,7 @@
     Timecop.freeze
 
     project.team << [user, :master]
+    project.team.add_developer(user2)
 
     login_as(user)
 
@@ -101,6 +103,26 @@
       expect(card).to have_selector('.avatar')
     end
 
+    it 'adds multiple assignees' do
+      click_card(card)
+
+      page.within('.assignee') do
+        click_link 'Edit'
+
+        wait_for_ajax
+
+        page.within('.dropdown-menu-user') do
+          click_link user.name
+          click_link user2.name
+        end
+
+        expect(page).to have_content(user.name)
+        expect(page).to have_content(user2.name)
+      end
+
+      expect(card.all('.avatar').length).to eq(2)
+    end
+
     it 'removes the assignee' do
       card_two = first('.board').find('.card:nth-child(2)')
       click_card(card_two)
@@ -112,10 +134,11 @@
 
         page.within('.dropdown-menu-user') do
           click_link 'Unassigned'
-
-          wait_for_vue_resource
         end
 
+        find('.dropdown-menu-toggle').click
+        wait_for_vue_resource
+
         expect(page).to have_content('No assignee')
       end
 
@@ -128,7 +151,7 @@
       page.within(find('.assignee')) do
         expect(page).to have_content('No assignee')
 
-        click_link 'assign yourself'
+        click_button 'assign yourself'
 
         wait_for_vue_resource
 
@@ -138,7 +161,7 @@
       expect(card).to have_selector('.avatar')
     end
 
-    it 'resets assignee dropdown' do
+    it 'updates assignee dropdown' do
       click_card(card)
 
       page.within('.assignee') do
@@ -162,7 +185,7 @@
       page.within('.assignee') do
         click_link 'Edit'
 
-        expect(page).not_to have_selector('.is-active')
+        expect(page).to have_selector('.is-active')
       end
     end
   end
diff --git a/spec/features/dashboard/issuables_counter_spec.rb b/spec/features/dashboard/issuables_counter_spec.rb
index 4fca7577e7431..3d536c5ba4025 100644
--- a/spec/features/dashboard/issuables_counter_spec.rb
+++ b/spec/features/dashboard/issuables_counter_spec.rb
@@ -7,7 +7,7 @@
   let(:merge_request) { create(:merge_request, source_project: project) }
 
   before do
-    issue.update(assignee: user)
+    issue.assignees = [user]
     merge_request.update(assignee: user)
     login_as(user)
   end
@@ -17,7 +17,7 @@
 
     expect_counters('issues', '1')
 
-    issue.update(assignee: nil)
+    issue.assignees = []
 
     Timecop.travel(3.minutes.from_now) do
       visit issues_path
diff --git a/spec/features/dashboard/issues_spec.rb b/spec/features/dashboard/issues_spec.rb
index f4420814c3a1b..86c7954e60cf0 100644
--- a/spec/features/dashboard/issues_spec.rb
+++ b/spec/features/dashboard/issues_spec.rb
@@ -11,7 +11,7 @@
 
   let!(:authored_issue) { create :issue, author: current_user, project: project }
   let!(:authored_issue_on_public_project) { create :issue, author: current_user, project: public_project }
-  let!(:assigned_issue) { create :issue, assignee: current_user, project: project }
+  let!(:assigned_issue) { create :issue, assignees: [current_user], project: project }
   let!(:other_issue) { create :issue, project: project }
 
   before do
@@ -30,6 +30,11 @@
     find('#assignee_id', visible: false).set('')
     find('.js-author-search', match: :first).click
     find('.dropdown-menu-author li a', match: :first, text: current_user.to_reference).click
+    find('.js-author-search', match: :first).click
+
+    page.within '.dropdown-menu-user' do
+      expect(find('.dropdown-menu-author li a.is-active', match: :first, text: current_user.to_reference)).to be_visible
+    end
 
     expect(page).to have_content(authored_issue.title)
     expect(page).to have_content(authored_issue_on_public_project.title)
diff --git a/spec/features/dashboard_issues_spec.rb b/spec/features/dashboard_issues_spec.rb
index b6b879052315c..ad60fb2c74f51 100644
--- a/spec/features/dashboard_issues_spec.rb
+++ b/spec/features/dashboard_issues_spec.rb
@@ -10,8 +10,8 @@
       project.team << [user, :master]
       login_as(user)
 
-      create(:issue, project: project, author: user, assignee: user)
-      create(:issue, project: project, author: user, assignee: user, milestone: milestone)
+      create(:issue, project: project, author: user, assignees: [user])
+      create(:issue, project: project, author: user, assignees: [user], milestone: milestone)
 
       visit_issues
     end
diff --git a/spec/features/gitlab_flavored_markdown_spec.rb b/spec/features/gitlab_flavored_markdown_spec.rb
index f5b54463df8ff..005a029a39319 100644
--- a/spec/features/gitlab_flavored_markdown_spec.rb
+++ b/spec/features/gitlab_flavored_markdown_spec.rb
@@ -54,11 +54,11 @@
     before do
       @other_issue = create(:issue,
                             author: @user,
-                            assignee: @user,
+                            assignees: [@user],
                             project: project)
       @issue = create(:issue,
                       author: @user,
-                      assignee: @user,
+                      assignees: [@user],
                       project: project,
                       title: "fix #{@other_issue.to_reference}",
                       description: "ask #{fred.to_reference} for details")
diff --git a/spec/features/issues/award_emoji_spec.rb b/spec/features/issues/award_emoji_spec.rb
index 71df3c949db4b..853632614c447 100644
--- a/spec/features/issues/award_emoji_spec.rb
+++ b/spec/features/issues/award_emoji_spec.rb
@@ -7,7 +7,7 @@
   let!(:user)      { create(:user) }
   let(:issue) do
     create(:issue,
-           assignee: @user,
+           assignees: [user],
            project: project)
   end
 
diff --git a/spec/features/issues/filtered_search/filter_issues_spec.rb b/spec/features/issues/filtered_search/filter_issues_spec.rb
index c824aa6a41450..ece62c8da4131 100644
--- a/spec/features/issues/filtered_search/filter_issues_spec.rb
+++ b/spec/features/issues/filtered_search/filter_issues_spec.rb
@@ -51,15 +51,16 @@ def select_search_at_index(pos)
     create(:issue, project: project, title: "issue with 'single quotes'")
     create(:issue, project: project, title: "issue with \"double quotes\"")
     create(:issue, project: project, title: "issue with !@\#{$%^&*()-+")
-    create(:issue, project: project, title: "issue by assignee", milestone: milestone, author: user, assignee: user)
-    create(:issue, project: project, title: "issue by assignee with searchTerm", milestone: milestone, author: user, assignee: user)
+    create(:issue, project: project, title: "issue by assignee", milestone: milestone, author: user, assignees: [user])
+    create(:issue, project: project, title: "issue by assignee with searchTerm", milestone: milestone, author: user, assignees: [user])
+
 
     issue = create(:issue,
       title: "Bug 2",
       project: project,
       milestone: milestone,
       author: user,
-      assignee: user)
+      assignees: [user])
     issue.labels << bug_label
 
     issue_with_caps_label = create(:issue,
@@ -67,7 +68,7 @@ def select_search_at_index(pos)
       project: project,
       milestone: milestone,
       author: user,
-      assignee: user)
+      assignees: [user])
     issue_with_caps_label.labels << caps_sensitive_label
 
     issue_with_everything = create(:issue,
@@ -75,7 +76,7 @@ def select_search_at_index(pos)
       project: project,
       milestone: milestone,
       author: user,
-      assignee: user)
+      assignees: [user])
     issue_with_everything.labels << bug_label
     issue_with_everything.labels << caps_sensitive_label
 
diff --git a/spec/features/issues/form_spec.rb b/spec/features/issues/form_spec.rb
index 21b8cf3add5c5..5798292033b14 100644
--- a/spec/features/issues/form_spec.rb
+++ b/spec/features/issues/form_spec.rb
@@ -10,7 +10,7 @@
   let!(:milestone) { create(:milestone, project: project) }
   let!(:label)     { create(:label, project: project) }
   let!(:label2)    { create(:label, project: project) }
-  let!(:issue)     { create(:issue, project: project, assignee: user, milestone: milestone) }
+  let!(:issue)     { create(:issue, project: project, assignees: [user], milestone: milestone) }
 
   before do
     project.team << [user, :master]
@@ -23,25 +23,65 @@
       visit new_namespace_project_issue_path(project.namespace, project)
     end
 
+    describe 'multiple assignees' do
+      before do
+        click_button 'Unassigned'
+      end
+
+      it 'unselects other assignees when unassigned is selected' do
+        page.within '.dropdown-menu-user' do
+          click_link user2.name
+        end
+
+        page.within '.dropdown-menu-user' do
+          click_link 'Unassigned'
+        end
+
+        page.within '.js-assignee-search' do
+          expect(page).to have_content 'Unassigned'
+        end
+
+        expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match('0')
+      end
+
+      it 'toggles assign to me when current user is selected and unselected' do
+        page.within '.dropdown-menu-user' do
+          click_link user.name
+        end
+
+        expect(find('a', text: 'Assign to me', visible: false)).not_to be_visible
+
+        page.within '.dropdown-menu-user' do
+          click_link user.name
+        end
+
+        expect(find('a', text: 'Assign to me')).to be_visible
+      end
+    end
+
     it 'allows user to create new issue' do
       fill_in 'issue_title', with: 'title'
       fill_in 'issue_description', with: 'title'
 
       expect(find('a', text: 'Assign to me')).to be_visible
-      click_button 'Assignee'
+      click_button 'Unassigned'
       page.within '.dropdown-menu-user' do
         click_link user2.name
       end
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user2.id.to_s)
+      expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match(user2.id.to_s)
       page.within '.js-assignee-search' do
         expect(page).to have_content user2.name
       end
       expect(find('a', text: 'Assign to me')).to be_visible
 
       click_link 'Assign to me'
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user.id.to_s)
+      assignee_ids = page.all('input[name="issue[assignee_ids][]"]', visible: false)
+
+      expect(assignee_ids[0].value).to match(user2.id.to_s)
+      expect(assignee_ids[1].value).to match(user.id.to_s)
+
       page.within '.js-assignee-search' do
-        expect(page).to have_content user.name
+        expect(page).to have_content "#{user2.name} + 1 more"
       end
       expect(find('a', text: 'Assign to me', visible: false)).not_to be_visible
 
@@ -69,7 +109,7 @@
 
       page.within '.issuable-sidebar' do
         page.within '.assignee' do
-          expect(page).to have_content user.name
+          expect(page).to have_content "2 Assignees"
         end
 
         page.within '.milestone' do
@@ -141,7 +181,7 @@
     end
 
     it 'allows user to update issue' do
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user.id.to_s)
+      expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match(user.id.to_s)
       expect(find('input[name="issue[milestone_id]"]', visible: false).value).to match(milestone.id.to_s)
       expect(find('a', text: 'Assign to me', visible: false)).not_to be_visible
 
diff --git a/spec/features/issues/issue_sidebar_spec.rb b/spec/features/issues/issue_sidebar_spec.rb
index 82b80a69bed5f..e9a05f56543b0 100644
--- a/spec/features/issues/issue_sidebar_spec.rb
+++ b/spec/features/issues/issue_sidebar_spec.rb
@@ -42,6 +42,21 @@
         expect(page).to have_content(user2.name)
       end
     end
+
+    it 'assigns yourself' do
+      find('.block.assignee .dropdown-menu-toggle').click
+
+      click_button 'assign yourself'
+
+      wait_for_ajax
+
+      find('.block.assignee .edit-link').click
+
+      page.within '.dropdown-menu-user' do
+        expect(page.find('.dropdown-header')).to be_visible
+        expect(page.find('.dropdown-menu-user-link.is-active')).to have_content(user.name)
+      end
+    end
   end
 
   context 'as a allowed user' do
diff --git a/spec/features/issues/update_issues_spec.rb b/spec/features/issues/update_issues_spec.rb
index 7fa83c1fcf764..b250fa2ed3c80 100644
--- a/spec/features/issues/update_issues_spec.rb
+++ b/spec/features/issues/update_issues_spec.rb
@@ -99,7 +99,7 @@ def create_closed
   end
 
   def create_assigned
-    create(:issue, project: project, assignee: user)
+    create(:issue, project: project, assignees: [user])
   end
 
   def create_with_milestone
diff --git a/spec/features/issues_spec.rb b/spec/features/issues_spec.rb
index 81cc8513454ef..cc81303f03292 100644
--- a/spec/features/issues_spec.rb
+++ b/spec/features/issues_spec.rb
@@ -18,7 +18,7 @@
     let!(:issue) do
       create(:issue,
              author: @user,
-             assignee: @user,
+             assignees: [@user],
              project: project)
     end
 
@@ -43,7 +43,7 @@
     let!(:issue) do
       create(:issue,
              author: @user,
-             assignee: @user,
+             assignees: [@user],
              project: project)
     end
 
@@ -61,7 +61,7 @@
         expect(page).to have_content 'No assignee - assign yourself'
       end
 
-      expect(issue.reload.assignee).to be_nil
+      expect(issue.reload.assignees).to be_empty
     end
   end
 
@@ -138,7 +138,7 @@
 
   describe 'Issue info' do
     it 'excludes award_emoji from comment count' do
-      issue = create(:issue, author: @user, assignee: @user, project: project, title: 'foobar')
+      issue = create(:issue, author: @user, assignees: [@user], project: project, title: 'foobar')
       create(:award_emoji, awardable: issue)
 
       visit namespace_project_issues_path(project.namespace, project, assignee_id: @user.id)
@@ -153,14 +153,14 @@
       %w(foobar barbaz gitlab).each do |title|
         create(:issue,
                author: @user,
-               assignee: @user,
+               assignees: [@user],
                project: project,
                title: title)
       end
 
       @issue = Issue.find_by(title: 'foobar')
       @issue.milestone = create(:milestone, project: project)
-      @issue.assignee = nil
+      @issue.assignees = []
       @issue.save
     end
 
@@ -351,9 +351,9 @@
       let(:user2) { create(:user) }
 
       before do
-        foo.assignee = user2
+        foo.assignees << user2
         foo.save
-        bar.assignee = user2
+        bar.assignees << user2
         bar.save
       end
 
@@ -396,7 +396,7 @@
   end
 
   describe 'update labels from issue#show', js: true do
-    let(:issue) { create(:issue, project: project, author: @user, assignee: @user) }
+    let(:issue) { create(:issue, project: project, author: @user, assignees: [@user]) }
     let!(:label) { create(:label, project: project) }
 
     before do
@@ -415,7 +415,7 @@
   end
 
   describe 'update assignee from issue#show' do
-    let(:issue) { create(:issue, project: project, author: @user, assignee: @user) }
+    let(:issue) { create(:issue, project: project, author: @user, assignees: [@user]) }
 
     context 'by authorized user' do
       it 'allows user to select unassigned', js: true do
@@ -426,10 +426,14 @@
 
           click_link 'Edit'
           click_link 'Unassigned'
+          first('.title').click
           expect(page).to have_content 'No assignee'
         end
 
-        expect(issue.reload.assignee).to be_nil
+        # wait_for_ajax does not work with vue-resource at the moment
+        sleep 1
+
+        expect(issue.reload.assignees).to be_empty
       end
 
       it 'allows user to select an assignee', js: true do
@@ -461,14 +465,18 @@
           click_link 'Edit'
           click_link @user.name
 
-          page.within '.value' do
+          find('.dropdown-menu-toggle').click
+
+          page.within '.value .author' do
             expect(page).to have_content @user.name
           end
 
           click_link 'Edit'
           click_link @user.name
 
-          page.within '.value' do
+          find('.dropdown-menu-toggle').click
+
+          page.within '.value .assign-yourself' do
             expect(page).to have_content "No assignee"
           end
         end
@@ -487,7 +495,7 @@
         login_with guest
 
         visit namespace_project_issue_path(project.namespace, project, issue)
-        expect(page).to have_content issue.assignee.name
+        expect(page).to have_content issue.assignees.first.name
       end
     end
   end
@@ -558,7 +566,7 @@
       let(:user2) { create(:user) }
 
       before do
-        issue.assignee = user2
+        issue.assignees << user2
         issue.save
       end
     end
@@ -655,7 +663,7 @@
 
   describe 'due date' do
     context 'update due on issue#show', js: true do
-      let(:issue) { create(:issue, project: project, author: @user, assignee: @user) }
+      let(:issue) { create(:issue, project: project, author: @user, assignees: [@user]) }
 
       before do
         visit namespace_project_issue_path(project.namespace, project, issue)
diff --git a/spec/features/merge_requests/assign_issues_spec.rb b/spec/features/merge_requests/assign_issues_spec.rb
index 43cc6f2a2a7d7..ec49003772b82 100644
--- a/spec/features/merge_requests/assign_issues_spec.rb
+++ b/spec/features/merge_requests/assign_issues_spec.rb
@@ -33,7 +33,7 @@ def visit_merge_request(current_user = nil)
     end
 
     it "doesn't display if related issues are already assigned" do
-      [issue1, issue2].each { |issue| issue.update!(assignee: user) }
+      [issue1, issue2].each { |issue| issue.update!(assignees: [user]) }
 
       visit_merge_request
 
diff --git a/spec/features/milestones/show_spec.rb b/spec/features/milestones/show_spec.rb
index 40b4dc6369734..227eb04ba7282 100644
--- a/spec/features/milestones/show_spec.rb
+++ b/spec/features/milestones/show_spec.rb
@@ -5,7 +5,7 @@
   let(:project) { create(:empty_project) }
   let(:milestone) { create(:milestone, project: project) }
   let(:labels) { create_list(:label, 2, project: project) }
-  let(:issue_params) { { project: project, assignee: user, author: user, milestone: milestone, labels: labels } }
+  let(:issue_params) { { project: project, assignees: [user], author: user, milestone: milestone, labels: labels } }
 
   before do
     project.add_user(user, :developer) 
diff --git a/spec/features/projects/issuable_templates_spec.rb b/spec/features/projects/issuable_templates_spec.rb
index d28a853bbc2fc..fa5e30075e305 100644
--- a/spec/features/projects/issuable_templates_spec.rb
+++ b/spec/features/projects/issuable_templates_spec.rb
@@ -12,7 +12,7 @@
   context 'user creates an issue using templates' do
     let(:template_content) { 'this is a test "bug" template' }
     let(:longtemplate_content) { %Q(this\n\n\n\n\nis\n\n\n\n\na\n\n\n\n\nbug\n\n\n\n\ntemplate) }
-    let(:issue) { create(:issue, author: user, assignee: user, project: project) }
+    let(:issue) { create(:issue, author: user, assignees: [user], project: project) }
     let(:description_addition) { ' appending to description' }
 
     background do
@@ -72,7 +72,7 @@
   context 'user creates an issue using templates, with a prior description' do
     let(:prior_description) { 'test issue description' }
     let(:template_content) { 'this is a test "bug" template' }
-    let(:issue) { create(:issue, author: user, assignee: user, project: project) }
+    let(:issue) { create(:issue, author: user, assignees: [user], project: project) }
 
     background do
       project.repository.create_file(
diff --git a/spec/features/search_spec.rb b/spec/features/search_spec.rb
index da6388dcdf20d..498a4a5cba04c 100644
--- a/spec/features/search_spec.rb
+++ b/spec/features/search_spec.rb
@@ -5,7 +5,7 @@
 
   let(:user) { create(:user) }
   let(:project) { create(:empty_project, namespace: user.namespace) }
-  let!(:issue) { create(:issue, project: project, assignee: user) }
+  let!(:issue) { create(:issue, project: project, assignees: [user]) }
   let!(:issue2) { create(:issue, project: project, author: user) }
 
   before do
diff --git a/spec/features/unsubscribe_links_spec.rb b/spec/features/unsubscribe_links_spec.rb
index e2d9cfdd0b037..a23c4ca2b92f6 100644
--- a/spec/features/unsubscribe_links_spec.rb
+++ b/spec/features/unsubscribe_links_spec.rb
@@ -6,7 +6,7 @@
   let(:recipient) { create(:user) }
   let(:author) { create(:user) }
   let(:project) { create(:empty_project, :public) }
-  let(:params) { { title: 'A bug!', description: 'Fix it!', assignee: recipient } }
+  let(:params) { { title: 'A bug!', description: 'Fix it!', assignees: [recipient] } }
   let(:issue) { Issues::CreateService.new(project, author, params).execute }
 
   let(:mail) { ActionMailer::Base.deliveries.last }
diff --git a/spec/finders/issues_finder_spec.rb b/spec/finders/issues_finder_spec.rb
index a5f717e623389..e6bf77aee6a61 100644
--- a/spec/finders/issues_finder_spec.rb
+++ b/spec/finders/issues_finder_spec.rb
@@ -1,19 +1,19 @@
 require 'spec_helper'
 
 describe IssuesFinder do
-  set(:user) { create(:user) }
-  set(:user2) { create(:user) }
-  set(:project1) { create(:empty_project) }
-  set(:project2) { create(:empty_project) }
-  set(:milestone) { create(:milestone, project: project1) }
-  set(:label) { create(:label, project: project2) }
-  set(:issue1) { create(:issue, author: user, assignee: user, project: project1, milestone: milestone, title: 'gitlab') }
-  set(:issue2) { create(:issue, author: user, assignee: user, project: project2, description: 'gitlab') }
-  set(:issue3) { create(:issue, author: user2, assignee: user2, project: project2, title: 'tanuki', description: 'tanuki') }
+  let(:user) { create(:user) }
+  let(:user2) { create(:user) }
+  let(:project1) { create(:empty_project) }
+  let(:project2) { create(:empty_project) }
+  let(:milestone) { create(:milestone, project: project1) }
+  let(:label) { create(:label, project: project2) }
+  let(:issue1) { create(:issue, author: user, assignees: [user], project: project1, milestone: milestone, title: 'gitlab') }
+  let(:issue2) { create(:issue, author: user, assignees: [user], project: project2, description: 'gitlab') }
+  let(:issue3) { create(:issue, author: user2, assignees: [user2], project: project2, title: 'tanuki', description: 'tanuki') }
 
   describe '#execute' do
-    set(:closed_issue) { create(:issue, author: user2, assignee: user2, project: project2, state: 'closed') }
-    set(:label_link) { create(:label_link, label: label, target: issue2) }
+    let(:closed_issue) { create(:issue, author: user2, assignees: [user2], project: project2, state: 'closed') }
+    let!(:label_link) { create(:label_link, label: label, target: issue2) }
     let(:search_user) { user }
     let(:params) { {} }
     let(:issues) { described_class.new(search_user, params.reverse_merge(scope: scope, state: 'opened')).execute }
@@ -91,7 +91,7 @@
 
         before do
           milestones.each do |milestone|
-            create(:issue, project: milestone.project, milestone: milestone, author: user, assignee: user)
+            create(:issue, project: milestone.project, milestone: milestone, author: user, assignees: [user])
           end
         end
 
@@ -126,7 +126,7 @@
 
         before do
           milestones.each do |milestone|
-            create(:issue, project: milestone.project, milestone: milestone, author: user, assignee: user)
+            create(:issue, project: milestone.project, milestone: milestone, author: user, assignees: [user])
           end
         end
 
diff --git a/spec/fixtures/api/schemas/issue.json b/spec/fixtures/api/schemas/issue.json
index 21c078e0f44ec..983beb838b736 100644
--- a/spec/fixtures/api/schemas/issue.json
+++ b/spec/fixtures/api/schemas/issue.json
@@ -40,11 +40,31 @@
         "additionalProperties": false
       }
     },
+<<<<<<< HEAD
     "assignee": {
       "id": { "type": "integet" },
       "name": { "type": "string" },
       "username": { "type": "string" },
       "avatar_url": { "type": "uri" }
+=======
+    "assignees": {
+      "type": "array",
+      "items": {
+        "type": ["object", "null"],
+        "required": [
+          "id",
+          "name",
+          "username",
+          "avatar_url"
+        ],
+        "properties": {
+          "id": { "type": "integer" },
+          "name": { "type": "string" },
+          "username": { "type": "string" },
+          "avatar_url": { "type": "uri" }
+        }
+      }
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     },
     "subscribed": { "type": ["boolean", "null"] }
   },
diff --git a/spec/fixtures/api/schemas/public_api/v4/issues.json b/spec/fixtures/api/schemas/public_api/v4/issues.json
index 52199e7573420..2d1c84ee93d21 100644
--- a/spec/fixtures/api/schemas/public_api/v4/issues.json
+++ b/spec/fixtures/api/schemas/public_api/v4/issues.json
@@ -33,6 +33,21 @@
         },
         "additionalProperties": false
       },
+      "assignees": {
+        "type": "array",
+        "items": {
+          "type": ["object", "null"],
+          "properties": {
+            "name": { "type": "string" },
+            "username": { "type": "string" },
+            "id": { "type": "integer" },
+            "state": { "type": "string" },
+            "avatar_url": { "type": "uri" },
+            "web_url": { "type": "uri" }
+          },
+          "additionalProperties": false
+        }
+      },
       "assignee": {
         "type": ["object", "null"],
         "properties": {
@@ -67,7 +82,7 @@
     "required": [
       "id", "iid", "project_id", "title", "description",
       "state", "created_at", "updated_at", "labels",
-      "milestone", "assignee", "author", "user_notes_count",
+      "milestone", "assignees", "author", "user_notes_count",
       "upvotes", "downvotes", "due_date", "confidential",
       "web_url"
     ],
diff --git a/spec/helpers/issuables_helper_spec.rb b/spec/helpers/issuables_helper_spec.rb
index 93bb711f29ab3..c1ecb46aece84 100644
--- a/spec/helpers/issuables_helper_spec.rb
+++ b/spec/helpers/issuables_helper_spec.rb
@@ -4,6 +4,23 @@
   let(:label)  { build_stubbed(:label) }
   let(:label2) { build_stubbed(:label) }
 
+  describe '#users_dropdown_label' do
+    let(:user)  { build_stubbed(:user) }
+    let(:user2)  { build_stubbed(:user) }
+
+    it 'returns unassigned' do
+      expect(users_dropdown_label([])).to eq('Unassigned')
+    end
+
+    it 'returns selected user\'s name' do
+      expect(users_dropdown_label([user])).to eq(user.name)
+    end
+
+    it 'returns selected user\'s name and counter' do
+      expect(users_dropdown_label([user, user2])).to eq("#{user.name} + 1 more")
+    end
+  end
+
   describe '#issuable_labels_tooltip' do
     it 'returns label text' do
       expect(issuable_labels_tooltip([label])).to eq(label.title)
diff --git a/spec/javascripts/boards/boards_store_spec.js b/spec/javascripts/boards/boards_store_spec.js
index b55ff2f473a23..5ea160b7790c4 100644
--- a/spec/javascripts/boards/boards_store_spec.js
+++ b/spec/javascripts/boards/boards_store_spec.js
@@ -8,14 +8,14 @@
 import Vue from 'vue';
 import Cookies from 'js-cookie';
 
-require('~/lib/utils/url_utility');
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/services/board_service');
-require('~/boards/stores/boards_store');
-require('./mock_data');
+import '~/lib/utils/url_utility';
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/services/board_service';
+import '~/boards/stores/boards_store';
+import './mock_data';
 
 describe('Store', () => {
   beforeEach(() => {
@@ -212,7 +212,8 @@ describe('Store', () => {
         title: 'Testing',
         iid: 2,
         confidential: false,
-        labels: []
+        labels: [],
+        assignees: [],
       });
       const list = gl.issueBoards.BoardsStore.addList(listObj);
 
diff --git a/spec/javascripts/boards/issue_card_spec.js b/spec/javascripts/boards/issue_card_spec.js
index 1a5e9e9fd0794..b1907ac3070b4 100644
--- a/spec/javascripts/boards/issue_card_spec.js
+++ b/spec/javascripts/boards/issue_card_spec.js
@@ -1,20 +1,20 @@
-/* global ListUser */
+/* global ListAssignee */
 /* global ListLabel */
 /* global listObj */
 /* global ListIssue */
 
 import Vue from 'vue';
 
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/stores/boards_store');
-require('~/boards/components/issue_card_inner');
-require('./mock_data');
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/stores/boards_store';
+import '~/boards/components/issue_card_inner';
+import './mock_data';
 
 describe('Issue card component', () => {
-  const user = new ListUser({
+  const user = new ListAssignee({
     id: 1,
     name: 'testing 123',
     username: 'test',
@@ -40,6 +40,7 @@ describe('Issue card component', () => {
       iid: 1,
       confidential: false,
       labels: [list.label],
+      assignees: [],
     });
 
     component = new Vue({
@@ -92,12 +93,12 @@ describe('Issue card component', () => {
   it('renders confidential icon', (done) => {
     component.issue.confidential = true;
 
-    setTimeout(() => {
+    Vue.nextTick(() => {
       expect(
         component.$el.querySelector('.confidential-icon'),
       ).not.toBeNull();
       done();
-    }, 0);
+    });
   });
 
   it('renders issue ID with #', () => {
@@ -109,34 +110,32 @@ describe('Issue card component', () => {
   describe('assignee', () => {
     it('does not render assignee', () => {
       expect(
-        component.$el.querySelector('.card-assignee'),
+        component.$el.querySelector('.card-assignee .avatar'),
       ).toBeNull();
     });
 
     describe('exists', () => {
       beforeEach((done) => {
-        component.issue.assignee = user;
+        component.issue.assignees = [user];
 
-        setTimeout(() => {
-          done();
-        }, 0);
+        Vue.nextTick(() => done());
       });
 
       it('renders assignee', () => {
         expect(
-          component.$el.querySelector('.card-assignee'),
+          component.$el.querySelector('.card-assignee .avatar'),
         ).not.toBeNull();
       });
 
       it('sets title', () => {
         expect(
-          component.$el.querySelector('.card-assignee').getAttribute('title'),
+          component.$el.querySelector('.card-assignee a').getAttribute('title'),
         ).toContain(`Assigned to ${user.name}`);
       });
 
       it('sets users path', () => {
         expect(
-          component.$el.querySelector('.card-assignee').getAttribute('href'),
+          component.$el.querySelector('.card-assignee a').getAttribute('href'),
         ).toBe('/test');
       });
 
@@ -148,6 +147,75 @@ describe('Issue card component', () => {
     });
   });
 
+  describe('multiple assignees', () => {
+    beforeEach((done) => {
+      component.issue.assignees = [
+        user,
+        new ListAssignee({
+          id: 2,
+          name: 'user2',
+          username: 'user2',
+          avatar: 'test_image',
+        }),
+        new ListAssignee({
+          id: 3,
+          name: 'user3',
+          username: 'user3',
+          avatar: 'test_image',
+        }),
+        new ListAssignee({
+          id: 4,
+          name: 'user4',
+          username: 'user4',
+          avatar: 'test_image',
+        })];
+
+      Vue.nextTick(() => done());
+    });
+
+    it('renders all four assignees', () => {
+      expect(component.$el.querySelectorAll('.card-assignee .avatar').length).toEqual(4);
+    });
+
+    describe('more than four assignees', () => {
+      beforeEach((done) => {
+        component.issue.assignees.push(new ListAssignee({
+          id: 5,
+          name: 'user5',
+          username: 'user5',
+          avatar: 'test_image',
+        }));
+
+        Vue.nextTick(() => done());
+      });
+
+      it('renders more avatar counter', () => {
+        expect(component.$el.querySelector('.card-assignee .avatar-counter').innerText).toEqual('+2');
+      });
+
+      it('renders three assignees', () => {
+        expect(component.$el.querySelectorAll('.card-assignee .avatar').length).toEqual(3);
+      });
+
+      it('renders 99+ avatar counter', (done) => {
+        for (let i = 5; i < 104; i += 1) {
+          const u = new ListAssignee({
+            id: i,
+            name: 'name',
+            username: 'username',
+            avatar: 'test_image',
+          });
+          component.issue.assignees.push(u);
+        }
+
+        Vue.nextTick(() => {
+          expect(component.$el.querySelector('.card-assignee .avatar-counter').innerText).toEqual('99+');
+          done();
+        });
+      });
+    });
+  });
+
   describe('labels', () => {
     it('does not render any', () => {
       expect(
@@ -159,9 +227,7 @@ describe('Issue card component', () => {
       beforeEach((done) => {
         component.issue.addLabel(label1);
 
-        setTimeout(() => {
-          done();
-        }, 0);
+        Vue.nextTick(() => done());
       });
 
       it('does not render list label', () => {
diff --git a/spec/javascripts/boards/issue_spec.js b/spec/javascripts/boards/issue_spec.js
index c96dfe94a4a5d..cd1497bc5e615 100644
--- a/spec/javascripts/boards/issue_spec.js
+++ b/spec/javascripts/boards/issue_spec.js
@@ -2,14 +2,15 @@
 /* global BoardService */
 /* global ListIssue */
 
-require('~/lib/utils/url_utility');
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/services/board_service');
-require('~/boards/stores/boards_store');
-require('./mock_data');
+import Vue from 'vue';
+import '~/lib/utils/url_utility';
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/services/board_service';
+import '~/boards/stores/boards_store';
+import './mock_data';
 
 describe('Issue model', () => {
   let issue;
@@ -27,7 +28,13 @@ describe('Issue model', () => {
         title: 'test',
         color: 'red',
         description: 'testing'
-      }]
+      }],
+      assignees: [{
+        id: 1,
+        name: 'name',
+        username: 'username',
+        avatar_url: 'http://avatar_url',
+      }],
     });
   });
 
@@ -80,6 +87,33 @@ describe('Issue model', () => {
     expect(issue.labels.length).toBe(0);
   });
 
+  it('adds assignee', () => {
+    issue.addAssignee({
+      id: 2,
+      name: 'Bruce Wayne',
+      username: 'batman',
+      avatar_url: 'http://batman',
+    });
+
+    expect(issue.assignees.length).toBe(2);
+  });
+
+  it('finds assignee', () => {
+    const assignee = issue.findAssignee(issue.assignees[0]);
+    expect(assignee).toBeDefined();
+  });
+
+  it('removes assignee', () => {
+    const assignee = issue.findAssignee(issue.assignees[0]);
+    issue.removeAssignee(assignee);
+    expect(issue.assignees.length).toBe(0);
+  });
+
+  it('removes all assignees', () => {
+    issue.removeAllAssignees();
+    expect(issue.assignees.length).toBe(0);
+  });
+
   it('sets position to infinity if no position is stored', () => {
     expect(issue.position).toBe(Infinity);
   });
@@ -90,9 +124,31 @@ describe('Issue model', () => {
       iid: 1,
       confidential: false,
       relative_position: 1,
-      labels: []
+      labels: [],
+      assignees: [],
     });
 
     expect(relativePositionIssue.position).toBe(1);
   });
+
+  describe('update', () => {
+    it('passes assignee ids when there are assignees', (done) => {
+      spyOn(Vue.http, 'patch').and.callFake((url, data) => {
+        expect(data.issue.assignee_ids).toEqual([1]);
+        done();
+      });
+
+      issue.update('url');
+    });
+
+    it('passes assignee ids of [0] when there are no assignees', (done) => {
+      spyOn(Vue.http, 'patch').and.callFake((url, data) => {
+        expect(data.issue.assignee_ids).toEqual([0]);
+        done();
+      });
+
+      issue.removeAllAssignees();
+      issue.update('url');
+    });
+  });
 });
diff --git a/spec/javascripts/boards/list_spec.js b/spec/javascripts/boards/list_spec.js
index 24a2da9f6b629..88a0e6855f4fc 100644
--- a/spec/javascripts/boards/list_spec.js
+++ b/spec/javascripts/boards/list_spec.js
@@ -8,14 +8,14 @@
 
 import Vue from 'vue';
 
-require('~/lib/utils/url_utility');
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/services/board_service');
-require('~/boards/stores/boards_store');
-require('./mock_data');
+import '~/lib/utils/url_utility';
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/services/board_service';
+import '~/boards/stores/boards_store';
+import './mock_data';
 
 describe('List model', () => {
   let list;
@@ -94,7 +94,8 @@ describe('List model', () => {
       title: 'Testing',
       iid: _.random(10000),
       confidential: false,
-      labels: [list.label, listDup.label]
+      labels: [list.label, listDup.label],
+      assignees: [],
     });
 
     list.issues.push(issue);
diff --git a/spec/javascripts/boards/mock_data.js b/spec/javascripts/boards/mock_data.js
index a4fa694eebec2..a64c3964ee3e3 100644
--- a/spec/javascripts/boards/mock_data.js
+++ b/spec/javascripts/boards/mock_data.js
@@ -33,7 +33,8 @@ const BoardsMockData = {
         title: 'Testing',
         iid: 1,
         confidential: false,
-        labels: []
+        labels: [],
+        assignees: [],
       }],
       size: 1
     }
diff --git a/spec/javascripts/boards/modal_store_spec.js b/spec/javascripts/boards/modal_store_spec.js
index 80db816aff852..32e6d04df9fed 100644
--- a/spec/javascripts/boards/modal_store_spec.js
+++ b/spec/javascripts/boards/modal_store_spec.js
@@ -1,10 +1,10 @@
 /* global ListIssue */
 
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/stores/modal_store');
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/stores/modal_store';
 
 describe('Modal store', () => {
   let issue;
@@ -21,12 +21,14 @@ describe('Modal store', () => {
       iid: 1,
       confidential: false,
       labels: [],
+      assignees: [],
     });
     issue2 = new ListIssue({
       title: 'Testing',
       iid: 2,
       confidential: false,
       labels: [],
+      assignees: [],
     });
     Store.store.issues.push(issue);
     Store.store.issues.push(issue2);
diff --git a/spec/javascripts/issuable_time_tracker_spec.js b/spec/javascripts/issuable_time_tracker_spec.js
index 0a830f25e29ed..8ff93c4f918d2 100644
--- a/spec/javascripts/issuable_time_tracker_spec.js
+++ b/spec/javascripts/issuable_time_tracker_spec.js
@@ -2,7 +2,7 @@
 
 import Vue from 'vue';
 
-require('~/issuable/time_tracking/components/time_tracker');
+import timeTracker from '~/sidebar/components/time_tracking/time_tracker';
 
 function initTimeTrackingComponent(opts) {
   setFixtures(`
@@ -16,187 +16,185 @@ function initTimeTrackingComponent(opts) {
     time_spent: opts.timeSpent,
     human_time_estimate: opts.timeEstimateHumanReadable,
     human_time_spent: opts.timeSpentHumanReadable,
-    docsUrl: '/help/workflow/time_tracking.md',
+    rootPath: '/',
   };
 
-  const TimeTrackingComponent = Vue.component('issuable-time-tracker');
+  const TimeTrackingComponent = Vue.extend(timeTracker);
   this.timeTracker = new TimeTrackingComponent({
     el: '#mock-container',
     propsData: this.initialData,
   });
 }
 
-((gl) => {
-  describe('Issuable Time Tracker', function() {
-    describe('Initialization', function() {
-      beforeEach(function() {
-        initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 5000, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '1h 23m' });
-      });
+describe('Issuable Time Tracker', function() {
+  describe('Initialization', function() {
+    beforeEach(function() {
+      initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 5000, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '1h 23m' });
+    });
 
-      it('should return something defined', function() {
-        expect(this.timeTracker).toBeDefined();
-      });
+    it('should return something defined', function() {
+      expect(this.timeTracker).toBeDefined();
+    });
 
-      it ('should correctly set timeEstimate', function(done) {
-        Vue.nextTick(() => {
-          expect(this.timeTracker.timeEstimate).toBe(this.initialData.time_estimate);
-          done();
-        });
+    it ('should correctly set timeEstimate', function(done) {
+      Vue.nextTick(() => {
+        expect(this.timeTracker.timeEstimate).toBe(this.initialData.time_estimate);
+        done();
       });
-      it ('should correctly set time_spent', function(done) {
-        Vue.nextTick(() => {
-          expect(this.timeTracker.timeSpent).toBe(this.initialData.time_spent);
-          done();
-        });
+    });
+    it ('should correctly set time_spent', function(done) {
+      Vue.nextTick(() => {
+        expect(this.timeTracker.timeSpent).toBe(this.initialData.time_spent);
+        done();
       });
     });
+  });
 
-    describe('Content Display', function() {
-      describe('Panes', function() {
-        describe('Comparison pane', function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 5000, timeEstimateHumanReadable: '', timeSpentHumanReadable: '' });
+  describe('Content Display', function() {
+    describe('Panes', function() {
+      describe('Comparison pane', function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 5000, timeEstimateHumanReadable: '', timeSpentHumanReadable: '' });
+        });
+
+        it('should show the "Comparison" pane when timeEstimate and time_spent are truthy', function(done) {
+          Vue.nextTick(() => {
+            const $comparisonPane = this.timeTracker.$el.querySelector('.time-tracking-comparison-pane');
+            expect(this.timeTracker.showComparisonState).toBe(true);
+            done();
           });
+        });
 
-          it('should show the "Comparison" pane when timeEstimate and time_spent are truthy', function(done) {
+        describe('Remaining meter', function() {
+          it('should display the remaining meter with the correct width', function(done) {
             Vue.nextTick(() => {
-              const $comparisonPane = this.timeTracker.$el.querySelector('.time-tracking-comparison-pane');
-              expect(this.timeTracker.showComparisonState).toBe(true);
+              const meterWidth = this.timeTracker.$el.querySelector('.time-tracking-comparison-pane .meter-fill').style.width;
+              const correctWidth = '5%';
+
+              expect(meterWidth).toBe(correctWidth);
               done();
-            });
+            })
           });
 
-          describe('Remaining meter', function() {
-            it('should display the remaining meter with the correct width', function(done) {
-              Vue.nextTick(() => {
-                const meterWidth = this.timeTracker.$el.querySelector('.time-tracking-comparison-pane .meter-fill').style.width;
-                const correctWidth = '5%';
-
-                expect(meterWidth).toBe(correctWidth);
-                done();
-              })
-            });
-
-            it('should display the remaining meter with the correct background color when within estimate', function(done) {
-              Vue.nextTick(() => {
-                const styledMeter = $(this.timeTracker.$el).find('.time-tracking-comparison-pane .within_estimate .meter-fill');
-                expect(styledMeter.length).toBe(1);
-                done()
-              });
+          it('should display the remaining meter with the correct background color when within estimate', function(done) {
+            Vue.nextTick(() => {
+              const styledMeter = $(this.timeTracker.$el).find('.time-tracking-comparison-pane .within_estimate .meter-fill');
+              expect(styledMeter.length).toBe(1);
+              done()
             });
+          });
 
-            it('should display the remaining meter with the correct background color when over estimate', function(done) {
-              this.timeTracker.time_estimate = 100000;
-              this.timeTracker.time_spent = 20000000;
-              Vue.nextTick(() => {
-                const styledMeter = $(this.timeTracker.$el).find('.time-tracking-comparison-pane .over_estimate .meter-fill');
-                expect(styledMeter.length).toBe(1);
-                done();
-              });
+          it('should display the remaining meter with the correct background color when over estimate', function(done) {
+            this.timeTracker.time_estimate = 100000;
+            this.timeTracker.time_spent = 20000000;
+            Vue.nextTick(() => {
+              const styledMeter = $(this.timeTracker.$el).find('.time-tracking-comparison-pane .over_estimate .meter-fill');
+              expect(styledMeter.length).toBe(1);
+              done();
             });
           });
         });
+      });
 
-        describe("Estimate only pane", function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 0, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '' });
-          });
+      describe("Estimate only pane", function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 0, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '' });
+        });
 
-          it('should display the human readable version of time estimated', function(done) {
-            Vue.nextTick(() => {
-              const estimateText = this.timeTracker.$el.querySelector('.time-tracking-estimate-only-pane').innerText;
-              const correctText = 'Estimated: 2h 46m';
+        it('should display the human readable version of time estimated', function(done) {
+          Vue.nextTick(() => {
+            const estimateText = this.timeTracker.$el.querySelector('.time-tracking-estimate-only-pane').innerText;
+            const correctText = 'Estimated: 2h 46m';
 
-              expect(estimateText).toBe(correctText);
-              done();
-            });
+            expect(estimateText).toBe(correctText);
+            done();
           });
         });
+      });
 
-        describe('Spent only pane', function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 5000, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '1h 23m' });
-          });
+      describe('Spent only pane', function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 5000, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '1h 23m' });
+        });
 
-          it('should display the human readable version of time spent', function(done) {
-            Vue.nextTick(() => {
-              const spentText = this.timeTracker.$el.querySelector('.time-tracking-spend-only-pane').innerText;
-              const correctText = 'Spent: 1h 23m';
+        it('should display the human readable version of time spent', function(done) {
+          Vue.nextTick(() => {
+            const spentText = this.timeTracker.$el.querySelector('.time-tracking-spend-only-pane').innerText;
+            const correctText = 'Spent: 1h 23m';
 
-              expect(spentText).toBe(correctText);
-              done();
-            });
+            expect(spentText).toBe(correctText);
+            done();
           });
         });
+      });
 
-        describe('No time tracking pane', function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 0, timeEstimateHumanReadable: 0, timeSpentHumanReadable: 0 });
-          });
+      describe('No time tracking pane', function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 0, timeEstimateHumanReadable: '', timeSpentHumanReadable: '' });
+        });
 
-          it('should only show the "No time tracking" pane when both timeEstimate and time_spent are falsey', function(done) {
-            Vue.nextTick(() => {
-              const $noTrackingPane = this.timeTracker.$el.querySelector('.time-tracking-no-tracking-pane');
-              const noTrackingText =$noTrackingPane.innerText;
-              const correctText = 'No estimate or time spent';
+        it('should only show the "No time tracking" pane when both timeEstimate and time_spent are falsey', function(done) {
+          Vue.nextTick(() => {
+            const $noTrackingPane = this.timeTracker.$el.querySelector('.time-tracking-no-tracking-pane');
+            const noTrackingText =$noTrackingPane.innerText;
+            const correctText = 'No estimate or time spent';
 
-              expect(this.timeTracker.showNoTimeTrackingState).toBe(true);
-              expect($noTrackingPane).toBeVisible();
-              expect(noTrackingText).toBe(correctText);
-              done();
-            });
+            expect(this.timeTracker.showNoTimeTrackingState).toBe(true);
+            expect($noTrackingPane).toBeVisible();
+            expect(noTrackingText).toBe(correctText);
+            done();
           });
         });
+      });
 
-        describe("Help pane", function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 0 });
-          });
+      describe("Help pane", function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 0 });
+        });
 
-          it('should not show the "Help" pane by default', function(done) {
-            Vue.nextTick(() => {
-              const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
+        it('should not show the "Help" pane by default', function(done) {
+          Vue.nextTick(() => {
+            const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
 
-              expect(this.timeTracker.showHelpState).toBe(false);
-              expect($helpPane).toBeNull();
-              done();
-            });
+            expect(this.timeTracker.showHelpState).toBe(false);
+            expect($helpPane).toBeNull();
+            done();
           });
+        });
 
-          it('should show the "Help" pane when help button is clicked', function(done) {
-            Vue.nextTick(() => {
-              $(this.timeTracker.$el).find('.help-button').click();
+        it('should show the "Help" pane when help button is clicked', function(done) {
+          Vue.nextTick(() => {
+            $(this.timeTracker.$el).find('.help-button').click();
 
-              setTimeout(() => {
-                const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
-                expect(this.timeTracker.showHelpState).toBe(true);
-                expect($helpPane).toBeVisible();
-                done();
-              }, 10);
-            });
+            setTimeout(() => {
+              const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
+              expect(this.timeTracker.showHelpState).toBe(true);
+              expect($helpPane).toBeVisible();
+              done();
+            }, 10);
           });
+        });
 
-          it('should not show the "Help" pane when help button is clicked and then closed', function(done) {
-            Vue.nextTick(() => {
-              $(this.timeTracker.$el).find('.help-button').click();
+        it('should not show the "Help" pane when help button is clicked and then closed', function(done) {
+          Vue.nextTick(() => {
+            $(this.timeTracker.$el).find('.help-button').click();
 
-              setTimeout(() => {
+            setTimeout(() => {
 
-                $(this.timeTracker.$el).find('.close-help-button').click();
+              $(this.timeTracker.$el).find('.close-help-button').click();
 
-                setTimeout(() => {
-                  const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
+              setTimeout(() => {
+                const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
 
-                  expect(this.timeTracker.showHelpState).toBe(false);
-                  expect($helpPane).toBeNull();
+                expect(this.timeTracker.showHelpState).toBe(false);
+                expect($helpPane).toBeNull();
 
-                  done();
-                }, 1000);
+                done();
               }, 1000);
-            });
+            }, 1000);
           });
         });
       });
     });
   });
-})(window.gl || (window.gl = {}));
+});
diff --git a/spec/javascripts/subbable_resource_spec.js b/spec/javascripts/subbable_resource_spec.js
deleted file mode 100644
index 454386697f54f..0000000000000
--- a/spec/javascripts/subbable_resource_spec.js
+++ /dev/null
@@ -1,63 +0,0 @@
-/* eslint-disable max-len, arrow-parens, comma-dangle */
-
-require('~/subbable_resource');
-
-/*
-* Test that each rest verb calls the publish and subscribe function and passes the correct value back
-*
-*
-* */
-((global) => {
-  describe('Subbable Resource', function () {
-    describe('PubSub', function () {
-      beforeEach(function () {
-        this.MockResource = new global.SubbableResource('https://example.com');
-      });
-      it('should successfully add a single subscriber', function () {
-        const callback = () => {};
-        this.MockResource.subscribe(callback);
-
-        expect(this.MockResource.subscribers.length).toBe(1);
-        expect(this.MockResource.subscribers[0]).toBe(callback);
-      });
-
-      it('should successfully add multiple subscribers', function () {
-        const callbackOne = () => {};
-        const callbackTwo = () => {};
-        const callbackThree = () => {};
-
-        this.MockResource.subscribe(callbackOne);
-        this.MockResource.subscribe(callbackTwo);
-        this.MockResource.subscribe(callbackThree);
-
-        expect(this.MockResource.subscribers.length).toBe(3);
-      });
-
-      it('should successfully publish an update to a single subscriber', function () {
-        const state = { myprop: 1 };
-
-        const callbacks = {
-          one: (data) => expect(data.myprop).toBe(2),
-          two: (data) => expect(data.myprop).toBe(2),
-          three: (data) => expect(data.myprop).toBe(2)
-        };
-
-        const spyOne = spyOn(callbacks, 'one');
-        const spyTwo = spyOn(callbacks, 'two');
-        const spyThree = spyOn(callbacks, 'three');
-
-        this.MockResource.subscribe(callbacks.one);
-        this.MockResource.subscribe(callbacks.two);
-        this.MockResource.subscribe(callbacks.three);
-
-        state.myprop += 1;
-
-        this.MockResource.publish(state);
-
-        expect(spyOne).toHaveBeenCalled();
-        expect(spyTwo).toHaveBeenCalled();
-        expect(spyThree).toHaveBeenCalled();
-      });
-    });
-  });
-})(window.gl || (window.gl = {}));
diff --git a/spec/lib/banzai/filter/redactor_filter_spec.rb b/spec/lib/banzai/filter/redactor_filter_spec.rb
index 8a6fe1ad6a39d..7c4a0f32c7b73 100644
--- a/spec/lib/banzai/filter/redactor_filter_spec.rb
+++ b/spec/lib/banzai/filter/redactor_filter_spec.rb
@@ -113,7 +113,7 @@ def reference_link(data)
       it 'allows references for assignee' do
         assignee = create(:user)
         project = create(:empty_project, :public)
-        issue = create(:issue, :confidential, project: project, assignee: assignee)
+        issue = create(:issue, :confidential, project: project, assignees: [assignee])
 
         link = reference_link(project: project.id, issue: issue.id, reference_type: 'issue')
         doc = filter(link, current_user: assignee)
diff --git a/spec/lib/gitlab/github_import/issue_formatter_spec.rb b/spec/lib/gitlab/github_import/issue_formatter_spec.rb
index f34d09f2c1d9a..a4089592cf215 100644
--- a/spec/lib/gitlab/github_import/issue_formatter_spec.rb
+++ b/spec/lib/gitlab/github_import/issue_formatter_spec.rb
@@ -43,7 +43,7 @@
           description: "*Created by: octocat*\n\nI'm having a problem with this.",
           state: 'opened',
           author_id: project.creator_id,
-          assignee_id: nil,
+          assignee_ids: [],
           created_at: created_at,
           updated_at: updated_at
         }
@@ -64,7 +64,7 @@
           description: "*Created by: octocat*\n\nI'm having a problem with this.",
           state: 'closed',
           author_id: project.creator_id,
-          assignee_id: nil,
+          assignee_ids: [],
           created_at: created_at,
           updated_at: updated_at
         }
@@ -77,19 +77,19 @@
       let(:raw_data) { double(base_data.merge(assignee: octocat)) }
 
       it 'returns nil as assignee_id when is not a GitLab user' do
-        expect(issue.attributes.fetch(:assignee_id)).to be_nil
+        expect(issue.attributes.fetch(:assignee_ids)).to be_empty
       end
 
       it 'returns GitLab user id associated with GitHub id as assignee_id' do
         gl_user = create(:omniauth_user, extern_uid: octocat.id, provider: 'github')
 
-        expect(issue.attributes.fetch(:assignee_id)).to eq gl_user.id
+        expect(issue.attributes.fetch(:assignee_ids)).to eq [gl_user.id]
       end
 
       it 'returns GitLab user id associated with GitHub email as assignee_id' do
         gl_user = create(:user, email: octocat.email)
 
-        expect(issue.attributes.fetch(:assignee_id)).to eq gl_user.id
+        expect(issue.attributes.fetch(:assignee_ids)).to eq [gl_user.id]
       end
     end
 
diff --git a/spec/lib/gitlab/google_code_import/importer_spec.rb b/spec/lib/gitlab/google_code_import/importer_spec.rb
index ccaa88a5c798f..622a0f513f43a 100644
--- a/spec/lib/gitlab/google_code_import/importer_spec.rb
+++ b/spec/lib/gitlab/google_code_import/importer_spec.rb
@@ -49,7 +49,7 @@
       expect(issue).not_to be_nil
       expect(issue.iid).to eq(169)
       expect(issue.author).to eq(project.creator)
-      expect(issue.assignee).to eq(mapped_user)
+      expect(issue.assignees).to eq([mapped_user])
       expect(issue.state).to eq("closed")
       expect(issue.label_names).to include("Priority: Medium")
       expect(issue.label_names).to include("Status: Fixed")
diff --git a/spec/lib/gitlab/import_export/all_models.yml b/spec/lib/gitlab/import_export/all_models.yml
index 0abf89d060cec..cd2fa27bb9f88 100644
--- a/spec/lib/gitlab/import_export/all_models.yml
+++ b/spec/lib/gitlab/import_export/all_models.yml
@@ -3,7 +3,7 @@ issues:
 - subscriptions
 - award_emoji
 - author
-- assignee
+- assignees
 - updated_by
 - milestone
 - notes
@@ -16,6 +16,7 @@ issues:
 - merge_requests_closing_issues
 - metrics
 - timelogs
+- issue_assignees
 events:
 - author
 - project
diff --git a/spec/lib/gitlab/import_export/project_tree_saver_spec.rb b/spec/lib/gitlab/import_export/project_tree_saver_spec.rb
index 1035428b2e7d3..5aeb29b7fecb7 100644
--- a/spec/lib/gitlab/import_export/project_tree_saver_spec.rb
+++ b/spec/lib/gitlab/import_export/project_tree_saver_spec.rb
@@ -203,7 +203,7 @@
   end
 
   def setup_project
-    issue = create(:issue, assignee: user)
+    issue = create(:issue, assignees: [user])
     snippet = create(:project_snippet)
     release = create(:release)
     group = create(:group)
diff --git a/spec/lib/gitlab/project_search_results_spec.rb b/spec/lib/gitlab/project_search_results_spec.rb
index e0ebea63eb40e..a7c8e7f1f5708 100644
--- a/spec/lib/gitlab/project_search_results_spec.rb
+++ b/spec/lib/gitlab/project_search_results_spec.rb
@@ -89,7 +89,7 @@
     let(:project) { create(:empty_project, :internal) }
     let!(:issue) { create(:issue, project: project, title: 'Issue 1') }
     let!(:security_issue_1) { create(:issue, :confidential, project: project, title: 'Security issue 1', author: author) }
-    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignee: assignee) }
+    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignees: [assignee]) }
 
     it 'does not list project confidential issues for non project members' do
       results = described_class.new(non_member, project, query)
diff --git a/spec/lib/gitlab/search_results_spec.rb b/spec/lib/gitlab/search_results_spec.rb
index 847fb97740015..31c3cd4d53c8e 100644
--- a/spec/lib/gitlab/search_results_spec.rb
+++ b/spec/lib/gitlab/search_results_spec.rb
@@ -72,9 +72,9 @@
     let(:admin) { create(:admin) }
     let!(:issue) { create(:issue, project: project_1, title: 'Issue 1') }
     let!(:security_issue_1) { create(:issue, :confidential, project: project_1, title: 'Security issue 1', author: author) }
-    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project_1, assignee: assignee) }
+    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project_1, assignees: [assignee]) }
     let!(:security_issue_3) { create(:issue, :confidential, project: project_2, title: 'Security issue 3', author: author) }
-    let!(:security_issue_4) { create(:issue, :confidential, project: project_3, title: 'Security issue 4', assignee: assignee) }
+    let!(:security_issue_4) { create(:issue, :confidential, project: project_3, title: 'Security issue 4', assignees: [assignee]) }
     let!(:security_issue_5) { create(:issue, :confidential, project: project_4, title: 'Security issue 5') }
 
     it 'does not list confidential issues for non project members' do
diff --git a/spec/mailers/notify_spec.rb b/spec/mailers/notify_spec.rb
index 9f12e40d808e5..1e6260270fe31 100644
--- a/spec/mailers/notify_spec.rb
+++ b/spec/mailers/notify_spec.rb
@@ -36,11 +36,11 @@ def have_referable_subject(referable, reply: false)
       end
 
       context 'for issues' do
-        let(:issue) { create(:issue, author: current_user, assignee: assignee, project: project) }
-        let(:issue_with_description) { create(:issue, author: current_user, assignee: assignee, project: project, description: 'My awesome description') }
+        let(:issue) { create(:issue, author: current_user, assignees: [assignee], project: project) }
+        let(:issue_with_description) { create(:issue, author: current_user, assignees: [assignee], project: project, description: 'My awesome description') }
 
         describe 'that are new' do
-          subject { described_class.new_issue_email(issue.assignee_id, issue.id) }
+          subject { described_class.new_issue_email(issue.assignees.first.id, issue.id) }
 
           it_behaves_like 'an assignee email'
           it_behaves_like 'an email starting a new thread with reply-by-email enabled' do
@@ -69,7 +69,7 @@ def have_referable_subject(referable, reply: false)
         end
 
         describe 'that are new with a description' do
-          subject { described_class.new_issue_email(issue_with_description.assignee_id, issue_with_description.id) }
+          subject { described_class.new_issue_email(issue_with_description.assignees.first.id, issue_with_description.id) }
 
           it_behaves_like 'it should show Gmail Actions View Issue link'
 
@@ -79,7 +79,7 @@ def have_referable_subject(referable, reply: false)
         end
 
         describe 'that have been reassigned' do
-          subject { described_class.reassigned_issue_email(recipient.id, issue.id, previous_assignee.id, current_user.id) }
+          subject { described_class.reassigned_issue_email(recipient.id, issue.id, [previous_assignee.id], current_user.id) }
 
           it_behaves_like 'a multiple recipients email'
           it_behaves_like 'an answer to an existing thread with reply-by-email enabled' do
diff --git a/spec/models/concerns/issuable_spec.rb b/spec/models/concerns/issuable_spec.rb
index 3ecba2e96870b..27890e33b49d8 100644
--- a/spec/models/concerns/issuable_spec.rb
+++ b/spec/models/concerns/issuable_spec.rb
@@ -10,7 +10,6 @@
 
     it { is_expected.to belong_to(:project) }
     it { is_expected.to belong_to(:author) }
-    it { is_expected.to belong_to(:assignee) }
     it { is_expected.to have_many(:notes).dependent(:destroy) }
     it { is_expected.to have_many(:todos).dependent(:destroy) }
 
@@ -66,60 +65,6 @@
     end
   end
 
-  describe 'assignee_name' do
-    it 'is delegated to assignee' do
-      issue.update!(assignee: create(:user))
-
-      expect(issue.assignee_name).to eq issue.assignee.name
-    end
-
-    it 'returns nil when assignee is nil' do
-      issue.assignee_id = nil
-      issue.save(validate: false)
-
-      expect(issue.assignee_name).to eq nil
-    end
-  end
-
-  describe "before_save" do
-    describe "#update_cache_counts" do
-      context "when previous assignee exists" do
-        before do
-          assignee = create(:user)
-          issue.project.team << [assignee, :developer]
-          issue.update(assignee: assignee)
-        end
-
-        it "updates cache counts for new assignee" do
-          user = create(:user)
-
-          expect(user).to receive(:update_cache_counts)
-
-          issue.update(assignee: user)
-        end
-
-        it "updates cache counts for previous assignee" do
-          old_assignee = issue.assignee
-          allow(User).to receive(:find_by_id).with(old_assignee.id).and_return(old_assignee)
-
-          expect(old_assignee).to receive(:update_cache_counts)
-
-          issue.update(assignee: nil)
-        end
-      end
-
-      context "when previous assignee does not exist" do
-        before{ issue.update(assignee: nil) }
-
-        it "updates cache count for the new assignee" do
-          expect_any_instance_of(User).to receive(:update_cache_counts)
-
-          issue.update(assignee: user)
-        end
-      end
-    end
-  end
-
   describe ".search" do
     let!(:searchable_issue) { create(:issue, title: "Searchable issue") }
 
@@ -307,7 +252,20 @@
     end
 
     context "issue is assigned" do
-      before { issue.update_attribute(:assignee, user) }
+      before { issue.assignees << user }
+
+      it "returns correct hook data" do
+        expect(data[:assignees].first).to eq(user.hook_attrs)
+      end
+    end
+
+    context "merge_request is assigned" do
+      let(:merge_request) { create(:merge_request) }
+      let(:data) { merge_request.to_hook_data(user) }
+
+      before do
+        merge_request.update_attribute(:assignee, user)
+      end
 
       it "returns correct hook data" do
         expect(data[:object_attributes]['assignee_id']).to eq(user.id)
@@ -329,24 +287,6 @@
     include_examples 'deprecated repository hook data'
   end
 
-  describe '#card_attributes' do
-    it 'includes the author name' do
-      allow(issue).to receive(:author).and_return(double(name: 'Robert'))
-      allow(issue).to receive(:assignee).and_return(nil)
-
-      expect(issue.card_attributes).
-        to eq({ 'Author' => 'Robert', 'Assignee' => nil })
-    end
-
-    it 'includes the assignee name' do
-      allow(issue).to receive(:author).and_return(double(name: 'Robert'))
-      allow(issue).to receive(:assignee).and_return(double(name: 'Douwe'))
-
-      expect(issue.card_attributes).
-        to eq({ 'Author' => 'Robert', 'Assignee' => 'Douwe' })
-    end
-  end
-
   describe '#labels_array' do
     let(:project) { create(:empty_project) }
     let(:bug) { create(:label, project: project, title: 'bug') }
@@ -475,27 +415,6 @@ def create_issue(milestone, labels)
     end
   end
 
-  describe '#assignee_or_author?' do
-    let(:user) { build(:user, id: 1) }
-    let(:issue) { build(:issue) }
-
-    it 'returns true for a user that is assigned to an issue' do
-      issue.assignee = user
-
-      expect(issue.assignee_or_author?(user)).to eq(true)
-    end
-
-    it 'returns true for a user that is the author of an issue' do
-      issue.author = user
-
-      expect(issue.assignee_or_author?(user)).to eq(true)
-    end
-
-    it 'returns false for a user that is not the assignee or author' do
-      expect(issue.assignee_or_author?(user)).to eq(false)
-    end
-  end
-
   describe '#spend_time' do
     let(:user) { create(:user) }
     let(:issue) { create(:issue) }
diff --git a/spec/models/concerns/milestoneish_spec.rb b/spec/models/concerns/milestoneish_spec.rb
index 68e4c0a522bc8..675b730c5575a 100644
--- a/spec/models/concerns/milestoneish_spec.rb
+++ b/spec/models/concerns/milestoneish_spec.rb
@@ -11,13 +11,13 @@
   let(:milestone) { create(:milestone, project: project) }
   let!(:issue) { create(:issue, project: project, milestone: milestone) }
   let!(:security_issue_1) { create(:issue, :confidential, project: project, author: author, milestone: milestone) }
-  let!(:security_issue_2) { create(:issue, :confidential, project: project, assignee: assignee, milestone: milestone) }
+  let!(:security_issue_2) { create(:issue, :confidential, project: project, assignees: [assignee], milestone: milestone) }
   let!(:closed_issue_1) { create(:issue, :closed, project: project, milestone: milestone) }
   let!(:closed_issue_2) { create(:issue, :closed, project: project, milestone: milestone) }
   let!(:closed_security_issue_1) { create(:issue, :confidential, :closed, project: project, author: author, milestone: milestone) }
-  let!(:closed_security_issue_2) { create(:issue, :confidential, :closed, project: project, assignee: assignee, milestone: milestone) }
+  let!(:closed_security_issue_2) { create(:issue, :confidential, :closed, project: project, assignees: [assignee], milestone: milestone) }
   let!(:closed_security_issue_3) { create(:issue, :confidential, :closed, project: project, author: author, milestone: milestone) }
-  let!(:closed_security_issue_4) { create(:issue, :confidential, :closed, project: project, assignee: assignee, milestone: milestone) }
+  let!(:closed_security_issue_4) { create(:issue, :confidential, :closed, project: project, assignees: [assignee], milestone: milestone) }
   let!(:merge_request) { create(:merge_request, source_project: project, target_project: project, milestone: milestone) }
 
   before do
diff --git a/spec/models/event_spec.rb b/spec/models/event_spec.rb
index 8c90a538f5723..e5954d80f00aa 100644
--- a/spec/models/event_spec.rb
+++ b/spec/models/event_spec.rb
@@ -92,8 +92,8 @@
     let(:author) { create(:author) }
     let(:assignee) { create(:user) }
     let(:admin) { create(:admin) }
-    let(:issue) { create(:issue, project: project, author: author, assignee: assignee) }
-    let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee) }
+    let(:issue) { create(:issue, project: project, author: author, assignees: [assignee]) }
+    let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee]) }
     let(:note_on_commit) { create(:note_on_commit, project: project) }
     let(:note_on_issue) { create(:note_on_issue, noteable: issue, project: project) }
     let(:note_on_confidential_issue) { create(:note_on_issue, noteable: confidential_issue, project: project) }
diff --git a/spec/models/issue_collection_spec.rb b/spec/models/issue_collection_spec.rb
index d8aed25c041f5..93c2c538e1056 100644
--- a/spec/models/issue_collection_spec.rb
+++ b/spec/models/issue_collection_spec.rb
@@ -28,7 +28,7 @@
       end
 
       it 'returns the issues the user is assigned to' do
-        issue1.assignee = user
+        issue1.assignees << user
 
         expect(collection.updatable_by_user(user)).to eq([issue1])
       end
diff --git a/spec/models/issue_spec.rb b/spec/models/issue_spec.rb
index 8748b98a4e357..725f5c2311fd3 100644
--- a/spec/models/issue_spec.rb
+++ b/spec/models/issue_spec.rb
@@ -3,6 +3,7 @@
 describe Issue, models: true do
   describe "Associations" do
     it { is_expected.to belong_to(:milestone) }
+    it { is_expected.to have_many(:assignees) }
   end
 
   describe 'modules' do
@@ -37,6 +38,64 @@
     end
   end
 
+  describe "before_save" do
+    describe "#update_cache_counts when an issue is reassigned" do
+      let(:issue) { create(:issue) }
+      let(:assignee) { create(:user) }
+
+      context "when previous assignee exists" do
+        before do
+          issue.project.team << [assignee, :developer]
+          issue.assignees << assignee
+        end
+
+        it "updates cache counts for new assignee" do
+          user = create(:user)
+
+          expect(user).to receive(:update_cache_counts)
+
+          issue.assignees << user
+        end
+
+        it "updates cache counts for previous assignee" do
+          issue.assignees.first
+
+          expect_any_instance_of(User).to receive(:update_cache_counts)
+
+          issue.assignees.destroy_all
+        end
+      end
+
+      context "when previous assignee does not exist" do
+        it "updates cache count for the new assignee" do
+          issue.assignees = []
+
+          expect_any_instance_of(User).to receive(:update_cache_counts)
+
+          issue.assignees << assignee
+        end
+      end
+    end
+  end
+
+  describe '#card_attributes' do
+    it 'includes the author name' do
+      allow(subject).to receive(:author).and_return(double(name: 'Robert'))
+      allow(subject).to receive(:assignees).and_return([])
+
+      expect(subject.card_attributes).
+        to eq({ 'Author' => 'Robert', 'Assignee' => '' })
+    end
+
+    it 'includes the assignee name' do
+      allow(subject).to receive(:author).and_return(double(name: 'Robert'))
+      allow(subject).to receive(:assignees).and_return([double(name: 'Douwe')])
+
+      expect(subject.card_attributes).
+        to eq({ 'Author' => 'Robert', 'Assignee' => 'Douwe' })
+    end
+  end
+
   describe '#closed_at' do
     after do
       Timecop.return
@@ -124,13 +183,24 @@
     end
   end
 
-  describe '#is_being_reassigned?' do
-    it 'returns true if the issue assignee has changed' do
-      subject.assignee = create(:user)
-      expect(subject.is_being_reassigned?).to be_truthy
+  describe '#assignee_or_author?' do
+    let(:user) { create(:user) }
+    let(:issue) { create(:issue) }
+
+    it 'returns true for a user that is assigned to an issue' do
+      issue.assignees << user
+
+      expect(issue.assignee_or_author?(user)).to be_truthy
     end
-    it 'returns false if the issue assignee has not changed' do
-      expect(subject.is_being_reassigned?).to be_falsey
+
+    it 'returns true for a user that is the author of an issue' do
+      issue.update(author: user)
+
+      expect(issue.assignee_or_author?(user)).to be_truthy
+    end
+
+    it 'returns false for a user that is not the assignee or author' do
+      expect(issue.assignee_or_author?(user)).to be_falsey
     end
   end
 
@@ -383,14 +453,14 @@
       user1 = create(:user)
       user2 = create(:user)
       project = create(:empty_project)
-      issue = create(:issue, assignee: user1, project: project)
+      issue = create(:issue, assignees: [user1], project: project)
       project.add_developer(user1)
       project.add_developer(user2)
 
       expect(user1.assigned_open_issues_count).to eq(1)
       expect(user2.assigned_open_issues_count).to eq(0)
 
-      issue.assignee = user2
+      issue.assignees = [user2]
       issue.save
 
       expect(user1.assigned_open_issues_count).to eq(0)
@@ -676,6 +746,11 @@
       expect(attrs_hash).to include(:human_total_time_spent)
       expect(attrs_hash).to include('time_estimate')
     end
+
+    it 'includes assignee_ids and deprecated assignee_id' do
+      expect(attrs_hash).to include(:assignee_id)
+      expect(attrs_hash).to include(:assignee_ids)
+    end
   end
 
   describe '#check_for_spam' do
diff --git a/spec/models/merge_request_spec.rb b/spec/models/merge_request_spec.rb
index 8b72125dd5d1c..6cf3dd30ead92 100644
--- a/spec/models/merge_request_spec.rb
+++ b/spec/models/merge_request_spec.rb
@@ -9,6 +9,7 @@
     it { is_expected.to belong_to(:target_project).class_name('Project') }
     it { is_expected.to belong_to(:source_project).class_name('Project') }
     it { is_expected.to belong_to(:merge_user).class_name("User") }
+    it { is_expected.to belong_to(:assignee) }
     it { is_expected.to have_many(:merge_request_diffs).dependent(:destroy) }
   end
 
@@ -86,6 +87,86 @@
     end
   end
 
+  describe "before_save" do
+    describe "#update_cache_counts when a merge request is reassigned" do
+      let(:project) { create :project }
+      let(:merge_request) { create(:merge_request, source_project: project, target_project: project) }
+      let(:assignee) { create :user }
+
+      context "when previous assignee exists" do
+        before do
+          project.team << [assignee, :developer]
+          merge_request.update(assignee: assignee)
+        end
+
+        it "updates cache counts for new assignee" do
+          user = create(:user)
+
+          expect(user).to receive(:update_cache_counts)
+
+          merge_request.update(assignee: user)
+        end
+
+        it "updates cache counts for previous assignee" do
+          old_assignee = merge_request.assignee
+          allow(User).to receive(:find_by_id).with(old_assignee.id).and_return(old_assignee)
+
+          expect(old_assignee).to receive(:update_cache_counts)
+
+          merge_request.update(assignee: nil)
+        end
+      end
+
+      context "when previous assignee does not exist" do
+        it "updates cache count for the new assignee" do
+          merge_request.update(assignee: nil)
+
+          expect_any_instance_of(User).to receive(:update_cache_counts)
+
+          merge_request.update(assignee: assignee)
+        end
+      end
+    end
+  end
+
+  describe '#card_attributes' do
+    it 'includes the author name' do
+      allow(subject).to receive(:author).and_return(double(name: 'Robert'))
+      allow(subject).to receive(:assignee).and_return(nil)
+
+      expect(subject.card_attributes).
+        to eq({ 'Author' => 'Robert', 'Assignee' => nil })
+    end
+
+    it 'includes the assignee name' do
+      allow(subject).to receive(:author).and_return(double(name: 'Robert'))
+      allow(subject).to receive(:assignee).and_return(double(name: 'Douwe'))
+
+      expect(subject.card_attributes).
+        to eq({ 'Author' => 'Robert', 'Assignee' => 'Douwe' })
+    end
+  end
+
+  describe '#assignee_or_author?' do
+    let(:user) { create(:user) }
+
+    it 'returns true for a user that is assigned to a merge request' do
+      subject.assignee = user
+
+      expect(subject.assignee_or_author?(user)).to eq(true)
+    end
+
+    it 'returns true for a user that is the author of a merge request' do
+      subject.author = user
+
+      expect(subject.assignee_or_author?(user)).to eq(true)
+    end
+
+    it 'returns false for a user that is not the assignee or author' do
+      expect(subject.assignee_or_author?(user)).to eq(false)
+    end
+  end
+
   describe '#cache_merge_request_closes_issues!' do
     before do
       subject.project.team << [subject.author, :developer]
@@ -295,16 +376,6 @@ def set_compare(merge_request)
     end
   end
 
-  describe '#is_being_reassigned?' do
-    it 'returns true if the merge_request assignee has changed' do
-      subject.assignee = create(:user)
-      expect(subject.is_being_reassigned?).to be_truthy
-    end
-    it 'returns false if the merge request assignee has not changed' do
-      expect(subject.is_being_reassigned?).to be_falsey
-    end
-  end
-
   describe '#for_fork?' do
     it 'returns true if the merge request is for a fork' do
       subject.source_project = build_stubbed(:empty_project, namespace: create(:group))
diff --git a/spec/requests/api/issues_spec.rb b/spec/requests/api/issues_spec.rb
index 3ca13111acb0b..f7c317021dc76 100644
--- a/spec/requests/api/issues_spec.rb
+++ b/spec/requests/api/issues_spec.rb
@@ -19,7 +19,7 @@
   let!(:closed_issue) do
     create :closed_issue,
            author: user,
-           assignee: user,
+           assignees: [user],
            project: project,
            state: :closed,
            milestone: milestone,
@@ -31,14 +31,14 @@
            :confidential,
            project: project,
            author: author,
-           assignee: assignee,
+           assignees: [assignee],
            created_at: generate(:past_time),
            updated_at: 2.hours.ago
   end
   let!(:issue) do
     create :issue,
            author: user,
-           assignee: user,
+           assignees: [user],
            project: project,
            milestone: milestone,
            created_at: generate(:past_time),
@@ -265,7 +265,7 @@
     let!(:group_closed_issue) do
       create :closed_issue,
              author: user,
-             assignee: user,
+             assignees: [user],
              project: group_project,
              state: :closed,
              milestone: group_milestone,
@@ -276,13 +276,13 @@
              :confidential,
              project: group_project,
              author: author,
-             assignee: assignee,
+             assignees: [assignee],
              updated_at: 2.hours.ago
     end
     let!(:group_issue) do
       create :issue,
              author: user,
-             assignee: user,
+             assignees: [user],
              project: group_project,
              milestone: group_milestone,
              updated_at: 1.hour.ago,
@@ -687,6 +687,7 @@
       expect(json_response['updated_at']).to be_present
       expect(json_response['labels']).to eq(issue.label_names)
       expect(json_response['milestone']).to be_a Hash
+      expect(json_response['assignees']).to be_a Array
       expect(json_response['assignee']).to be_a Hash
       expect(json_response['author']).to be_a Hash
       expect(json_response['confidential']).to be_falsy
@@ -759,15 +760,38 @@
   end
 
   describe "POST /projects/:id/issues" do
+    context 'support for deprecated assignee_id' do
+      it 'creates a new project issue' do
+        post api("/projects/#{project.id}/issues", user),
+          title: 'new issue', assignee_id: user2.id
+
+        expect(response).to have_http_status(201)
+        expect(json_response['title']).to eq('new issue')
+        expect(json_response['assignee']['name']).to eq(user2.name)
+        expect(json_response['assignees'].first['name']).to eq(user2.name)
+      end
+    end
+
     it 'creates a new project issue' do
       post api("/projects/#{project.id}/issues", user),
+<<<<<<< HEAD
         title: 'new issue', labels: 'label, label2'
+=======
+        title: 'new issue', labels: 'label, label2', weight: 3,
+        assignee_ids: [user2.id]
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
 
       expect(response).to have_http_status(201)
       expect(json_response['title']).to eq('new issue')
       expect(json_response['description']).to be_nil
       expect(json_response['labels']).to eq(%w(label label2))
       expect(json_response['confidential']).to be_falsy
+<<<<<<< HEAD
+=======
+      expect(json_response['weight']).to eq(3)
+      expect(json_response['assignee']['name']).to eq(user2.name)
+      expect(json_response['assignees'].first['name']).to eq(user2.name)
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     end
 
     it 'creates a new confidential project issue' do
@@ -1057,6 +1081,46 @@
     end
   end
 
+  describe 'PUT /projects/:id/issues/:issue_iid to update assignee' do
+    context 'support for deprecated assignee_id' do
+      it 'removes assignee' do
+        put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+          assignee_id: 0
+
+        expect(response).to have_http_status(200)
+
+        expect(json_response['assignee']).to be_nil
+      end
+
+      it 'updates an issue with new assignee' do
+        put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+          assignee_id: user2.id
+
+        expect(response).to have_http_status(200)
+
+        expect(json_response['assignee']['name']).to eq(user2.name)
+      end
+    end
+
+    it 'removes assignee' do
+      put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+        assignee_ids: [0]
+
+      expect(response).to have_http_status(200)
+
+      expect(json_response['assignees']).to be_empty
+    end
+
+    it 'updates an issue with new assignee' do
+      put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+        assignee_ids: [user2.id]
+
+      expect(response).to have_http_status(200)
+
+      expect(json_response['assignees'].first['name']).to eq(user2.name)
+    end
+  end
+
   describe 'PUT /projects/:id/issues/:issue_iid to update labels' do
     let!(:label) { create(:label, title: 'dummy', project: project) }
     let!(:label_link) { create(:label_link, label: label, target: issue) }
diff --git a/spec/requests/api/v3/issues_spec.rb b/spec/requests/api/v3/issues_spec.rb
index ef5b10a161569..a1124aebcec9f 100644
--- a/spec/requests/api/v3/issues_spec.rb
+++ b/spec/requests/api/v3/issues_spec.rb
@@ -14,7 +14,7 @@
   let!(:closed_issue) do
     create :closed_issue,
            author: user,
-           assignee: user,
+           assignees: [user],
            project: project,
            state: :closed,
            milestone: milestone,
@@ -26,14 +26,14 @@
            :confidential,
            project: project,
            author: author,
-           assignee: assignee,
+           assignees: [assignee],
            created_at: generate(:past_time),
            updated_at: 2.hours.ago
   end
   let!(:issue) do
     create :issue,
            author: user,
-           assignee: user,
+           assignees: [user],
            project: project,
            milestone: milestone,
            created_at: generate(:past_time),
@@ -247,7 +247,7 @@
     let!(:group_closed_issue) do
       create :closed_issue,
              author: user,
-             assignee: user,
+             assignees: [user],
              project: group_project,
              state: :closed,
              milestone: group_milestone,
@@ -258,13 +258,13 @@
              :confidential,
              project: group_project,
              author: author,
-             assignee: assignee,
+             assignees: [assignee],
              updated_at: 2.hours.ago
     end
     let!(:group_issue) do
       create :issue,
              author: user,
-             assignee: user,
+             assignees: [user],
              project: group_project,
              milestone: group_milestone,
              updated_at: 1.hour.ago
@@ -737,13 +737,22 @@
   describe "POST /projects/:id/issues" do
     it 'creates a new project issue' do
       post v3_api("/projects/#{project.id}/issues", user),
+<<<<<<< HEAD
         title: 'new issue', labels: 'label, label2'
+=======
+        title: 'new issue', labels: 'label, label2', weight: 3, assignee_id: assignee.id
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
 
       expect(response).to have_http_status(201)
       expect(json_response['title']).to eq('new issue')
       expect(json_response['description']).to be_nil
       expect(json_response['labels']).to eq(%w(label label2))
       expect(json_response['confidential']).to be_falsy
+<<<<<<< HEAD
+=======
+      expect(json_response['weight']).to eq(3)
+      expect(json_response['assignee']['name']).to eq(assignee.name)
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     end
 
     it 'creates a new confidential project issue' do
@@ -1140,6 +1149,57 @@
     end
   end
 
+<<<<<<< HEAD
+=======
+  describe 'PUT /projects/:id/issues/:issue_id to update assignee' do
+    it 'updates an issue with no assignee' do
+      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), assignee_id: 0
+
+      expect(response).to have_http_status(200)
+      expect(json_response['assignee']).to eq(nil)
+    end
+
+    it 'updates an issue with assignee' do
+      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), assignee_id: user2.id
+
+      expect(response).to have_http_status(200)
+      expect(json_response['assignee']['name']).to eq(user2.name)
+    end
+  end
+
+  describe 'PUT /projects/:id/issues/:issue_id to update weight' do
+    it 'updates an issue with no weight' do
+      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), weight: 5
+
+      expect(response).to have_http_status(200)
+      expect(json_response['weight']).to eq(5)
+    end
+
+    it 'removes a weight from an issue' do
+      weighted_issue = create(:issue, project: project, weight: 2)
+
+      put v3_api("/projects/#{project.id}/issues/#{weighted_issue.id}", user), weight: nil
+
+      expect(response).to have_http_status(200)
+      expect(json_response['weight']).to be_nil
+    end
+
+    it 'returns 400 if weight is less than minimum weight' do
+      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), weight: -1
+
+      expect(response).to have_http_status(400)
+      expect(json_response['error']).to eq('weight does not have a valid value')
+    end
+
+    it 'returns 400 if weight is more than maximum weight' do
+      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), weight: 10
+
+      expect(response).to have_http_status(400)
+      expect(json_response['error']).to eq('weight does not have a valid value')
+    end
+  end
+
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
   describe "DELETE /projects/:id/issues/:issue_id" do
     it "rejects a non member from deleting an issue" do
       delete v3_api("/projects/#{project.id}/issues/#{issue.id}", non_member)
diff --git a/spec/services/issuable/bulk_update_service_spec.rb b/spec/services/issuable/bulk_update_service_spec.rb
index 7a1ac02731075..5b1639ca0d6e5 100644
--- a/spec/services/issuable/bulk_update_service_spec.rb
+++ b/spec/services/issuable/bulk_update_service_spec.rb
@@ -4,11 +4,12 @@
   let(:user)    { create(:user) }
   let(:project) { create(:empty_project, namespace: user.namespace) }
 
-  def bulk_update(issues, extra_params = {})
+  def bulk_update(issuables, extra_params = {})
     bulk_update_params = extra_params
-      .reverse_merge(issuable_ids: Array(issues).map(&:id).join(','))
+      .reverse_merge(issuable_ids: Array(issuables).map(&:id).join(','))
 
-    Issuable::BulkUpdateService.new(project, user, bulk_update_params).execute('issue')
+    type = Array(issuables).first.model_name.param_key
+    Issuable::BulkUpdateService.new(project, user, bulk_update_params).execute(type)
   end
 
   describe 'close issues' do
@@ -47,15 +48,15 @@ def bulk_update(issues, extra_params = {})
     end
   end
 
-  describe 'updating assignee' do
-    let(:issue) { create(:issue, project: project, assignee: user) }
+  describe 'updating merge request assignee' do
+    let(:merge_request) { create(:merge_request, target_project: project, source_project: project, assignee: user) }
 
     context 'when the new assignee ID is a valid user' do
       it 'succeeds' do
         new_assignee = create(:user)
         project.team << [new_assignee, :developer]
 
-        result = bulk_update(issue, assignee_id: new_assignee.id)
+        result = bulk_update(merge_request, assignee_id: new_assignee.id)
 
         expect(result[:success]).to be_truthy
         expect(result[:count]).to eq(1)
@@ -65,22 +66,59 @@ def bulk_update(issues, extra_params = {})
         assignee = create(:user)
         project.team << [assignee, :developer]
 
-        expect { bulk_update(issue, assignee_id: assignee.id) }
-          .to change { issue.reload.assignee }.from(user).to(assignee)
+        expect { bulk_update(merge_request, assignee_id: assignee.id) }
+          .to change { merge_request.reload.assignee }.from(user).to(assignee)
       end
     end
 
     context "when the new assignee ID is #{IssuableFinder::NONE}" do
       it "unassigns the issues" do
-        expect { bulk_update(issue, assignee_id: IssuableFinder::NONE) }
-          .to change { issue.reload.assignee }.to(nil)
+        expect { bulk_update(merge_request, assignee_id: IssuableFinder::NONE) }
+          .to change { merge_request.reload.assignee }.to(nil)
       end
     end
 
     context 'when the new assignee ID is not present' do
       it 'does not unassign' do
-        expect { bulk_update(issue, assignee_id: nil) }
-          .not_to change { issue.reload.assignee }
+        expect { bulk_update(merge_request, assignee_id: nil) }
+          .not_to change { merge_request.reload.assignee }
+      end
+    end
+  end
+
+  describe 'updating issue assignee' do
+    let(:issue) { create(:issue, project: project, assignees: [user]) }
+
+    context 'when the new assignee ID is a valid user' do
+      it 'succeeds' do
+        new_assignee = create(:user)
+        project.team << [new_assignee, :developer]
+
+        result = bulk_update(issue, assignee_ids: [new_assignee.id])
+
+        expect(result[:success]).to be_truthy
+        expect(result[:count]).to eq(1)
+      end
+
+      it 'updates the assignee to the use ID passed' do
+        assignee = create(:user)
+        project.team << [assignee, :developer]
+        expect { bulk_update(issue, assignee_ids: [assignee.id]) }
+          .to change { issue.reload.assignees.first }.from(user).to(assignee)
+      end
+    end
+
+    context "when the new assignee ID is #{IssuableFinder::NONE}" do
+      it "unassigns the issues" do
+        expect { bulk_update(issue, assignee_ids: [IssuableFinder::NONE.to_s]) }
+          .to change { issue.reload.assignees.count }.from(1).to(0)
+      end
+    end
+
+    context 'when the new assignee ID is not present' do
+      it 'does not unassign' do
+        expect { bulk_update(issue, assignee_ids: []) }
+          .not_to change{ issue.reload.assignees }
       end
     end
   end
diff --git a/spec/services/issues/close_service_spec.rb b/spec/services/issues/close_service_spec.rb
index 7a54373963e54..5184053171186 100644
--- a/spec/services/issues/close_service_spec.rb
+++ b/spec/services/issues/close_service_spec.rb
@@ -4,7 +4,7 @@
   let(:user) { create(:user) }
   let(:user2) { create(:user) }
   let(:guest) { create(:user) }
-  let(:issue) { create(:issue, assignee: user2) }
+  let(:issue) { create(:issue, assignees: [user2]) }
   let(:project) { issue.project }
   let!(:todo) { create(:todo, :assigned, user: user, project: project, target: issue, author: user2) }
 
diff --git a/spec/services/issues/create_service_spec.rb b/spec/services/issues/create_service_spec.rb
index 80bfb7315505a..01edc46496d97 100644
--- a/spec/services/issues/create_service_spec.rb
+++ b/spec/services/issues/create_service_spec.rb
@@ -6,10 +6,10 @@
 
   describe '#execute' do
     let(:issue) { described_class.new(project, user, opts).execute }
+    let(:assignee) { create(:user) }
+    let(:milestone) { create(:milestone, project: project) }
 
     context 'when params are valid' do
-      let(:assignee) { create(:user) }
-      let(:milestone) { create(:milestone, project: project) }
       let(:labels) { create_pair(:label, project: project) }
 
       before do
@@ -20,7 +20,7 @@
       let(:opts) do
         { title: 'Awesome issue',
           description: 'please fix',
-          assignee_id: assignee.id,
+          assignee_ids: [assignee.id],
           label_ids: labels.map(&:id),
           milestone_id: milestone.id,
           due_date: Date.tomorrow }
@@ -29,7 +29,7 @@
       it 'creates the issue with the given params' do
         expect(issue).to be_persisted
         expect(issue.title).to eq('Awesome issue')
-        expect(issue.assignee).to eq assignee
+        expect(issue.assignees).to eq [assignee]
         expect(issue.labels).to match_array labels
         expect(issue.milestone).to eq milestone
         expect(issue.due_date).to eq Date.tomorrow
@@ -37,6 +37,7 @@
 
       context 'when current user cannot admin issues in the project' do
         let(:guest) { create(:user) }
+
         before do
           project.team << [guest, :guest]
         end
@@ -47,7 +48,7 @@
           expect(issue).to be_persisted
           expect(issue.title).to eq('Awesome issue')
           expect(issue.description).to eq('please fix')
-          expect(issue.assignee).to be_nil
+          expect(issue.assignees).to be_empty
           expect(issue.labels).to be_empty
           expect(issue.milestone).to be_nil
           expect(issue.due_date).to be_nil
@@ -136,10 +137,83 @@
       end
     end
 
-    it_behaves_like 'issuable create service'
+    context 'issue create service' do
+      context 'assignees' do
+        before { project.team << [user, :master] }
+
+        it 'removes assignee when user id is invalid' do
+          opts = { title: 'Title', description: 'Description', assignee_ids: [-1] }
+
+          issue = described_class.new(project, user, opts).execute
+
+          expect(issue.assignees).to be_empty
+        end
+
+        it 'removes assignee when user id is 0' do
+          opts = { title: 'Title', description: 'Description',  assignee_ids: [0] }
+
+          issue = described_class.new(project, user, opts).execute
+
+          expect(issue.assignees).to be_empty
+        end
+
+        it 'saves assignee when user id is valid' do
+          project.team << [assignee, :master]
+          opts = { title: 'Title', description: 'Description', assignee_ids: [assignee.id] }
+
+          issue = described_class.new(project, user, opts).execute
+
+          expect(issue.assignees).to eq([assignee])
+        end
+
+        context "when issuable feature is private" do
+          before do
+            project.project_feature.update(issues_access_level: ProjectFeature::PRIVATE,
+                                           merge_requests_access_level: ProjectFeature::PRIVATE)
+          end
+
+          levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
+
+          levels.each do |level|
+            it "removes not authorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
+              project.update(visibility_level: level)
+              opts = { title: 'Title', description: 'Description', assignee_ids: [assignee.id] }
+
+              issue = described_class.new(project, user, opts).execute
+
+              expect(issue.assignees).to be_empty
+            end
+          end
+        end
+      end
+    end
 
     it_behaves_like 'new issuable record that supports slash commands'
 
+    context 'Slash commands' do
+      context 'with assignee and milestone in params and command' do
+        let(:opts) do
+          {
+            assignee_ids: [create(:user).id],
+            milestone_id: 1,
+            title: 'Title',
+            description: %(/assign @#{assignee.username}\n/milestone %"#{milestone.name}")
+          }
+        end
+
+        before do
+          project.team << [user, :master]
+          project.team << [assignee, :master]
+        end
+
+        it 'assigns and sets milestone to issuable from command' do
+          expect(issue).to be_persisted
+          expect(issue.assignees).to eq([assignee])
+          expect(issue.milestone).to eq(milestone)
+        end
+      end
+    end
+
     context 'resolving discussions' do
       let(:discussion) { create(:diff_note_on_merge_request).to_discussion }
       let(:merge_request) { discussion.noteable }
diff --git a/spec/services/issues/update_service_spec.rb b/spec/services/issues/update_service_spec.rb
index 5b324f3c706d6..1797a23ee8afd 100644
--- a/spec/services/issues/update_service_spec.rb
+++ b/spec/services/issues/update_service_spec.rb
@@ -14,7 +14,7 @@
   let(:issue) do
     create(:issue, title: 'Old title',
                    description: "for #{user2.to_reference}",
-                   assignee_id: user3.id,
+                   assignee_ids: [user3.id],
                    project: project)
   end
 
@@ -40,7 +40,7 @@ def update_issue(opts)
         {
           title: 'New title',
           description: 'Also please fix',
-          assignee_id: user2.id,
+          assignee_ids: [user2.id, user3.id],
           state_event: 'close',
           label_ids: [label.id],
           due_date: Date.tomorrow
@@ -53,15 +53,15 @@ def update_issue(opts)
         expect(issue).to be_valid
         expect(issue.title).to eq 'New title'
         expect(issue.description).to eq 'Also please fix'
-        expect(issue.assignee).to eq user2
+        expect(issue.assignees).to match_array([user2, user3])
         expect(issue).to be_closed
         expect(issue.labels).to match_array [label]
         expect(issue.due_date).to eq Date.tomorrow
       end
 
       it 'sorts issues as specified by parameters' do
-        issue1 = create(:issue, project: project, assignee_id: user3.id)
-        issue2 = create(:issue, project: project, assignee_id: user3.id)
+        issue1 = create(:issue, project: project, assignees: [user3])
+        issue2 = create(:issue, project: project, assignees: [user3])
 
         [issue, issue1, issue2].each do |issue|
           issue.move_to_end
@@ -87,7 +87,7 @@ def update_issue(opts)
           expect(issue).to be_valid
           expect(issue.title).to eq 'New title'
           expect(issue.description).to eq 'Also please fix'
-          expect(issue.assignee).to eq user3
+          expect(issue.assignees).to match_array [user3]
           expect(issue.labels).to be_empty
           expect(issue.milestone).to be_nil
           expect(issue.due_date).to be_nil
@@ -137,7 +137,7 @@ def update_issue(opts)
         {
           title: 'New title',
           description: 'Also please fix',
-          assignee_id: user2.id,
+          assignee_ids: [user2],
           state_event: 'close',
           label_ids: [label.id],
           confidential: true
@@ -163,12 +163,12 @@ def update_issue(opts)
       it 'does not update assignee_id with unauthorized users' do
         project.update(visibility_level: Gitlab::VisibilityLevel::PUBLIC)
         update_issue(confidential: true)
-        non_member        = create(:user)
-        original_assignee = issue.assignee
+        non_member = create(:user)
+        original_assignees = issue.assignees
 
-        update_issue(assignee_id: non_member.id)
+        update_issue(assignee_ids: [non_member.id])
 
-        expect(issue.reload.assignee_id).to eq(original_assignee.id)
+        expect(issue.reload.assignees).to eq(original_assignees)
       end
     end
 
@@ -205,7 +205,7 @@ def update_issue(opts)
 
       context 'when is reassigned' do
         before do
-          update_issue(assignee: user2)
+          update_issue(assignees: [user2])
         end
 
         it 'marks previous assignee todos as done' do
@@ -408,6 +408,41 @@ def update_issue(opts)
       end
     end
 
+    context 'updating asssignee_id' do
+      it 'does not update assignee when assignee_id is invalid' do
+        update_issue(assignee_ids: [-1])
+
+        expect(issue.reload.assignees).to eq([user3])
+      end
+
+      it 'unassigns assignee when user id is 0' do
+        update_issue(assignee_ids: [0])
+
+        expect(issue.reload.assignees).to be_empty
+      end
+
+      it 'does not update assignee_id when user cannot read issue' do
+        update_issue(assignee_ids: [create(:user).id])
+
+        expect(issue.reload.assignees).to eq([user3])
+      end
+
+      context "when issuable feature is private" do
+        levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
+
+        levels.each do |level|
+          it "does not update with unauthorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
+            assignee = create(:user)
+            project.update(visibility_level: level)
+            feature_visibility_attr = :"#{issue.model_name.plural}_access_level"
+            project.project_feature.update_attribute(feature_visibility_attr, ProjectFeature::PRIVATE)
+
+            expect{ update_issue(assignee_ids: [assignee.id]) }.not_to change{ issue.assignees }
+          end
+        end
+      end
+    end
+
     context 'updating mentions' do
       let(:mentionable) { issue }
       include_examples 'updating mentions', Issues::UpdateService
diff --git a/spec/services/merge_requests/assign_issues_service_spec.rb b/spec/services/merge_requests/assign_issues_service_spec.rb
index fe75757dd29c4..d3556020d4d06 100644
--- a/spec/services/merge_requests/assign_issues_service_spec.rb
+++ b/spec/services/merge_requests/assign_issues_service_spec.rb
@@ -15,14 +15,14 @@
     expect(service.assignable_issues.map(&:id)).to include(issue.id)
   end
 
-  it 'ignores issues already assigned to any user' do
-    issue.update!(assignee: create(:user))
+  it 'ignores issues the user cannot update assignee on' do
+    project.team.truncate
 
     expect(service.assignable_issues).to be_empty
   end
 
-  it 'ignores issues the user cannot update assignee on' do
-    project.team.truncate
+  it 'ignores issues already assigned to any user' do
+    issue.assignees = [create(:user)]
 
     expect(service.assignable_issues).to be_empty
   end
@@ -44,7 +44,7 @@
   end
 
   it 'assigns these to the merge request owner' do
-    expect { service.execute }.to change { issue.reload.assignee }.to(user)
+    expect { service.execute }.to change { issue.assignees.first }.to(user)
   end
 
   it 'ignores external issues' do
diff --git a/spec/services/merge_requests/create_service_spec.rb b/spec/services/merge_requests/create_service_spec.rb
index 0e16c7cc94bbd..ace82380cc999 100644
--- a/spec/services/merge_requests/create_service_spec.rb
+++ b/spec/services/merge_requests/create_service_spec.rb
@@ -84,7 +84,87 @@
       end
     end
 
-    it_behaves_like 'issuable create service'
+    context 'Slash commands' do
+      context 'with assignee and milestone in params and command' do
+        let(:merge_request) { described_class.new(project, user, opts).execute }
+        let(:milestone) { create(:milestone, project: project) }
+
+        let(:opts) do
+          {
+            assignee_id: create(:user).id,
+            milestone_id: 1,
+            title: 'Title',
+            description: %(/assign @#{assignee.username}\n/milestone %"#{milestone.name}"),
+            source_branch: 'feature',
+            target_branch: 'master'
+          }
+        end
+
+        before do
+          project.team << [user, :master]
+          project.team << [assignee, :master]
+        end
+
+        it 'assigns and sets milestone to issuable from command' do
+          expect(merge_request).to be_persisted
+          expect(merge_request.assignee).to eq(assignee)
+          expect(merge_request.milestone).to eq(milestone)
+        end
+      end
+    end
+
+    context 'merge request create service' do
+      context 'asssignee_id' do
+        let(:assignee) { create(:user) }
+
+        before { project.team << [user, :master] }
+
+        it 'removes assignee_id when user id is invalid' do
+          opts = { title: 'Title', description: 'Description', assignee_id: -1 }
+
+          merge_request = described_class.new(project, user, opts).execute
+
+          expect(merge_request.assignee_id).to be_nil
+        end
+
+        it 'removes assignee_id when user id is 0' do
+          opts = { title: 'Title', description: 'Description',  assignee_id: 0 }
+
+          merge_request = described_class.new(project, user, opts).execute
+
+          expect(merge_request.assignee_id).to be_nil
+        end
+
+        it 'saves assignee when user id is valid' do
+          project.team << [assignee, :master]
+          opts = { title: 'Title', description: 'Description', assignee_id: assignee.id }
+
+          merge_request = described_class.new(project, user, opts).execute
+
+          expect(merge_request.assignee).to eq(assignee)
+        end
+
+        context "when issuable feature is private" do
+          before do
+            project.project_feature.update(issues_access_level: ProjectFeature::PRIVATE,
+                                           merge_requests_access_level: ProjectFeature::PRIVATE)
+          end
+
+          levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
+
+          levels.each do |level|
+            it "removes not authorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
+              project.update(visibility_level: level)
+              opts = { title: 'Title', description: 'Description', assignee_id: assignee.id }
+
+              merge_request = described_class.new(project, user, opts).execute
+
+              expect(merge_request.assignee_id).to be_nil
+            end
+          end
+        end
+      end
+    end
 
     context 'while saving references to issues that the created merge request closes' do
       let(:first_issue) { create(:issue, project: project) }
diff --git a/spec/services/merge_requests/update_service_spec.rb b/spec/services/merge_requests/update_service_spec.rb
index f2ca1e6fcbd80..694ec3a579f85 100644
--- a/spec/services/merge_requests/update_service_spec.rb
+++ b/spec/services/merge_requests/update_service_spec.rb
@@ -423,6 +423,54 @@ def update_merge_request(opts)
       end
     end
 
+    context 'updating asssignee_id' do
+      it 'does not update assignee when assignee_id is invalid' do
+        merge_request.update(assignee_id: user.id)
+
+        update_merge_request(assignee_id: -1)
+
+        expect(merge_request.reload.assignee).to eq(user)
+      end
+
+      it 'unassigns assignee when user id is 0' do
+        merge_request.update(assignee_id: user.id)
+
+        update_merge_request(assignee_id: 0)
+
+        expect(merge_request.assignee_id).to be_nil
+      end
+
+      it 'saves assignee when user id is valid' do
+        update_merge_request(assignee_id: user.id)
+
+        expect(merge_request.assignee_id).to eq(user.id)
+      end
+
+      it 'does not update assignee_id when user cannot read issue' do
+        non_member        = create(:user)
+        original_assignee = merge_request.assignee
+
+        update_merge_request(assignee_id: non_member.id)
+
+        expect(merge_request.assignee_id).to eq(original_assignee.id)
+      end
+
+      context "when issuable feature is private" do
+        levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
+
+        levels.each do |level|
+          it "does not update with unauthorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
+            assignee = create(:user)
+            project.update(visibility_level: level)
+            feature_visibility_attr = :"#{merge_request.model_name.plural}_access_level"
+            project.project_feature.update_attribute(feature_visibility_attr, ProjectFeature::PRIVATE)
+
+            expect{ update_merge_request(assignee_id: assignee) }.not_to change{ merge_request.assignee }
+          end
+        end
+      end
+    end
+
     include_examples 'issuable update service' do
       let(:open_issuable) { merge_request }
       let(:closed_issuable) { create(:closed_merge_request, source_project: project) }
diff --git a/spec/services/notes/slash_commands_service_spec.rb b/spec/services/notes/slash_commands_service_spec.rb
index 1a64c8bbf00df..0edcc50ed7b22 100644
--- a/spec/services/notes/slash_commands_service_spec.rb
+++ b/spec/services/notes/slash_commands_service_spec.rb
@@ -66,7 +66,7 @@
           expect(content).to eq ''
           expect(note.noteable).to be_closed
           expect(note.noteable.labels).to match_array(labels)
-          expect(note.noteable.assignee).to eq(assignee)
+          expect(note.noteable.assignees).to eq([assignee])
           expect(note.noteable.milestone).to eq(milestone)
         end
       end
@@ -113,7 +113,7 @@
           expect(content).to eq "HELLO\nWORLD"
           expect(note.noteable).to be_closed
           expect(note.noteable.labels).to match_array(labels)
-          expect(note.noteable.assignee).to eq(assignee)
+          expect(note.noteable.assignees).to eq([assignee])
           expect(note.noteable.milestone).to eq(milestone)
         end
       end
diff --git a/spec/services/notification_service_spec.rb b/spec/services/notification_service_spec.rb
index 989fd90cda9d9..74f96b9790988 100644
--- a/spec/services/notification_service_spec.rb
+++ b/spec/services/notification_service_spec.rb
@@ -4,6 +4,7 @@
   include EmailHelpers
 
   let(:notification) { NotificationService.new }
+  let(:assignee) { create(:user) }
 
   around(:each) do |example|
     perform_enqueued_jobs do
@@ -52,7 +53,11 @@ def send_notifications(*new_mentions)
 
   shared_examples 'participating by assignee notification' do
     it 'emails the participant' do
-      issuable.update_attribute(:assignee, participant)
+      if issuable.is_a?(Issue)
+        issuable.assignees << participant
+      else
+        issuable.update_attribute(:assignee, participant)
+      end
 
       notification_trigger
 
@@ -103,14 +108,14 @@ def send_notifications(*new_mentions)
   describe 'Notes' do
     context 'issue note' do
       let(:project) { create(:empty_project, :private) }
-      let(:issue) { create(:issue, project: project, assignee: create(:user)) }
-      let(:mentioned_issue) { create(:issue, assignee: issue.assignee) }
+      let(:issue) { create(:issue, project: project, assignees: [assignee]) }
+      let(:mentioned_issue) { create(:issue, assignees: issue.assignees) }
       let(:note) { create(:note_on_issue, noteable: issue, project_id: issue.project_id, note: '@mention referenced, @outsider also') }
 
       before do
         build_team(note.project)
         project.add_master(issue.author)
-        project.add_master(issue.assignee)
+        project.add_master(assignee)
         project.add_master(note.author)
         create(:note_on_issue, noteable: issue, project_id: issue.project_id, note: '@subscribed_participant cc this guy')
         update_custom_notification(:new_note, @u_guest_custom, resource: project)
@@ -130,7 +135,7 @@ def send_notifications(*new_mentions)
 
           should_email(@u_watcher)
           should_email(note.noteable.author)
-          should_email(note.noteable.assignee)
+          should_email(note.noteable.assignees.first)
           should_email(@u_custom_global)
           should_email(@u_mentioned)
           should_email(@subscriber)
@@ -196,7 +201,7 @@ def send_notifications(*new_mentions)
           notification.new_note(note)
 
           should_email(note.noteable.author)
-          should_email(note.noteable.assignee)
+          should_email(note.noteable.assignees.first)
           should_email(@u_mentioned)
           should_email(@u_custom_global)
           should_not_email(@u_guest_custom)
@@ -218,7 +223,7 @@ def send_notifications(*new_mentions)
       let(:member) { create(:user) }
       let(:guest) { create(:user) }
       let(:admin) { create(:admin) }
-      let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee) }
+      let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee]) }
       let(:note) { create(:note_on_issue, noteable: confidential_issue, project: project, note: "#{author.to_reference} #{assignee.to_reference} #{non_member.to_reference} #{member.to_reference} #{admin.to_reference}") }
       let(:guest_watcher) { create_user_with_notification(:watch, "guest-watcher-confidential") }
 
@@ -244,8 +249,8 @@ def send_notifications(*new_mentions)
 
     context 'issue note mention' do
       let(:project) { create(:empty_project, :public) }
-      let(:issue) { create(:issue, project: project, assignee: create(:user)) }
-      let(:mentioned_issue) { create(:issue, assignee: issue.assignee) }
+      let(:issue) { create(:issue, project: project, assignees: [assignee]) }
+      let(:mentioned_issue) { create(:issue, assignees: issue.assignees) }
       let(:note) { create(:note_on_issue, noteable: issue, project_id: issue.project_id, note: '@all mentioned') }
 
       before do
@@ -269,7 +274,7 @@ def send_notifications(*new_mentions)
 
           should_email(@u_guest_watcher)
           should_email(note.noteable.author)
-          should_email(note.noteable.assignee)
+          should_email(note.noteable.assignees.first)
           should_not_email(note.author)
           should_email(@u_mentioned)
           should_not_email(@u_disabled)
@@ -449,7 +454,7 @@ def send_notifications(*new_mentions)
     let(:group) { create(:group) }
     let(:project) { create(:empty_project, :public, namespace: group) }
     let(:another_project) { create(:empty_project, :public, namespace: group) }
-    let(:issue) { create :issue, project: project, assignee: create(:user), description: 'cc @participant' }
+    let(:issue) { create :issue, project: project, assignees: [assignee], description: 'cc @participant' }
 
     before do
       build_team(issue.project)
@@ -465,7 +470,7 @@ def send_notifications(*new_mentions)
       it do
         notification.new_issue(issue, @u_disabled)
 
-        should_email(issue.assignee)
+        should_email(assignee)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
@@ -480,10 +485,10 @@ def send_notifications(*new_mentions)
       end
 
       it do
-        create_global_setting_for(issue.assignee, :mention)
+        create_global_setting_for(issue.assignees.first, :mention)
         notification.new_issue(issue, @u_disabled)
 
-        should_not_email(issue.assignee)
+        should_not_email(issue.assignees.first)
       end
 
       it "emails the author if they've opted into notifications about their activity" do
@@ -528,7 +533,7 @@ def send_notifications(*new_mentions)
         let(:member) { create(:user) }
         let(:guest) { create(:user) }
         let(:admin) { create(:admin) }
-        let(:confidential_issue) { create(:issue, :confidential, project: project, title: 'Confidential issue', author: author, assignee: assignee) }
+        let(:confidential_issue) { create(:issue, :confidential, project: project, title: 'Confidential issue', author: author, assignees: [assignee]) }
 
         it "emails subscribers of the issue's labels that can read the issue" do
           project.add_developer(member)
@@ -572,9 +577,9 @@ def send_notifications(*new_mentions)
       end
 
       it 'emails new assignee' do
-        notification.reassigned_issue(issue, @u_disabled)
+        notification.reassigned_issue(issue, @u_disabled, [assignee])
 
-        should_email(issue.assignee)
+        should_email(issue.assignees.first)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
@@ -588,9 +593,8 @@ def send_notifications(*new_mentions)
       end
 
       it 'emails previous assignee even if he has the "on mention" notif level' do
-        issue.update_attribute(:assignee, @u_mentioned)
-        issue.update_attributes(assignee: @u_watcher)
-        notification.reassigned_issue(issue, @u_disabled)
+        issue.assignees = [@u_mentioned]
+        notification.reassigned_issue(issue, @u_disabled, [@u_watcher])
 
         should_email(@u_mentioned)
         should_email(@u_watcher)
@@ -606,11 +610,11 @@ def send_notifications(*new_mentions)
       end
 
       it 'emails new assignee even if he has the "on mention" notif level' do
-        issue.update_attributes(assignee: @u_mentioned)
-        notification.reassigned_issue(issue, @u_disabled)
+        issue.assignees = [@u_mentioned]
+        notification.reassigned_issue(issue, @u_disabled, [@u_mentioned])
 
-        expect(issue.assignee).to be @u_mentioned
-        should_email(issue.assignee)
+        expect(issue.assignees.first).to be @u_mentioned
+        should_email(issue.assignees.first)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
@@ -624,11 +628,11 @@ def send_notifications(*new_mentions)
       end
 
       it 'emails new assignee' do
-        issue.update_attribute(:assignee, @u_mentioned)
-        notification.reassigned_issue(issue, @u_disabled)
+        issue.assignees = [@u_mentioned]
+        notification.reassigned_issue(issue, @u_disabled, [@u_mentioned])
 
-        expect(issue.assignee).to be @u_mentioned
-        should_email(issue.assignee)
+        expect(issue.assignees.first).to be @u_mentioned
+        should_email(issue.assignees.first)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
@@ -642,17 +646,17 @@ def send_notifications(*new_mentions)
       end
 
       it 'does not email new assignee if they are the current user' do
-        issue.update_attribute(:assignee, @u_mentioned)
-        notification.reassigned_issue(issue, @u_mentioned)
+        issue.assignees = [@u_mentioned]
+        notification.reassigned_issue(issue, @u_mentioned, [@u_mentioned])
 
-        expect(issue.assignee).to be @u_mentioned
+        expect(issue.assignees.first).to be @u_mentioned
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
         should_email(@u_participant_mentioned)
         should_email(@subscriber)
         should_email(@u_custom_global)
-        should_not_email(issue.assignee)
+        should_not_email(issue.assignees.first)
         should_not_email(@unsubscriber)
         should_not_email(@u_participating)
         should_not_email(@u_disabled)
@@ -662,7 +666,7 @@ def send_notifications(*new_mentions)
       it_behaves_like 'participating notifications' do
         let(:participant) { create(:user, username: 'user-participant') }
         let(:issuable) { issue }
-        let(:notification_trigger) { notification.reassigned_issue(issue, @u_disabled) }
+        let(:notification_trigger) { notification.reassigned_issue(issue, @u_disabled, [assignee]) }
       end
     end
 
@@ -705,7 +709,7 @@ def send_notifications(*new_mentions)
       it "doesn't send email to anyone but subscribers of the given labels" do
         notification.relabeled_issue(issue, [group_label_2, label_2], @u_disabled)
 
-        should_not_email(issue.assignee)
+        should_not_email(issue.assignees.first)
         should_not_email(issue.author)
         should_not_email(@u_watcher)
         should_not_email(@u_guest_watcher)
@@ -729,7 +733,7 @@ def send_notifications(*new_mentions)
         let(:member) { create(:user) }
         let(:guest) { create(:user) }
         let(:admin) { create(:admin) }
-        let(:confidential_issue) { create(:issue, :confidential, project: project, title: 'Confidential issue', author: author, assignee: assignee) }
+        let(:confidential_issue) { create(:issue, :confidential, project: project, title: 'Confidential issue', author: author, assignees: [assignee]) }
         let!(:label_1) { create(:label, project: project, issues: [confidential_issue]) }
         let!(:label_2) { create(:label, project: project) }
 
@@ -767,7 +771,7 @@ def send_notifications(*new_mentions)
       it 'sends email to issue assignee and issue author' do
         notification.close_issue(issue, @u_disabled)
 
-        should_email(issue.assignee)
+        should_email(issue.assignees.first)
         should_email(issue.author)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
@@ -798,7 +802,7 @@ def send_notifications(*new_mentions)
       it 'sends email to issue notification recipients' do
         notification.reopen_issue(issue, @u_disabled)
 
-        should_email(issue.assignee)
+        should_email(issue.assignees.first)
         should_email(issue.author)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
@@ -826,7 +830,7 @@ def send_notifications(*new_mentions)
       it 'sends email to issue notification recipients' do
         notification.issue_moved(issue, new_issue, @u_disabled)
 
-        should_email(issue.assignee)
+        should_email(issue.assignees.first)
         should_email(issue.author)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
diff --git a/spec/services/projects/autocomplete_service_spec.rb b/spec/services/projects/autocomplete_service_spec.rb
index 7916c2d957cc0..c198c3eedfca2 100644
--- a/spec/services/projects/autocomplete_service_spec.rb
+++ b/spec/services/projects/autocomplete_service_spec.rb
@@ -11,7 +11,7 @@
       let(:project) { create(:empty_project, :public) }
       let!(:issue) { create(:issue, project: project, title: 'Issue 1') }
       let!(:security_issue_1) { create(:issue, :confidential, project: project, title: 'Security issue 1', author: author) }
-      let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignee: assignee) }
+      let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignees: [assignee]) }
 
       it 'does not list project confidential issues for guests' do
         autocomplete = described_class.new(project, nil)
diff --git a/spec/services/slash_commands/interpret_service_spec.rb b/spec/services/slash_commands/interpret_service_spec.rb
index 29e65fe7ce6da..865a7c698759e 100644
--- a/spec/services/slash_commands/interpret_service_spec.rb
+++ b/spec/services/slash_commands/interpret_service_spec.rb
@@ -3,6 +3,7 @@
 describe SlashCommands::InterpretService, services: true do
   let(:project) { create(:empty_project, :public) }
   let(:developer) { create(:user) }
+  let(:developer2) { create(:user) }
   let(:issue) { create(:issue, project: project) }
   let(:milestone) { create(:milestone, project: project, title: '9.10') }
   let(:inprogress) { create(:label, project: project, title: 'In Progress') }
@@ -42,6 +43,7 @@
       end
     end
 
+<<<<<<< HEAD
     shared_examples 'assign command' do
       it 'fetches assignee and populates assignee_id if content contains /assign' do
         _, updates = service.execute(content, issuable)
@@ -59,6 +61,8 @@
       end
     end
 
+=======
+>>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     shared_examples 'milestone command' do
       it 'fetches milestone and populates milestone_id if content contains /milestone' do
         milestone # populate the milestone
@@ -371,14 +375,46 @@
       let(:issuable) { issue }
     end
 
-    it_behaves_like 'assign command' do
+    context 'assign command' do
       let(:content) { "/assign @#{developer.username}" }
-      let(:issuable) { issue }
+
+      context 'Issue' do
+        it 'fetches assignee and populates assignee_id if content contains /assign' do
+          _, updates = service.execute(content, issue)
+
+          expect(updates).to eq(assignee_ids: [developer.id])
+        end
+      end
+
+      context 'Merge Request' do
+        it 'fetches assignee and populates assignee_id if content contains /assign' do
+          _, updates = service.execute(content, merge_request)
+
+          expect(updates).to eq(assignee_id: developer.id)
+        end
+      end
     end
 
-    it_behaves_like 'assign command' do
-      let(:content) { "/assign @#{developer.username}" }
-      let(:issuable) { merge_request }
+    context 'assign command with multiple assignees' do
+      let(:content) { "/assign @#{developer.username} @#{developer2.username}" }
+
+      before{ project.team << [developer2, :developer] }
+
+      context 'Issue' do
+        it 'fetches assignee and populates assignee_id if content contains /assign' do
+          _, updates = service.execute(content, issue)
+
+          expect(updates[:assignee_ids]).to match_array([developer.id, developer2.id])
+        end
+      end
+
+      context 'Merge Request' do
+        it 'fetches assignee and populates assignee_id if content contains /assign' do
+          _, updates = service.execute(content, merge_request)
+
+          expect(updates).to eq(assignee_id: developer.id)
+        end
+      end
     end
 
     it_behaves_like 'empty command' do
@@ -391,14 +427,26 @@
       let(:issuable) { issue }
     end
 
-    it_behaves_like 'unassign command' do
+    context 'unassign command' do
       let(:content) { '/unassign' }
-      let(:issuable) { issue }
-    end
 
-    it_behaves_like 'unassign command' do
-      let(:content) { '/unassign' }
-      let(:issuable) { merge_request }
+      context 'Issue' do
+        it 'populates assignee_ids: [] if content contains /unassign' do
+          issue.update(assignee_ids: [developer.id])
+          _, updates = service.execute(content, issue)
+
+          expect(updates).to eq(assignee_ids: [])
+        end
+      end
+
+      context 'Merge Request' do
+        it 'populates assignee_id: nil if content contains /unassign' do
+          merge_request.update(assignee_id: developer.id)
+          _, updates = service.execute(content, merge_request)
+
+          expect(updates).to eq(assignee_id: nil)
+        end
+      end
     end
 
     it_behaves_like 'milestone command' do
diff --git a/spec/services/system_note_service_spec.rb b/spec/services/system_note_service_spec.rb
index 75d7caf2508cd..5f1b82e835568 100644
--- a/spec/services/system_note_service_spec.rb
+++ b/spec/services/system_note_service_spec.rb
@@ -6,6 +6,7 @@
   let(:project)  { create(:empty_project) }
   let(:author)   { create(:user) }
   let(:noteable) { create(:issue, project: project) }
+  let(:issue)    { noteable }
 
   shared_examples_for 'a system note' do
     let(:expected_noteable) { noteable }
@@ -155,6 +156,50 @@
     end
   end
 
+  describe '.change_issue_assignees' do
+    subject { described_class.change_issue_assignees(noteable, project, author, [assignee]) }
+
+    let(:assignee) { create(:user) }
+    let(:assignee1) { create(:user) }
+    let(:assignee2) { create(:user) }
+    let(:assignee3) { create(:user) }
+
+    it_behaves_like 'a system note'
+
+    def build_note(old_assignees, new_assignees)
+      issue.assignees = new_assignees
+      described_class.change_issue_assignees(issue, project, author, old_assignees).note
+    end
+
+    it 'builds a correct phrase when an assignee is added to a non-assigned issue' do
+      expect(build_note([], [assignee1])).to eq "assigned to @#{assignee1.username}"
+    end
+
+    it 'builds a correct phrase when assignee removed' do
+      expect(build_note([assignee1], [])).to eq 'removed all assignees'
+    end
+
+    it 'builds a correct phrase when assignees changed' do
+      expect(build_note([assignee1], [assignee2])).to eq \
+        "assigned to @#{assignee2.username} and unassigned @#{assignee1.username}"
+    end
+
+    it 'builds a correct phrase when three assignees removed and one added' do
+      expect(build_note([assignee, assignee1, assignee2], [assignee3])).to eq \
+        "assigned to @#{assignee3.username} and unassigned @#{assignee.username}, @#{assignee1.username}, and @#{assignee2.username}"
+    end
+
+    it 'builds a correct phrase when one assignee changed from a set' do
+      expect(build_note([assignee, assignee1], [assignee, assignee2])).to eq \
+        "assigned to @#{assignee2.username} and unassigned @#{assignee1.username}"
+    end
+
+    it 'builds a correct phrase when one assignee removed from a set' do
+      expect(build_note([assignee, assignee1, assignee2], [assignee, assignee1])).to eq \
+        "unassigned @#{assignee2.username}"
+    end
+  end
+
   describe '.change_label' do
     subject { described_class.change_label(noteable, project, author, added, removed) }
 
diff --git a/spec/services/todo_service_spec.rb b/spec/services/todo_service_spec.rb
index 89b3b6aad103c..175a42a32d9b7 100644
--- a/spec/services/todo_service_spec.rb
+++ b/spec/services/todo_service_spec.rb
@@ -25,11 +25,11 @@
   end
 
   describe 'Issues' do
-    let(:issue) { create(:issue, project: project, assignee: john_doe, author: author, description: "- [ ] Task 1\n- [ ] Task 2 #{mentions}") }
-    let(:addressed_issue) { create(:issue, project: project, assignee: john_doe, author: author, description: "#{directly_addressed}\n- [ ] Task 1\n- [ ] Task 2") }
-    let(:unassigned_issue) { create(:issue, project: project, assignee: nil) }
-    let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee, description: mentions) }
-    let(:addressed_confident_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee, description: directly_addressed) }
+    let(:issue) { create(:issue, project: project, assignees: [john_doe], author: author, description: "- [ ] Task 1\n- [ ] Task 2 #{mentions}") }
+    let(:addressed_issue) { create(:issue, project: project, assignees: [john_doe], author: author, description: "#{directly_addressed}\n- [ ] Task 1\n- [ ] Task 2") }
+    let(:unassigned_issue) { create(:issue, project: project, assignees: []) }
+    let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee], description: mentions) }
+    let(:addressed_confident_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee], description: directly_addressed) }
 
     describe '#new_issue' do
       it 'creates a todo if assigned' do
@@ -43,7 +43,7 @@
       end
 
       it 'creates a todo if assignee is the current user' do
-        unassigned_issue.update_attribute(:assignee, john_doe)
+        unassigned_issue.assignees = [john_doe]
         service.new_issue(unassigned_issue, john_doe)
 
         should_create_todo(user: john_doe, target: unassigned_issue, author: john_doe, action: Todo::ASSIGNED)
@@ -258,20 +258,20 @@
 
     describe '#reassigned_issue' do
       it 'creates a pending todo for new assignee' do
-        unassigned_issue.update_attribute(:assignee, john_doe)
+        unassigned_issue.assignees << john_doe
         service.reassigned_issue(unassigned_issue, author)
 
         should_create_todo(user: john_doe, target: unassigned_issue, action: Todo::ASSIGNED)
       end
 
       it 'does not create a todo if unassigned' do
-        issue.update_attribute(:assignee, nil)
+        issue.assignees.destroy_all
 
         should_not_create_any_todo { service.reassigned_issue(issue, author) }
       end
 
       it 'creates a todo if new assignee is the current user' do
-        unassigned_issue.update_attribute(:assignee, john_doe)
+        unassigned_issue.assignees << john_doe
         service.reassigned_issue(unassigned_issue, john_doe)
 
         should_create_todo(user: john_doe, target: unassigned_issue, author: john_doe, action: Todo::ASSIGNED)
@@ -361,7 +361,7 @@
     describe '#new_note' do
       let!(:first_todo) { create(:todo, :assigned, user: john_doe, project: project, target: issue, author: author) }
       let!(:second_todo) { create(:todo, :assigned, user: john_doe, project: project, target: issue, author: author) }
-      let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee) }
+      let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee]) }
       let(:note) { create(:note, project: project, noteable: issue, author: john_doe, note: mentions) }
       let(:addressed_note) { create(:note, project: project, noteable: issue, author: john_doe, note: directly_addressed) }
       let(:note_on_commit) { create(:note_on_commit, project: project, author: john_doe, note: mentions) }
@@ -854,7 +854,7 @@
   end
 
   it 'updates cached counts when a todo is created' do
-    issue = create(:issue, project: project, assignee: john_doe, author: author, description: mentions)
+    issue = create(:issue, project: project, assignees: [john_doe], author: author, description: mentions)
 
     expect(john_doe.todos_pending_count).to eq(0)
     expect(john_doe).to receive(:update_todos_count_cache).and_call_original
@@ -866,8 +866,8 @@
   end
 
   describe '#mark_todos_as_done' do
-    let(:issue) { create(:issue, project: project, author: author, assignee: john_doe) }
-    let(:another_issue) { create(:issue, project: project, author: author, assignee: john_doe) }
+    let(:issue) { create(:issue, project: project, author: author, assignees: [john_doe]) }
+    let(:another_issue) { create(:issue, project: project, author: author, assignees: [john_doe]) }
 
     it 'marks a relation of todos as done' do
       create(:todo, :mentioned, user: john_doe, target: issue, project: project)
diff --git a/spec/support/features/issuable_slash_commands_shared_examples.rb b/spec/support/features/issuable_slash_commands_shared_examples.rb
index 5bbe36d9b7fb8..f7499ede09cbd 100644
--- a/spec/support/features/issuable_slash_commands_shared_examples.rb
+++ b/spec/support/features/issuable_slash_commands_shared_examples.rb
@@ -62,7 +62,7 @@
         note = issuable.notes.user.first
 
         expect(note.note).to eq "Awesome!"
-        expect(issuable.assignee).to eq assignee
+        expect(issuable.assignees).to eq [assignee]
         expect(issuable.labels).to eq [label_bug]
         expect(issuable.milestone).to eq milestone
       end
@@ -80,7 +80,7 @@
         issuable.reload
 
         expect(issuable.notes.user).to be_empty
-        expect(issuable.assignee).to eq assignee
+        expect(issuable.assignees).to eq [assignee]
         expect(issuable.labels).to eq [label_bug]
         expect(issuable.milestone).to eq milestone
       end
diff --git a/spec/support/import_export/export_file_helper.rb b/spec/support/import_export/export_file_helper.rb
index 944ea30656fc9..57b6abe12b7ca 100644
--- a/spec/support/import_export/export_file_helper.rb
+++ b/spec/support/import_export/export_file_helper.rb
@@ -10,7 +10,7 @@ def setup_project
 
     create(:release, project: project)
 
-    issue = create(:issue, assignee: user, project: project)
+    issue = create(:issue, assignees: [user], project: project)
     snippet = create(:project_snippet, project: project)
     label = create(:label, project: project)
     milestone = create(:milestone, project: project)
diff --git a/spec/support/services/issuable_create_service_shared_examples.rb b/spec/support/services/issuable_create_service_shared_examples.rb
deleted file mode 100644
index 4f0c745b7ee48..0000000000000
--- a/spec/support/services/issuable_create_service_shared_examples.rb
+++ /dev/null
@@ -1,52 +0,0 @@
-shared_examples 'issuable create service' do
-  context 'asssignee_id' do
-    let(:assignee) { create(:user) }
-
-    before { project.team << [user, :master] }
-
-    it 'removes assignee_id when user id is invalid' do
-      opts = { title: 'Title', description: 'Description', assignee_id: -1 }
-
-      issuable = described_class.new(project, user, opts).execute
-
-      expect(issuable.assignee_id).to be_nil
-    end
-
-    it 'removes assignee_id when user id is 0' do
-      opts = { title: 'Title', description: 'Description',  assignee_id: 0 }
-
-      issuable = described_class.new(project, user, opts).execute
-
-      expect(issuable.assignee_id).to be_nil
-    end
-
-    it 'saves assignee when user id is valid' do
-      project.team << [assignee, :master]
-      opts = { title: 'Title', description: 'Description', assignee_id: assignee.id }
-
-      issuable = described_class.new(project, user, opts).execute
-
-      expect(issuable.assignee_id).to eq(assignee.id)
-    end
-
-    context "when issuable feature is private" do
-      before do
-        project.project_feature.update(issues_access_level: ProjectFeature::PRIVATE,
-                                       merge_requests_access_level: ProjectFeature::PRIVATE)
-      end
-
-      levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
-
-      levels.each do |level|
-        it "removes not authorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
-          project.update(visibility_level: level)
-          opts = { title: 'Title', description: 'Description', assignee_id: assignee.id }
-
-          issuable = described_class.new(project, user, opts).execute
-
-          expect(issuable.assignee_id).to be_nil
-        end
-      end
-    end
-  end
-end
diff --git a/spec/support/services/issuable_create_service_slash_commands_shared_examples.rb b/spec/support/services/issuable_create_service_slash_commands_shared_examples.rb
index 9e9cdf3e48b49..1dd3663b944fd 100644
--- a/spec/support/services/issuable_create_service_slash_commands_shared_examples.rb
+++ b/spec/support/services/issuable_create_service_slash_commands_shared_examples.rb
@@ -49,23 +49,7 @@
 
     it 'assigns and sets milestone to issuable' do
       expect(issuable).to be_persisted
-      expect(issuable.assignee).to eq(assignee)
-      expect(issuable.milestone).to eq(milestone)
-    end
-  end
-
-  context 'with assignee and milestone in params and command' do
-    let(:example_params) do
-      {
-        assignee: create(:user),
-        milestone_id: 1,
-        description: %(/assign @#{assignee.username}\n/milestone %"#{milestone.name}")
-      }
-    end
-
-    it 'assigns and sets milestone to issuable from command' do
-      expect(issuable).to be_persisted
-      expect(issuable.assignee).to eq(assignee)
+      expect(issuable.assignees).to eq([assignee])
       expect(issuable.milestone).to eq(milestone)
     end
   end
diff --git a/spec/support/services/issuable_update_service_shared_examples.rb b/spec/support/services/issuable_update_service_shared_examples.rb
index 49cea1e608cd9..8947f20562f8b 100644
--- a/spec/support/services/issuable_update_service_shared_examples.rb
+++ b/spec/support/services/issuable_update_service_shared_examples.rb
@@ -18,52 +18,4 @@ def update_issuable(opts)
       end
     end
   end
-
-  context 'asssignee_id' do
-    it 'does not update assignee when assignee_id is invalid' do
-      open_issuable.update(assignee_id: user.id)
-
-      update_issuable(assignee_id: -1)
-
-      expect(open_issuable.reload.assignee).to eq(user)
-    end
-
-    it 'unassigns assignee when user id is 0' do
-      open_issuable.update(assignee_id: user.id)
-
-      update_issuable(assignee_id: 0)
-
-      expect(open_issuable.assignee_id).to be_nil
-    end
-
-    it 'saves assignee when user id is valid' do
-      update_issuable(assignee_id: user.id)
-
-      expect(open_issuable.assignee_id).to eq(user.id)
-    end
-
-    it 'does not update assignee_id when user cannot read issue' do
-      non_member        = create(:user)
-      original_assignee = open_issuable.assignee
-
-      update_issuable(assignee_id: non_member.id)
-
-      expect(open_issuable.assignee_id).to eq(original_assignee.id)
-    end
-
-    context "when issuable feature is private" do
-      levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
-
-      levels.each do |level|
-        it "does not update with unauthorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
-          assignee = create(:user)
-          project.update(visibility_level: level)
-          feature_visibility_attr = :"#{open_issuable.model_name.plural}_access_level"
-          project.project_feature.update_attribute(feature_visibility_attr, ProjectFeature::PRIVATE)
-
-          expect{ update_issuable(assignee_id: assignee) }.not_to change{ open_issuable.assignee }
-        end
-      end
-    end
-  end
 end
diff --git a/spec/support/time_tracking_shared_examples.rb b/spec/support/time_tracking_shared_examples.rb
index 01bc80f957e24..bd5d71f964abe 100644
--- a/spec/support/time_tracking_shared_examples.rb
+++ b/spec/support/time_tracking_shared_examples.rb
@@ -34,7 +34,7 @@
     submit_time('/estimate 3w 1d 1h')
     submit_time('/remove_estimate')
 
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       expect(page).to have_content 'No estimate or time spent'
     end
   end
@@ -43,13 +43,13 @@
     submit_time('/spend 3w 1d 1h')
     submit_time('/remove_time_spent')
 
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       expect(page).to have_content 'No estimate or time spent'
     end
   end
 
   it 'shows the help state when icon is clicked' do
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       find('.help-button').click
       expect(page).to have_content 'Track time with slash commands'
       expect(page).to have_content 'Learn more'
@@ -57,7 +57,7 @@
   end
 
   it 'hides the help state when close icon is clicked' do
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       find('.help-button').click
       find('.close-help-button').click
 
@@ -67,7 +67,7 @@
   end
 
   it 'displays the correct help url' do
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       find('.help-button').click
 
       expect(find_link('Learn more')[:href]).to have_content('/help/workflow/time_tracking.md')
-- 
GitLab


From 1848ddf6562f1c0a7a53bd65c79e251f0726a532 Mon Sep 17 00:00:00 2001
From: Filipa Lacerda <filipa@gitlab.com>
Date: Thu, 4 May 2017 15:13:52 +0100
Subject: [PATCH 213/363] Add missing points

---
 doc/development/fe_guide/style_guide_js.md | 78 +++++++++++++++++++---
 1 file changed, 70 insertions(+), 8 deletions(-)

diff --git a/doc/development/fe_guide/style_guide_js.md b/doc/development/fe_guide/style_guide_js.md
index c8a77f6760708..a3f806c171382 100644
--- a/doc/development/fe_guide/style_guide_js.md
+++ b/doc/development/fe_guide/style_guide_js.md
@@ -11,7 +11,10 @@ See [our current .eslintrc][eslintrc] for specific rules and patterns.
 
 #### ESlint
 
-1. **Never** disable eslint rules unless you have a good reason.  You may see a lot of legacy files with `/* eslint-disable some-rule, some-other-rule */` at the top, but legacy files are a special case.  Any time you develop a new feature or refactor an existing one, you should abide by the eslint rules.
+1. **Never** disable eslint rules unless you have a good reason.  
+You may see a lot of legacy files with `/* eslint-disable some-rule, some-other-rule */`
+at the top, but legacy files are a special case.  Any time you develop a new feature or
+refactor an existing one, you should abide by the eslint rules.
 
 1. **Never Ever EVER** disable eslint globally for a file
   ```javascript
@@ -44,7 +47,8 @@ See [our current .eslintrc][eslintrc] for specific rules and patterns.
   1. [no-new][eslint-new]
   1. [class-methods-use-this][eslint-this]
 
-1. When they are needed _always_ place ESlint directive comment blocks on the first line of a script, followed by any global declarations, then a blank newline prior to any imports or code.
+1. When they are needed _always_ place ESlint directive comment blocks on the first line of a script,
+followed by any global declarations, then a blank newline prior to any imports or code.
   ```javascript
     // bad
     /* global Foo */
@@ -96,7 +100,8 @@ See [our current .eslintrc][eslintrc] for specific rules and patterns.
     export default Foo;
   ```
 
-1. Relative paths: Unless you are writing a test, always reference other scripts using relative paths instead of `~`
+1. Relative paths: Unless you are writing a test, always reference other scripts using
+relative paths instead of `~`
   * In **app/assets/javascripts**:
 
     ```javascript
@@ -116,7 +121,10 @@ See [our current .eslintrc][eslintrc] for specific rules and patterns.
       import Foo from '~/foo';
     ```
 
-1. Avoid using IIFE. Although we have a lot of examples of files which wrap their contents in IIFEs (immediately-invoked function expressions), this is no longer necessary after the transition from Sprockets to webpack. Do not use them anymore and feel free to remove them when refactoring legacy code.
+1. Avoid using IIFE. Although we have a lot of examples of files which wrap their
+contents in IIFEs (immediately-invoked function expressions),
+this is no longer necessary after the transition from Sprockets to webpack.
+Do not use them anymore and feel free to remove them when refactoring legacy code.
 
 1. Avoid adding to the global namespace.
   ```javascript
@@ -224,7 +232,45 @@ A forEach will cause side effects, it will be mutating the array being iterated.
       template: `<h1>I'm a component</h1>
     }
   ```
-1.
+1. The service has it's own file
+1. The store has it's own file
+1. Use a function in the bundle file to instantiate the Vue component:
+  ```javascript
+    // bad
+    class {
+      init() {
+        new Component({})
+      }
+    }
+
+    // good
+    document.addEventListener('DOMContentLoaded', () => new Vue({
+      el: '#element',
+      components: {
+        componentName
+      },
+      render: createElement => createElement('component-name'),
+    }));
+  ```
+
+1. Don not use a singleton for the service or the store
+  ```javascript
+    // bad
+    class Store {
+      constructor() {
+        if (!this.prototype.singleton) {
+          // do something
+        }
+      }
+    }
+
+    // good
+    class Store {
+      constructor() {
+        // do something
+      }
+    }
+  ```
 
 #### Naming
 1. **Extensions**: Use `.vue` extension for Vue components.
@@ -247,9 +293,8 @@ A forEach will cause side effects, it will be mutating the array being iterated.
     };
   ```
 
-1. **Props Naming:**
-1. Avoid using DOM component prop names.
-1. Use kebab-case instead of camelCase to provide props in templates.
+1. **Props Naming:**  Avoid using DOM component prop names.
+1. **Props Naming:** Use kebab-case instead of camelCase to provide props in templates.
   ```javascript
     // bad
     <component class="btn">
@@ -433,6 +478,23 @@ A forEach will cause side effects, it will be mutating the array being iterated.
   1. `beforeDestroy`
   1. `destroyed`
 
+#### Vue and Boostrap
+1. Tooltips: Do not rely on `has-tooltip` class name for vue components
+  ```javascript
+    // bad
+    <span class="has-tooltip">
+      Text
+    </span>
+
+    // good
+    <span data-toggle="tooltip">
+      Text
+    </span>
+  ```
+
+1. Tooltips: When using a tooltip, include the tooltip mixin
+
+1. Don't change `data-original-title`.
 
 ## SCSS
 - [SCSS](style_guide_scss.md)
-- 
GitLab


From b0a9a7cf5ee670ae07c4a9751d9c6e8bed1063b7 Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Thu, 4 May 2017 17:22:24 +0300
Subject: [PATCH 214/363] [Multiple issue assignee]: reslving some conflicts

---
 lib/api/v3/issues.rb                          |  4 ----
 spec/fixtures/api/schemas/issue.json          |  4 +---
 spec/requests/api/issues_spec.rb              |  8 --------
 spec/requests/api/v3/issues_spec.rb           | 13 +-----------
 .../slash_commands/interpret_service_spec.rb  | 20 -------------------
 5 files changed, 2 insertions(+), 47 deletions(-)

diff --git a/lib/api/v3/issues.rb b/lib/api/v3/issues.rb
index cedbeb0ded0f7..cb371fdbab8f7 100644
--- a/lib/api/v3/issues.rb
+++ b/lib/api/v3/issues.rb
@@ -61,12 +61,8 @@ def find_issues(args = {})
         end
         get do
           issues = find_issues(scope: 'authored')
-<<<<<<< HEAD
 
-          present paginate(issues), with: ::API::Entities::Issue, current_user: current_user
-=======
           present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
         end
       end
 
diff --git a/spec/fixtures/api/schemas/issue.json b/spec/fixtures/api/schemas/issue.json
index 983beb838b736..ff86437fdd5d4 100644
--- a/spec/fixtures/api/schemas/issue.json
+++ b/spec/fixtures/api/schemas/issue.json
@@ -40,13 +40,12 @@
         "additionalProperties": false
       }
     },
-<<<<<<< HEAD
     "assignee": {
       "id": { "type": "integet" },
       "name": { "type": "string" },
       "username": { "type": "string" },
       "avatar_url": { "type": "uri" }
-=======
+    },
     "assignees": {
       "type": "array",
       "items": {
@@ -64,7 +63,6 @@
           "avatar_url": { "type": "uri" }
         }
       }
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     },
     "subscribed": { "type": ["boolean", "null"] }
   },
diff --git a/spec/requests/api/issues_spec.rb b/spec/requests/api/issues_spec.rb
index f7c317021dc76..9b7b5798b71e2 100644
--- a/spec/requests/api/issues_spec.rb
+++ b/spec/requests/api/issues_spec.rb
@@ -774,24 +774,16 @@
 
     it 'creates a new project issue' do
       post api("/projects/#{project.id}/issues", user),
-<<<<<<< HEAD
-        title: 'new issue', labels: 'label, label2'
-=======
         title: 'new issue', labels: 'label, label2', weight: 3,
         assignee_ids: [user2.id]
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
 
       expect(response).to have_http_status(201)
       expect(json_response['title']).to eq('new issue')
       expect(json_response['description']).to be_nil
       expect(json_response['labels']).to eq(%w(label label2))
       expect(json_response['confidential']).to be_falsy
-<<<<<<< HEAD
-=======
-      expect(json_response['weight']).to eq(3)
       expect(json_response['assignee']['name']).to eq(user2.name)
       expect(json_response['assignees'].first['name']).to eq(user2.name)
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     end
 
     it 'creates a new confidential project issue' do
diff --git a/spec/requests/api/v3/issues_spec.rb b/spec/requests/api/v3/issues_spec.rb
index a1124aebcec9f..49191b2eb4e72 100644
--- a/spec/requests/api/v3/issues_spec.rb
+++ b/spec/requests/api/v3/issues_spec.rb
@@ -737,22 +737,14 @@
   describe "POST /projects/:id/issues" do
     it 'creates a new project issue' do
       post v3_api("/projects/#{project.id}/issues", user),
-<<<<<<< HEAD
-        title: 'new issue', labels: 'label, label2'
-=======
-        title: 'new issue', labels: 'label, label2', weight: 3, assignee_id: assignee.id
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
+        title: 'new issue', labels: 'label, label2', assignee_id: assignee.id
 
       expect(response).to have_http_status(201)
       expect(json_response['title']).to eq('new issue')
       expect(json_response['description']).to be_nil
       expect(json_response['labels']).to eq(%w(label label2))
       expect(json_response['confidential']).to be_falsy
-<<<<<<< HEAD
-=======
-      expect(json_response['weight']).to eq(3)
       expect(json_response['assignee']['name']).to eq(assignee.name)
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     end
 
     it 'creates a new confidential project issue' do
@@ -1149,8 +1141,6 @@
     end
   end
 
-<<<<<<< HEAD
-=======
   describe 'PUT /projects/:id/issues/:issue_id to update assignee' do
     it 'updates an issue with no assignee' do
       put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), assignee_id: 0
@@ -1199,7 +1189,6 @@
     end
   end
 
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
   describe "DELETE /projects/:id/issues/:issue_id" do
     it "rejects a non member from deleting an issue" do
       delete v3_api("/projects/#{project.id}/issues/#{issue.id}", non_member)
diff --git a/spec/services/slash_commands/interpret_service_spec.rb b/spec/services/slash_commands/interpret_service_spec.rb
index 865a7c698759e..9c16421fef03a 100644
--- a/spec/services/slash_commands/interpret_service_spec.rb
+++ b/spec/services/slash_commands/interpret_service_spec.rb
@@ -43,26 +43,6 @@
       end
     end
 
-<<<<<<< HEAD
-    shared_examples 'assign command' do
-      it 'fetches assignee and populates assignee_id if content contains /assign' do
-        _, updates = service.execute(content, issuable)
-
-        expect(updates).to eq(assignee_id: developer.id)
-      end
-    end
-
-    shared_examples 'unassign command' do
-      it 'populates assignee_id: nil if content contains /unassign' do
-        issuable.update!(assignee_id: developer.id)
-        _, updates = service.execute(content, issuable)
-
-        expect(updates).to eq(assignee_id: nil)
-      end
-    end
-
-=======
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     shared_examples 'milestone command' do
       it 'fetches milestone and populates milestone_id if content contains /milestone' do
         milestone # populate the milestone
-- 
GitLab


From 4d08ea6c8c27d6d52c2bc324c5c7ab800e05f297 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzegorz@gitlab.com>
Date: Thu, 4 May 2017 14:28:28 +0000
Subject: [PATCH 215/363] Document serializers

---
 app/serializers/README.md | 325 ++++++++++++++++++++++++++++++++++++++
 1 file changed, 325 insertions(+)
 create mode 100644 app/serializers/README.md

diff --git a/app/serializers/README.md b/app/serializers/README.md
new file mode 100644
index 0000000000000..0337f88db5f65
--- /dev/null
+++ b/app/serializers/README.md
@@ -0,0 +1,325 @@
+# Serializers
+
+This is a documentation for classes located in `app/serializers` directory.
+
+In GitLab, we use [grape-entities][grape-entity-project], accompanied by a
+serializer, to convert a Ruby object to its JSON representation.
+
+Serializers are typically used in controllers to build a JSON response
+that is usually consumed by a frontend code.
+
+## Why using a serializer is important?
+
+Using serializers, instead of `to_json` method, has several benefits:
+
+* it helps to prevent exposure of a sensitive data stored in the database
+* it makes it easier to test what should and should not be exposed
+* it makes it easier to reuse serialization entities that are building blocks
+* it makes it easier to move complexity from controllers to easily testable
+  classes
+* it encourages hiding complexity behind intentions-revealing interfaces
+* it makes it easier to take care about serialization performance concerns
+* it makes it easier to reduce merge conflicts between CE -> EE
+* it makes it easier to benefit from domain driven development techniques
+
+## What is a serializer?
+
+A serializer is a class that encapsulates all business rules for building a
+JSON response using serialization entities.
+
+It is designed to be testable and to support passing additional context from
+the controller.
+
+## What is a serialization entity?
+
+Entities are lightweight structures that allow to represent domain models
+in a consistent and abstracted way, and reuse them as building blocks to
+create a payload.
+
+Entities located in `app/serializers` are usually derived from a
+[`Grape::Entity`][grape-entity-class] class.
+
+Serialization entities that do require to have a knowledge about specific
+elements of the request, need to mix `RequestAwareEntity` in.
+
+A serialization entity usually maps a domain model class into its JSON
+representation. It rarely happens that a serialization entity exists without
+a corresponding domain model class. As an example, we have an `Issue` class and
+a corresponding `IssueSerializer`.
+
+Serialization entites are designed to reuse other serialization entities, which
+is a convenient way to create a multi-level JSON representation of a piece of
+a domain model you want to serialize.
+
+See [documentation for Grape Entites][grape-entity-readme] for more details.
+
+## How to implement a serializer?
+
+### Base implementation
+
+In order to effectively implement a serializer it is necessary to create a new
+class in `app/serializers`. See existing serializers as an example.
+
+A new serializer should inherit from a `BaseSerializer` class. It is necessary
+to specify which serialization entity will be used to serialize a resource.
+
+```ruby
+class MyResourceSerializer < BaseSerialize
+  entity MyResourceEntity
+end
+```
+
+The example above shows how a most simple serializer can look like.
+
+Given that the entity `MyResourceEntity` exists, you can now use
+`MyResourceSerializer` in the controller by creating an instance of it, and
+calling `MyResourceSerializer#represent(resource)` method.
+
+Note that a `resource` can be either a single object, an array of objects or an
+`ActiveRecord::Relation` object. A serialization entity should be smart enough
+to accurately represent each of these.
+
+It should not be necessary to use `Enumerable#map`, and it should be avoided
+from the performance reasons.
+
+### Choosing what gets serialized
+
+It often happens that you might want to use the same serializer in many places,
+but sometimes the intention is to only expose a small subset of object's
+attributes in one place, and a different subset in another.
+
+`BaseSerializer#represent(resource, opts = {})` method can take an additional
+hash argument, `opts`, that defines what is going to be serialized.
+
+`BaseSerializer` will pass these options to a serialization entity. See
+how it is [documented in the upstream project][grape-entity-only].
+
+With this approach you can extend the serializer to respond to methods that will
+create a JSON response according to your needs.
+
+```ruby
+class PipelineSerializer < BaseSerializer
+  entity PipelineEntity
+
+  def represent_details(resource)
+    represent(resource, only: [:details])
+  end
+
+  def represent_status(resource)
+    represent(resource, only: [:status])
+  end
+end
+```
+
+It is possible to use `only` and `except` keywords. Both keywords do support
+nested attributes, like `except: [:id, { user: [:id] }]`.
+
+Passing `only` and `except` to the `represent` method from a controller is
+possible, but it defies principles of encapsulation and testability, and it is
+better to avoid it, and to add a specific method to the serializer instead.
+
+### Reusing serialization entities from the API
+
+Public API in GitLab is implemented using [Grape][grape-project].
+
+Under the hood it also uses [`Grape::Entity`][grape-entity-class] classes.
+This means that it is possible to reuse these classes to implement internal
+serializers.
+
+You can either use such entity directly:
+
+```ruby
+class MyResourceSerializer < BaseSerializer
+  entity API::Entities::SomeEntity
+end
+```
+
+Or derive a new serialization entity class from it:
+
+```ruby
+class MyEntity < API::Entities::SomeEntity
+  include RequestAwareEntity
+
+  unexpose :something
+end
+```
+
+It might be a good idea to write specs for entities that do inherit from
+the API, because when API payloads are changed / extended, it is easy to forget
+about the impact on the internal API through a serializer that reuses API
+entities.
+
+It is usually safe to do that, because API entities rarely break backward
+compatibility, but additional exposure may have a performance impact when API
+gets extended significantly. Write tests that check if only necessary data is
+exposed.
+
+## How to write tests for a serializer?
+
+Like every other class in the project, creating a serializer warrants writing
+tests for it.
+
+It is usually a good idea to test each public method in the serializer against
+a valid payload. `BaseSerializer#represent` returns a hash, so it is possible
+to use usual RSpec matchers like `include`.
+
+Sometimes, when the payload is large, it makes sense to validate it entirely
+using `match_response_schema` matcher along with a new fixture that can be
+stored in `spec/fixtures/api/schemas/`. This matcher is using a `json-schema`
+gem, which is quite flexible, see a [documentation][json-schema-gem] for it.
+
+## How to use a serializer in a controller?
+
+Once a new serializer is implemented, it is possible to use it in a controller.
+
+Create an instance of the serializer and render the response.
+
+```ruby
+def index
+  format.json do
+    render json: MyResourceSerializer
+      .new(current_user: @current_user)
+      .represent_details(@project.resources)
+  nd
+end
+```
+
+If it is necessary to include additional information in the payload, it is
+possible to extend what is going to be rendered, the usual way:
+
+```ruby
+def index
+  format.json do
+    render json: {
+      resources: MyResourceSerializer
+        .new(current_user: @current_user)
+        .represent_details(@project.resources),
+      count: @project.resources.count
+    }
+  nd
+end
+```
+
+Note that in these examples an additional context is being passed to the
+serializer (`current_user: @current_user`).
+
+## How to pass an additional context from the controller?
+
+It is possible to pass an additional context from a controller to a
+serializer and each serialization entity that is used in the process.
+
+Serialization entities that do require an additional context have
+`RequestAwareEntity` concern mixed in. This piece of the code exposes a method
+called `request` in every serialization entity that is instantiated during
+serialization.
+
+An object returned by this method is an instance of `EntityRequest`, which
+behaves like an `OpenStruct` object, with the difference that it will raise
+an error if an unknown method is called.
+
+In other words, in the previous example, `request` method will return an
+instance of `EntityRequest` that responds to `current_user` method. It will be
+available in every serialization entity instantiated by `MyResourceSerializer`.
+
+`EntityRequest` is a workaround for [#20045][issue-20045] and is meant to be
+refactored soon. Please avoid passing an additional context that is not
+required by a serialization entity.
+
+At the moment, the context that is passed to entities most often is
+`current_user` and `project`.
+
+## How is this related to using presenters?
+
+Payload created by a serializer is usually a representation of the backed code,
+combined with the current request data. Therefore, technically, serializers
+are presenters that create payload consumed by a frontend code, usually Vue
+components.
+
+In GitLab, it is possible to use [presenters][presenters-readme], but
+`BaseSerializer` still needs to learn how to use it, see [#30898][issue-30898].
+
+It is possible to use presenters when serializer is used to represent only
+a single object. It is not supported when  `ActiveRecord::Relation` is being
+serialized.
+
+```ruby
+MyObjectSerializer.new.represent(object.present)
+```
+
+## Best practices
+
+1. Do not invoke a serializer from within a serialization entity.
+
+    If you need to use a serializer from within a serialization entity, it is
+    possible that you are missing a class for an important domain concept.
+
+    Consider creating a new domain class and a corresponding serialization
+    entity for it.
+
+1. Use only one approach to switch behavior of the serializer.
+
+    It is possible to use a few approaches to switch a behavior of the
+    serializer. Most common are using a [Fluent Interface][fluent-interface]
+    and creating a separate `represent_something` methods.
+
+    Whatever you choose, it might be better to use only one approach at a time.
+
+1. Do not forget about creating specs for serialization entities.
+
+    Writing tests for the serializer indeed does cover testing a behavior of
+    serialization entities that the serializer instantiates. However it might
+    be a good idea to write separate tests for entities as well, because these
+    are meant to be reused in different serializers, and a serializer can
+    change a behavior of a serialization entity.
+
+1. Use `ActiveRecord::Relation` where possible
+
+    Using an `ActiveRecord::Relation` might help from the performance perspective.
+
+1. Be diligent about passing an additional context from the controller.
+
+    Using `EntityRequest` and `RequestAwareEntity` is a workaround for the lack
+    of high-level mechanism. It is meant to be refactored, and current
+    implementation is error prone. Imagine the situation that one serialization
+    entity requires `request.user` attribute, but the second one wants
+    `request.current_user`. When it happens that these two entities are used in
+    the same serialization request, you might need to pass both parameters to
+    the serializer, which is obviously not a perfect situation.
+
+    When in doubt, pass only `current_user` and `project` if these are required.
+
+1. Keep performance concerns in mind
+
+    Using a serializer incorrectly can have significant impact on the
+    performance.
+
+    Because serializers are technically presenters, it is often necessary
+    to calculate, for example, paths to various controller-actions.
+    Since using URL helpers usually involve passing `project` and `namespace`
+    adding `includes(project: :namespace)` in the serializer, can help to avoid
+    N+1 queries.
+
+    Also, try to avoid using `Enumerable#map` or other methods that will
+    execute a database query eagerly.
+
+1. Avoid passing `only` and `except` from the controller.
+1. Write tests checking for N+1 queries.
+1. Write controller tests for actions / formats using serializers.
+1. Write tests that check if only necessary data is exposed.
+1. Write tests that check if no sensitive data is exposed.
+
+## Future
+
+* [Next iteration of serializers][issue-27569]
+
+[grape-project]: http://www.ruby-grape.org
+[grape-entity-project]: https://github.com/ruby-grape/grape-entity
+[grape-entity-readme]: https://github.com/ruby-grape/grape-entity/blob/master/README.md
+[grape-entity-class]: https://github.com/ruby-grape/grape-entity/blob/master/lib/grape_entity/entity.rb
+[grape-entity-only]: https://github.com/ruby-grape/grape-entity/blob/master/README.md#returning-only-the-fields-you-want
+[presenters-readme]: https://gitlab.com/gitlab-org/gitlab-ce/blob/master/app/presenters/README.md
+[fluent-interface]: https://en.wikipedia.org/wiki/Fluent_interface
+[json-schema-gem]: https://github.com/ruby-json-schema/json-schema
+[issue-20045]: https://gitlab.com/gitlab-org/gitlab-ce/issues/20045
+[issue-30898]: https://gitlab.com/gitlab-org/gitlab-ce/issues/30898
+[issue-27569]: https://gitlab.com/gitlab-org/gitlab-ce/issues/27569
-- 
GitLab


From 221dccab97766be810d87c6eb1d120145b41b364 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=A9my=20Coutable?= <remy@rymai.me>
Date: Thu, 4 May 2017 16:34:52 +0200
Subject: [PATCH 216/363] Include the bundler:audit job into the
 static-analysis job
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Signed-off-by: Rémy Coutable <remy@rymai.me>
---
 .gitlab-ci.yml          | 12 ------------
 scripts/static-analysis |  1 +
 2 files changed, 1 insertion(+), 12 deletions(-)

diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index 7f665f191326f..44620d390ad33 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -412,18 +412,6 @@ rake karma:
     paths:
     - coverage-javascript/
 
-bundler:audit:
-  stage: test
-  <<: *ruby-static-analysis
-  <<: *dedicated-runner
-  only:
-    - master@gitlab-org/gitlab-ce
-    - master@gitlab-org/gitlab-ee
-    - master@gitlab/gitlabhq
-    - master@gitlab/gitlab-ee
-  script:
-    - "bundle exec bundle-audit check --update --ignore CVE-2016-4658"
-
 .migration-paths: &migration-paths
   stage: test
   <<: *dedicated-runner
diff --git a/scripts/static-analysis b/scripts/static-analysis
index 1bd6b33983090..7dc8f67903640 100755
--- a/scripts/static-analysis
+++ b/scripts/static-analysis
@@ -3,6 +3,7 @@
 require ::File.expand_path('../lib/gitlab/popen', __dir__)
 
 tasks = [
+  %w[bundle exec bundle-audit check --update --ignore CVE-2016-4658],
   %w[bundle exec rake config_lint],
   %w[bundle exec rake flay],
   %w[bundle exec rake haml_lint],
-- 
GitLab


From 78d059141b5f3345d797992e03680654ce762c76 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Thu, 4 May 2017 16:41:07 +0200
Subject: [PATCH 217/363] add more examples for testing SQL

---
 .../projects/propagate_service_spec.rb        | 30 +++++++++++++++++++
 1 file changed, 30 insertions(+)

diff --git a/spec/services/projects/propagate_service_spec.rb b/spec/services/projects/propagate_service_spec.rb
index ee40a7ecbabce..409c14655cd33 100644
--- a/spec/services/projects/propagate_service_spec.rb
+++ b/spec/services/projects/propagate_service_spec.rb
@@ -22,8 +22,38 @@
         to change { Service.count }.by(1)
     end
 
+    it 'creates services for a project that has another service' do
+      other_service = BambooService.create(
+        template: true,
+        active: true,
+        properties: {
+          bamboo_url: 'http://gitlab.com',
+          username: 'mic',
+          password: "password",
+          build_key: 'build'
+        }
+      )
+
+      Service.build_from_template(project.id, other_service).save!
+
+      expect { described_class.propagate!(service_template) }.
+        to change { Service.count }.by(1)
+    end
+
     it 'does not create the service if it exists already' do
+      other_service = BambooService.create(
+        template: true,
+        active: true,
+        properties: {
+          bamboo_url: 'http://gitlab.com',
+          username: 'mic',
+          password: "password",
+          build_key: 'build'
+        }
+      )
+
       Service.build_from_template(project.id, service_template).save!
+      Service.build_from_template(project.id, other_service).save!
 
       expect { described_class.propagate!(service_template) }.
         not_to change { Service.count }
-- 
GitLab


From 9904c12746e744c4e271fe454b04a54dfb9b668e Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Thu, 4 May 2017 15:57:08 +0100
Subject: [PATCH 218/363] Updated webpack config

---
 app/assets/javascripts/deploy_keys/components/app.vue | 6 +-----
 config/webpack.config.js                              | 1 +
 2 files changed, 2 insertions(+), 5 deletions(-)

diff --git a/app/assets/javascripts/deploy_keys/components/app.vue b/app/assets/javascripts/deploy_keys/components/app.vue
index 2ba5001cf56a0..a6552125e6692 100644
--- a/app/assets/javascripts/deploy_keys/components/app.vue
+++ b/app/assets/javascripts/deploy_keys/components/app.vue
@@ -32,7 +32,6 @@
     methods: {
       fetchKeys() {
         this.isLoading = true;
-        this.store.keys = {};
 
         this.service.getKeys()
           .then((data) => {
@@ -46,9 +45,6 @@
           .then(() => this.fetchKeys())
           .catch(() => new Flash('Error enabling deploy key'));
       },
-      removeKey(deployKey) {
-        this.disableKey(deployKey);
-      },
       disableKey(deployKey) {
         // eslint-disable-next-line no-alert
         if (confirm('You are going to remove this deploy key. Are you sure?')) {
@@ -62,7 +58,7 @@
       this.service = new DeployKeysService(this.endpoint);
 
       eventHub.$on('enable.key', this.enableKey);
-      eventHub.$on('remove.key', this.removeKey);
+      eventHub.$on('remove.key', this.disableKey);
       eventHub.$on('disable.key', this.disableKey);
     },
     mounted() {
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 742d22d0c1fec..239bb5ec43620 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -123,6 +123,7 @@ var config = {
         'boards',
         'commit_pipelines',
         'cycle_analytics',
+        'deploy_keys',
         'diff_notes',
         'environments',
         'environments_folder',
-- 
GitLab


From 45e4c665653cd511b0d96119d3973652c43238bb Mon Sep 17 00:00:00 2001
From: Rares Sfirlogea <rrr.junior@gmail.com>
Date: Wed, 16 Nov 2016 12:09:09 +0100
Subject: [PATCH 219/363] Display slash commands outcome when previewing
 Markdown

Remove slash commands from Markdown preview and display their outcome next to
the text field.
Introduce new "explanation" block to our slash commands DSL.
Introduce optional "parse_params" block to slash commands DSL that allows to
process a parameter before it is passed to "explanation" or "command" blocks.
Pass path for previewing Markdown as "data" attribute instead of setting
a variable on "window".
---
 app/assets/javascripts/preview_markdown.js    |  48 +++-
 app/controllers/concerns/markdown_preview.rb  |  19 --
 app/controllers/projects/wikis_controller.rb  |  13 +-
 app/controllers/projects_controller.rb        |  11 +-
 app/controllers/snippets_controller.rb        |  10 +-
 app/helpers/gitlab_routing_helper.rb          |   4 +
 app/services/preview_markdown_service.rb      |  45 ++++
 .../slash_commands/interpret_service.rb       | 212 ++++++++++++++----
 app/views/groups/milestones/new.html.haml     |   2 +-
 app/views/layouts/project.html.haml           |   5 -
 app/views/projects/_md_preview.html.haml      |   7 +-
 app/views/projects/milestones/_form.html.haml |   2 +-
 app/views/projects/notes/_edit_form.html.haml |   2 +-
 app/views/projects/notes/_form.html.haml      |   6 +-
 .../projects/notes/_notes_with_form.html.haml |   2 +-
 app/views/projects/releases/edit.html.haml    |   2 +-
 app/views/projects/tags/new.html.haml         |   2 +-
 app/views/projects/wikis/_form.html.haml      |   2 +-
 app/views/shared/issuable/_form.html.haml     |   2 +-
 .../issuable/form/_description.html.haml      |  13 +-
 .../preview-separate-slash-commands.yml       |   4 +
 .../slash_commands/command_definition.rb      |  46 +++-
 lib/gitlab/slash_commands/dsl.rb              |  52 ++++-
 .../slash_commands/command_definition_spec.rb |  52 +++++
 spec/lib/gitlab/slash_commands/dsl_spec.rb    |  66 ++++--
 .../services/preview_markdown_service_spec.rb |  67 ++++++
 .../slash_commands/interpret_service_spec.rb  | 207 +++++++++++++++++
 ...issuable_slash_commands_shared_examples.rb |  15 ++
 28 files changed, 785 insertions(+), 133 deletions(-)
 delete mode 100644 app/controllers/concerns/markdown_preview.rb
 create mode 100644 app/services/preview_markdown_service.rb
 create mode 100644 changelogs/unreleased/preview-separate-slash-commands.yml
 create mode 100644 spec/services/preview_markdown_service_spec.rb

diff --git a/app/assets/javascripts/preview_markdown.js b/app/assets/javascripts/preview_markdown.js
index 07eea98e737b1..4a3df2fd465c8 100644
--- a/app/assets/javascripts/preview_markdown.js
+++ b/app/assets/javascripts/preview_markdown.js
@@ -2,8 +2,9 @@
 
 // MarkdownPreview
 //
-// Handles toggling the "Write" and "Preview" tab clicks, rendering the preview,
-// and showing a warning when more than `x` users are referenced.
+// Handles toggling the "Write" and "Preview" tab clicks, rendering the preview
+// (including the explanation of slash commands), and showing a warning when
+// more than `x` users are referenced.
 //
 (function () {
   var lastTextareaPreviewed;
@@ -17,32 +18,45 @@
 
     // Minimum number of users referenced before triggering a warning
     MarkdownPreview.prototype.referenceThreshold = 10;
+    MarkdownPreview.prototype.emptyMessage = 'Nothing to preview.';
 
     MarkdownPreview.prototype.ajaxCache = {};
 
     MarkdownPreview.prototype.showPreview = function ($form) {
       var mdText;
       var preview = $form.find('.js-md-preview');
+      var url = preview.data('url');
       if (preview.hasClass('md-preview-loading')) {
         return;
       }
       mdText = $form.find('textarea.markdown-area').val();
 
       if (mdText.trim().length === 0) {
-        preview.text('Nothing to preview.');
+        preview.text(this.emptyMessage);
         this.hideReferencedUsers($form);
       } else {
         preview.addClass('md-preview-loading').text('Loading...');
-        this.fetchMarkdownPreview(mdText, (function (response) {
-          preview.removeClass('md-preview-loading').html(response.body);
+        this.fetchMarkdownPreview(mdText, url, (function (response) {
+          var body;
+          if (response.body.length > 0) {
+            body = response.body;
+          } else {
+            body = this.emptyMessage;
+          }
+
+          preview.removeClass('md-preview-loading').html(body);
           preview.renderGFM();
           this.renderReferencedUsers(response.references.users, $form);
+
+          if (response.references.commands) {
+            this.renderReferencedCommands(response.references.commands, $form);
+          }
         }).bind(this));
       }
     };
 
-    MarkdownPreview.prototype.fetchMarkdownPreview = function (text, success) {
-      if (!window.preview_markdown_path) {
+    MarkdownPreview.prototype.fetchMarkdownPreview = function (text, url, success) {
+      if (!url) {
         return;
       }
       if (text === this.ajaxCache.text) {
@@ -51,7 +65,7 @@
       }
       $.ajax({
         type: 'POST',
-        url: window.preview_markdown_path,
+        url: url,
         data: {
           text: text
         },
@@ -83,6 +97,22 @@
       }
     };
 
+    MarkdownPreview.prototype.hideReferencedCommands = function ($form) {
+      $form.find('.referenced-commands').hide();
+    };
+
+    MarkdownPreview.prototype.renderReferencedCommands = function (commands, $form) {
+      var referencedCommands;
+      referencedCommands = $form.find('.referenced-commands');
+      if (commands.length > 0) {
+        referencedCommands.html(commands);
+        referencedCommands.show();
+      } else {
+        referencedCommands.html('');
+        referencedCommands.hide();
+      }
+    };
+
     return MarkdownPreview;
   }());
 
@@ -137,6 +167,8 @@
     $form.find('.md-write-holder').show();
     $form.find('textarea.markdown-area').focus();
     $form.find('.md-preview-holder').hide();
+
+    markdownPreview.hideReferencedCommands($form);
   });
 
   $(document).on('markdown-preview:toggle', function (e, keyboardEvent) {
diff --git a/app/controllers/concerns/markdown_preview.rb b/app/controllers/concerns/markdown_preview.rb
deleted file mode 100644
index 40eff267348fa..0000000000000
--- a/app/controllers/concerns/markdown_preview.rb
+++ /dev/null
@@ -1,19 +0,0 @@
-module MarkdownPreview
-  private
-
-  def render_markdown_preview(text, markdown_context = {})
-    render json: {
-      body: view_context.markdown(text, markdown_context),
-      references: {
-        users: preview_referenced_users(text)
-      }
-    }
-  end
-
-  def preview_referenced_users(text)
-    extractor = Gitlab::ReferenceExtractor.new(@project, current_user)
-    extractor.analyze(text, author: current_user)
-
-    extractor.users.map(&:username)
-  end
-end
diff --git a/app/controllers/projects/wikis_controller.rb b/app/controllers/projects/wikis_controller.rb
index 96125684da0d0..887d18dbec3ce 100644
--- a/app/controllers/projects/wikis_controller.rb
+++ b/app/controllers/projects/wikis_controller.rb
@@ -1,6 +1,4 @@
 class Projects::WikisController < Projects::ApplicationController
-  include MarkdownPreview
-
   before_action :authorize_read_wiki!
   before_action :authorize_create_wiki!, only: [:edit, :create, :history]
   before_action :authorize_admin_wiki!, only: :destroy
@@ -97,9 +95,14 @@ def git_access
   end
 
   def preview_markdown
-    context = { pipeline: :wiki, project_wiki: @project_wiki, page_slug: params[:id] }
-
-    render_markdown_preview(params[:text], context)
+    result = PreviewMarkdownService.new(@project, current_user, params).execute
+
+    render json: {
+      body: view_context.markdown(result[:text], pipeline: :wiki, project_wiki: @project_wiki, page_slug: params[:id]),
+      references: {
+        users: result[:users]
+      }
+    }
   end
 
   private
diff --git a/app/controllers/projects_controller.rb b/app/controllers/projects_controller.rb
index 9f6ee4826e6ee..69310b26e76cc 100644
--- a/app/controllers/projects_controller.rb
+++ b/app/controllers/projects_controller.rb
@@ -1,7 +1,6 @@
 class ProjectsController < Projects::ApplicationController
   include IssuableCollections
   include ExtractsPath
-  include MarkdownPreview
 
   before_action :authenticate_user!, except: [:index, :show, :activity, :refs]
   before_action :project, except: [:index, :new, :create]
@@ -240,7 +239,15 @@ def refs
   end
 
   def preview_markdown
-    render_markdown_preview(params[:text])
+    result = PreviewMarkdownService.new(@project, current_user, params).execute
+
+    render json: {
+      body: view_context.markdown(result[:text]),
+      references: {
+        users: result[:users],
+        commands: view_context.markdown(result[:commands])
+      }
+    }
   end
 
   private
diff --git a/app/controllers/snippets_controller.rb b/app/controllers/snippets_controller.rb
index da1ae9a34d980..afed27a41d1fb 100644
--- a/app/controllers/snippets_controller.rb
+++ b/app/controllers/snippets_controller.rb
@@ -3,7 +3,6 @@ class SnippetsController < ApplicationController
   include ToggleAwardEmoji
   include SpammableActions
   include SnippetsActions
-  include MarkdownPreview
   include RendersBlob
 
   before_action :snippet, only: [:show, :edit, :destroy, :update, :raw]
@@ -90,7 +89,14 @@ def destroy
   end
 
   def preview_markdown
-    render_markdown_preview(params[:text], skip_project_check: true)
+    result = PreviewMarkdownService.new(@project, current_user, params).execute
+
+    render json: {
+      body: view_context.markdown(result[:text], skip_project_check: true),
+      references: {
+        users: result[:users]
+      }
+    }
   end
 
   protected
diff --git a/app/helpers/gitlab_routing_helper.rb b/app/helpers/gitlab_routing_helper.rb
index e9b7cbbad6a5e..d3af241dd0e60 100644
--- a/app/helpers/gitlab_routing_helper.rb
+++ b/app/helpers/gitlab_routing_helper.rb
@@ -122,6 +122,10 @@ def project_snippet_url(entity, *args)
     namespace_project_snippet_url(entity.project.namespace, entity.project, entity, *args)
   end
 
+  def preview_markdown_path(project, *args)
+    preview_markdown_namespace_project_path(project.namespace, project, *args)
+  end
+
   def toggle_subscription_path(entity, *args)
     if entity.is_a?(Issue)
       toggle_subscription_namespace_project_issue_path(entity.project.namespace, entity.project, entity)
diff --git a/app/services/preview_markdown_service.rb b/app/services/preview_markdown_service.rb
new file mode 100644
index 0000000000000..10d45bbf73c20
--- /dev/null
+++ b/app/services/preview_markdown_service.rb
@@ -0,0 +1,45 @@
+class PreviewMarkdownService < BaseService
+  def execute
+    text, commands = explain_slash_commands(params[:text])
+    users = find_user_references(text)
+
+    success(
+      text: text,
+      users: users,
+      commands: commands.join(' ')
+    )
+  end
+
+  private
+
+  def explain_slash_commands(text)
+    return text, [] unless %w(Issue MergeRequest).include?(commands_target_type)
+
+    slash_commands_service = SlashCommands::InterpretService.new(project, current_user)
+    slash_commands_service.explain(text, find_commands_target)
+  end
+
+  def find_user_references(text)
+    extractor = Gitlab::ReferenceExtractor.new(project, current_user)
+    extractor.analyze(text, author: current_user)
+    extractor.users.map(&:username)
+  end
+
+  def find_commands_target
+    if commands_target_id.present?
+      finder = commands_target_type == 'Issue' ? IssuesFinder : MergeRequestsFinder
+      finder.new(current_user, project_id: project.id).find(commands_target_id)
+    else
+      collection = commands_target_type == 'Issue' ? project.issues : project.merge_requests
+      collection.build
+    end
+  end
+
+  def commands_target_type
+    params[:slash_commands_target_type]
+  end
+
+  def commands_target_id
+    params[:slash_commands_target_id]
+  end
+end
diff --git a/app/services/slash_commands/interpret_service.rb b/app/services/slash_commands/interpret_service.rb
index 6aeebc26685ae..f1bbc7032d566 100644
--- a/app/services/slash_commands/interpret_service.rb
+++ b/app/services/slash_commands/interpret_service.rb
@@ -2,7 +2,7 @@ module SlashCommands
   class InterpretService < BaseService
     include Gitlab::SlashCommands::Dsl
 
-    attr_reader :issuable, :options
+    attr_reader :issuable
 
     # Takes a text and interprets the commands that are extracted from it.
     # Returns the content without commands, and hash of changes to be applied to a record.
@@ -12,23 +12,21 @@ def execute(content, issuable)
       @issuable = issuable
       @updates = {}
 
-      opts = {
-        issuable:     issuable,
-        current_user: current_user,
-        project:      project,
-        params:       params
-      }
-
-      content, commands = extractor.extract_commands(content, opts)
+      content, commands = extractor.extract_commands(content, context)
+      extract_updates(commands, context)
+      [content, @updates]
+    end
 
-      commands.each do |name, arg|
-        definition = self.class.command_definitions_by_name[name.to_sym]
-        next unless definition
+    # Takes a text and interprets the commands that are extracted from it.
+    # Returns the content without commands, and array of changes explained.
+    def explain(content, issuable)
+      return [content, []] unless current_user.can?(:use_slash_commands)
 
-        definition.execute(self, opts, arg)
-      end
+      @issuable = issuable
 
-      [content, @updates]
+      content, commands = extractor.extract_commands(content, context)
+      commands = explain_commands(commands, context)
+      [content, commands]
     end
 
     private
@@ -40,6 +38,9 @@ def extractor
     desc do
       "Close this #{issuable.to_ability_name.humanize(capitalize: false)}"
     end
+    explanation do
+      "Closes this #{issuable.to_ability_name.humanize(capitalize: false)}."
+    end
     condition do
       issuable.persisted? &&
         issuable.open? &&
@@ -52,6 +53,9 @@ def extractor
     desc do
       "Reopen this #{issuable.to_ability_name.humanize(capitalize: false)}"
     end
+    explanation do
+      "Reopens this #{issuable.to_ability_name.humanize(capitalize: false)}."
+    end
     condition do
       issuable.persisted? &&
         issuable.closed? &&
@@ -62,6 +66,7 @@ def extractor
     end
 
     desc 'Merge (when the pipeline succeeds)'
+    explanation 'Merges this merge request when the pipeline succeeds.'
     condition do
       last_diff_sha = params && params[:merge_request_diff_head_sha]
       issuable.is_a?(MergeRequest) &&
@@ -73,6 +78,9 @@ def extractor
     end
 
     desc 'Change title'
+    explanation do |title_param|
+      "Changes the title to \"#{title_param}\"."
+    end
     params '<New title>'
     condition do
       issuable.persisted? &&
@@ -83,18 +91,25 @@ def extractor
     end
 
     desc 'Assign'
+    explanation do |user|
+      "Assigns #{user.to_reference}." if user
+    end
     params '@user'
     condition do
       current_user.can?(:"admin_#{issuable.to_ability_name}", project)
     end
-    command :assign do |assignee_param|
-      user = extract_references(assignee_param, :user).first
-      user ||= User.find_by(username: assignee_param)
-
+    parse_params do |assignee_param|
+      extract_references(assignee_param, :user).first ||
+        User.find_by(username: assignee_param)
+    end
+    command :assign do |user|
       @updates[:assignee_id] = user.id if user
     end
 
     desc 'Remove assignee'
+    explanation do
+      "Removes assignee #{issuable.assignee.to_reference}."
+    end
     condition do
       issuable.persisted? &&
         issuable.assignee_id? &&
@@ -105,19 +120,26 @@ def extractor
     end
 
     desc 'Set milestone'
+    explanation do |milestone|
+      "Sets the milestone to #{milestone.to_reference}." if milestone
+    end
     params '%"milestone"'
     condition do
       current_user.can?(:"admin_#{issuable.to_ability_name}", project) &&
         project.milestones.active.any?
     end
-    command :milestone do |milestone_param|
-      milestone = extract_references(milestone_param, :milestone).first
-      milestone ||= project.milestones.find_by(title: milestone_param.strip)
-
+    parse_params do |milestone_param|
+      extract_references(milestone_param, :milestone).first ||
+        project.milestones.find_by(title: milestone_param.strip)
+    end
+    command :milestone do |milestone|
       @updates[:milestone_id] = milestone.id if milestone
     end
 
     desc 'Remove milestone'
+    explanation do
+      "Removes #{issuable.milestone.to_reference(format: :name)} milestone."
+    end
     condition do
       issuable.persisted? &&
         issuable.milestone_id? &&
@@ -128,6 +150,11 @@ def extractor
     end
 
     desc 'Add label(s)'
+    explanation do |labels_param|
+      labels = find_label_references(labels_param)
+
+      "Adds #{labels.join(' ')} #{'label'.pluralize(labels.count)}." if labels.any?
+    end
     params '~label1 ~"label 2"'
     condition do
       available_labels = LabelsFinder.new(current_user, project_id: project.id).execute
@@ -147,6 +174,14 @@ def extractor
     end
 
     desc 'Remove all or specific label(s)'
+    explanation do |labels_param = nil|
+      if labels_param.present?
+        labels = find_label_references(labels_param)
+        "Removes #{labels.join(' ')} #{'label'.pluralize(labels.count)}." if labels.any?
+      else
+        'Removes all labels.'
+      end
+    end
     params '~label1 ~"label 2"'
     condition do
       issuable.persisted? &&
@@ -169,6 +204,10 @@ def extractor
     end
 
     desc 'Replace all label(s)'
+    explanation do |labels_param|
+      labels = find_label_references(labels_param)
+      "Replaces all labels with #{labels.join(' ')} #{'label'.pluralize(labels.count)}." if labels.any?
+    end
     params '~label1 ~"label 2"'
     condition do
       issuable.persisted? &&
@@ -187,6 +226,7 @@ def extractor
     end
 
     desc 'Add a todo'
+    explanation 'Adds a todo.'
     condition do
       issuable.persisted? &&
         !TodoService.new.todo_exist?(issuable, current_user)
@@ -196,6 +236,7 @@ def extractor
     end
 
     desc 'Mark todo as done'
+    explanation 'Marks todo as done.'
     condition do
       issuable.persisted? &&
         TodoService.new.todo_exist?(issuable, current_user)
@@ -205,6 +246,9 @@ def extractor
     end
 
     desc 'Subscribe'
+    explanation do
+      "Subscribes to this #{issuable.to_ability_name.humanize(capitalize: false)}."
+    end
     condition do
       issuable.persisted? &&
         !issuable.subscribed?(current_user, project)
@@ -214,6 +258,9 @@ def extractor
     end
 
     desc 'Unsubscribe'
+    explanation do
+      "Unsubscribes from this #{issuable.to_ability_name.humanize(capitalize: false)}."
+    end
     condition do
       issuable.persisted? &&
         issuable.subscribed?(current_user, project)
@@ -223,18 +270,23 @@ def extractor
     end
 
     desc 'Set due date'
+    explanation do |due_date|
+      "Sets the due date to #{due_date.to_s(:medium)}." if due_date
+    end
     params '<in 2 days | this Friday | December 31st>'
     condition do
       issuable.respond_to?(:due_date) &&
         current_user.can?(:"admin_#{issuable.to_ability_name}", project)
     end
-    command :due do |due_date_param|
-      due_date = Chronic.parse(due_date_param).try(:to_date)
-
+    parse_params do |due_date_param|
+      Chronic.parse(due_date_param).try(:to_date)
+    end
+    command :due do |due_date|
       @updates[:due_date] = due_date if due_date
     end
 
     desc 'Remove due date'
+    explanation 'Removes the due date.'
     condition do
       issuable.persisted? &&
         issuable.respond_to?(:due_date) &&
@@ -245,8 +297,11 @@ def extractor
       @updates[:due_date] = nil
     end
 
-    desc do
-      "Toggle the Work In Progress status"
+    desc 'Toggle the Work In Progress status'
+    explanation do
+      verb = issuable.work_in_progress? ? 'Unmarks' : 'Marks'
+      noun = issuable.to_ability_name.humanize(capitalize: false)
+      "#{verb} this #{noun} as Work In Progress."
     end
     condition do
       issuable.persisted? &&
@@ -257,45 +312,72 @@ def extractor
       @updates[:wip_event] = issuable.work_in_progress? ? 'unwip' : 'wip'
     end
 
-    desc 'Toggle emoji reward'
+    desc 'Toggle emoji award'
+    explanation do |name|
+      "Toggles :#{name}: emoji award." if name
+    end
     params ':emoji:'
     condition do
       issuable.persisted?
     end
-    command :award do |emoji|
-      name = award_emoji_name(emoji)
+    parse_params do |emoji_param|
+      match = emoji_param.match(Banzai::Filter::EmojiFilter.emoji_pattern)
+      match[1] if match
+    end
+    command :award do |name|
       if name && issuable.user_can_award?(current_user, name)
         @updates[:emoji_award] = name
       end
     end
 
     desc 'Set time estimate'
+    explanation do |time_estimate|
+      time_estimate = Gitlab::TimeTrackingFormatter.output(time_estimate)
+
+      "Sets time estimate to #{time_estimate}." if time_estimate
+    end
     params '<1w 3d 2h 14m>'
     condition do
       current_user.can?(:"admin_#{issuable.to_ability_name}", project)
     end
-    command :estimate do |raw_duration|
-      time_estimate = Gitlab::TimeTrackingFormatter.parse(raw_duration)
-
+    parse_params do |raw_duration|
+      Gitlab::TimeTrackingFormatter.parse(raw_duration)
+    end
+    command :estimate do |time_estimate|
       if time_estimate
         @updates[:time_estimate] = time_estimate
       end
     end
 
     desc 'Add or substract spent time'
+    explanation do |time_spent|
+      if time_spent
+        if time_spent > 0
+          verb = 'Adds'
+          value = time_spent
+        else
+          verb = 'Substracts'
+          value = -time_spent
+        end
+
+        "#{verb} #{Gitlab::TimeTrackingFormatter.output(value)} spent time."
+      end
+    end
     params '<1h 30m | -1h 30m>'
     condition do
       current_user.can?(:"admin_#{issuable.to_ability_name}", issuable)
     end
-    command :spend do |raw_duration|
-      time_spent = Gitlab::TimeTrackingFormatter.parse(raw_duration)
-
+    parse_params do |raw_duration|
+      Gitlab::TimeTrackingFormatter.parse(raw_duration)
+    end
+    command :spend do |time_spent|
       if time_spent
         @updates[:spend_time] = { duration: time_spent, user: current_user }
       end
     end
 
     desc 'Remove time estimate'
+    explanation 'Removes time estimate.'
     condition do
       issuable.persisted? &&
         current_user.can?(:"admin_#{issuable.to_ability_name}", project)
@@ -305,6 +387,7 @@ def extractor
     end
 
     desc 'Remove spent time'
+    explanation 'Removes spent time.'
     condition do
       issuable.persisted? &&
         current_user.can?(:"admin_#{issuable.to_ability_name}", project)
@@ -318,19 +401,28 @@ def extractor
     params '@user'
     command :cc
 
-    desc 'Defines target branch for MR'
+    desc 'Define target branch for MR'
+    explanation do |branch_name|
+      "Sets target branch to #{branch_name}."
+    end
     params '<Local branch name>'
     condition do
       issuable.respond_to?(:target_branch) &&
         (current_user.can?(:"update_#{issuable.to_ability_name}", issuable) ||
           issuable.new_record?)
     end
-    command :target_branch do |target_branch_param|
-      branch_name = target_branch_param.strip
+    parse_params do |target_branch_param|
+      target_branch_param.strip
+    end
+    command :target_branch do |branch_name|
       @updates[:target_branch] = branch_name if project.repository.branch_names.include?(branch_name)
     end
 
     desc 'Move issue from one column of the board to another'
+    explanation do |target_list_name|
+      label = find_label_references(target_list_name).first
+      "Moves issue to #{label} column in the board." if label
+    end
     params '~"Target column"'
     condition do
       issuable.is_a?(Issue) &&
@@ -352,11 +444,35 @@ def extractor
       end
     end
 
+    def find_labels(labels_param)
+      extract_references(labels_param, :label) |
+        LabelsFinder.new(current_user, project_id: project.id, name: labels_param.split).execute
+    end
+
+    def find_label_references(labels_param)
+      find_labels(labels_param).map(&:to_reference)
+    end
+
     def find_label_ids(labels_param)
-      label_ids_by_reference = extract_references(labels_param, :label).map(&:id)
-      labels_ids_by_name = LabelsFinder.new(current_user, project_id: project.id, name: labels_param.split).execute.select(:id)
+      find_labels(labels_param).map(&:id)
+    end
 
-      label_ids_by_reference | labels_ids_by_name
+    def explain_commands(commands, opts)
+      commands.map do |name, arg|
+        definition = self.class.definition_by_name(name)
+        next unless definition
+
+        definition.explain(self, opts, arg)
+      end.compact
+    end
+
+    def extract_updates(commands, opts)
+      commands.each do |name, arg|
+        definition = self.class.definition_by_name(name)
+        next unless definition
+
+        definition.execute(self, opts, arg)
+      end
     end
 
     def extract_references(arg, type)
@@ -366,9 +482,13 @@ def extract_references(arg, type)
       ext.references(type)
     end
 
-    def award_emoji_name(emoji)
-      match = emoji.match(Banzai::Filter::EmojiFilter.emoji_pattern)
-      match[1] if match
+    def context
+      {
+        issuable: issuable,
+        current_user: current_user,
+        project: project,
+        params: params
+      }
     end
   end
 end
diff --git a/app/views/groups/milestones/new.html.haml b/app/views/groups/milestones/new.html.haml
index 8d3aa4d1a74d2..7c7573862d02c 100644
--- a/app/views/groups/milestones/new.html.haml
+++ b/app/views/groups/milestones/new.html.haml
@@ -26,7 +26,7 @@
       .form-group.milestone-description
         = f.label :description, "Description", class: "control-label"
         .col-sm-10
-          = render layout: 'projects/md_preview', locals: { preview_class: "md-preview" } do
+          = render layout: 'projects/md_preview', locals: { url: '' } do
             = render 'projects/zen', f: f, attr: :description, classes: 'note-textarea', placeholder: 'Write milestone description...'
             .clearfix
             .error-alert
diff --git a/app/views/layouts/project.html.haml b/app/views/layouts/project.html.haml
index e9e06e5c8e388..3f5b0c54e5001 100644
--- a/app/views/layouts/project.html.haml
+++ b/app/views/layouts/project.html.haml
@@ -5,14 +5,9 @@
 
 - content_for :project_javascripts do
   - project = @target_project || @project
-  - if @project_wiki && @page
-    - preview_markdown_path = namespace_project_wiki_preview_markdown_path(project.namespace, project, @page.slug)
-  - else
-    - preview_markdown_path = preview_markdown_namespace_project_path(project.namespace, project)
   - if current_user
     :javascript
       window.uploads_path = "#{namespace_project_uploads_path project.namespace,project}";
-      window.preview_markdown_path = "#{preview_markdown_path}";
 
 - content_for :header_content do
   .js-dropdown-menu-projects
diff --git a/app/views/projects/_md_preview.html.haml b/app/views/projects/_md_preview.html.haml
index 23e27c1105c7d..d0698285f8491 100644
--- a/app/views/projects/_md_preview.html.haml
+++ b/app/views/projects/_md_preview.html.haml
@@ -1,3 +1,5 @@
+- referenced_users = local_assigns.fetch(:referenced_users, nil)
+
 .md-area
   .md-header
     %ul.nav-links.clearfix
@@ -28,9 +30,10 @@
 
   .md-write-holder
     = yield
-  .md.md-preview-holder.js-md-preview.hide{ class: (preview_class if defined?(preview_class)) }
+  .md.md-preview-holder.js-md-preview.hide.md-preview{ data: { url: url } }
+  .referenced-commands.hide
 
-  - if defined?(referenced_users) && referenced_users
+  - if referenced_users
     .referenced-users.hide
       %span
         = icon("exclamation-triangle")
diff --git a/app/views/projects/milestones/_form.html.haml b/app/views/projects/milestones/_form.html.haml
index 0f4a850875155..2e978fda62422 100644
--- a/app/views/projects/milestones/_form.html.haml
+++ b/app/views/projects/milestones/_form.html.haml
@@ -9,7 +9,7 @@
       .form-group.milestone-description
         = f.label :description, "Description", class: "control-label"
         .col-sm-10
-          = render layout: 'projects/md_preview', locals: { preview_class: "md-preview" } do
+          = render layout: 'projects/md_preview', locals: { url: preview_markdown_path(@project) } do
             = render 'projects/zen', f: f, attr: :description, classes: 'note-textarea', placeholder: 'Write milestone description...'
             = render 'projects/notes/hints'
           .clearfix
diff --git a/app/views/projects/notes/_edit_form.html.haml b/app/views/projects/notes/_edit_form.html.haml
index a1efc0b051a88..00230b0bdf822 100644
--- a/app/views/projects/notes/_edit_form.html.haml
+++ b/app/views/projects/notes/_edit_form.html.haml
@@ -2,7 +2,7 @@
   = form_tag '#', method: :put, class: 'edit-note common-note-form js-quick-submit' do
     = hidden_field_tag :target_id, '', class: 'js-form-target-id'
     = hidden_field_tag :target_type, '', class: 'js-form-target-type'
-    = render layout: 'projects/md_preview', locals: { preview_class: 'md-preview', referenced_users: true } do
+    = render layout: 'projects/md_preview', locals: { url: preview_markdown_path(project), referenced_users: true } do
       = render 'projects/zen', attr: 'note[note]', classes: 'note-textarea js-note-text js-task-list-field', placeholder: "Write a comment or drag your files here..."
       = render 'projects/notes/hints'
 
diff --git a/app/views/projects/notes/_form.html.haml b/app/views/projects/notes/_form.html.haml
index 0d835a9e949dd..46f785fefcafe 100644
--- a/app/views/projects/notes/_form.html.haml
+++ b/app/views/projects/notes/_form.html.haml
@@ -1,4 +1,8 @@
 - supports_slash_commands = note_supports_slash_commands?(@note)
+- if supports_slash_commands
+  - preview_url = preview_markdown_path(@project, slash_commands_target_type: @note.noteable_type, slash_commands_target_id: @note.noteable_id)
+- else
+  - preview_url = preview_markdown_path(@project)
 
 = form_for [@project.namespace.becomes(Namespace), @project, @note], remote: true, html: { :'data-type' => 'json', multipart: true, id: nil, class: "new-note js-new-note-form js-quick-submit common-note-form", "data-noteable-iid" => @note.noteable.try(:iid), }, authenticity_token: true do |f|
   = hidden_field_tag :view, diff_view
@@ -18,7 +22,7 @@
   -# DiffNote
   = f.hidden_field :position
 
-  = render layout: 'projects/md_preview', locals: { preview_class: "md-preview", referenced_users: true } do
+  = render layout: 'projects/md_preview', locals: { url: preview_url, referenced_users: true } do
     = render 'projects/zen', f: f,
       attr: :note,
       classes: 'note-textarea js-note-text',
diff --git a/app/views/projects/notes/_notes_with_form.html.haml b/app/views/projects/notes/_notes_with_form.html.haml
index 555228623cc6b..2a66addb08a1d 100644
--- a/app/views/projects/notes/_notes_with_form.html.haml
+++ b/app/views/projects/notes/_notes_with_form.html.haml
@@ -1,7 +1,7 @@
 %ul#notes-list.notes.main-notes-list.timeline
   = render "shared/notes/notes"
 
-= render 'projects/notes/edit_form'
+= render 'projects/notes/edit_form', project: @project
 
 %ul.notes.notes-form.timeline
   %li.timeline-entry
diff --git a/app/views/projects/releases/edit.html.haml b/app/views/projects/releases/edit.html.haml
index 79d8d721aa9b7..faa24a3c88e21 100644
--- a/app/views/projects/releases/edit.html.haml
+++ b/app/views/projects/releases/edit.html.haml
@@ -11,7 +11,7 @@
 
 
   = form_for(@release, method: :put, url: namespace_project_tag_release_path(@project.namespace, @project, @tag.name), html: { class: 'form-horizontal common-note-form release-form js-quick-submit' }) do |f|
-    = render layout: 'projects/md_preview', locals: { preview_class: "md-preview", referenced_users: true } do
+    = render layout: 'projects/md_preview', locals: { url: preview_markdown_path(@project), referenced_users: true } do
       = render 'projects/zen', f: f, attr: :description, classes: 'note-textarea', placeholder: "Write your release notes or drag files here..."
       = render 'projects/notes/hints'
     .error-alert
diff --git a/app/views/projects/tags/new.html.haml b/app/views/projects/tags/new.html.haml
index 160d4c7a22372..a6894b9adc001 100644
--- a/app/views/projects/tags/new.html.haml
+++ b/app/views/projects/tags/new.html.haml
@@ -28,7 +28,7 @@
   .form-group
     = label_tag :release_description, 'Release notes', class: 'control-label'
     .col-sm-10
-      = render layout: 'projects/md_preview', locals: { preview_class: "md-preview", referenced_users: true } do
+      = render layout: 'projects/md_preview', locals: { url: preview_markdown_path(@project), referenced_users: true } do
         = render 'projects/zen', attr: :release_description, classes: 'note-textarea', placeholder: "Write your release notes or drag files here..."
         = render 'projects/notes/hints'
       .help-block Optionally, add release notes to the tag. They will be stored in the GitLab database and displayed on the tags page.
diff --git a/app/views/projects/wikis/_form.html.haml b/app/views/projects/wikis/_form.html.haml
index 0d2cd4a7476f9..00869aff27bf4 100644
--- a/app/views/projects/wikis/_form.html.haml
+++ b/app/views/projects/wikis/_form.html.haml
@@ -12,7 +12,7 @@
   .form-group
     = f.label :content, class: 'control-label'
     .col-sm-10
-      = render layout: 'projects/md_preview', locals: { preview_class: "md-preview" } do
+      = render layout: 'projects/md_preview', locals: { url: namespace_project_wiki_preview_markdown_path(@project.namespace, @project, @page.slug) } do
         = render 'projects/zen', f: f, attr: :content, classes: 'note-textarea', placeholder: 'Write your content or drag files here...'
         = render 'projects/notes/hints'
 
diff --git a/app/views/shared/issuable/_form.html.haml b/app/views/shared/issuable/_form.html.haml
index 17107f55a2d61..7748351b333f6 100644
--- a/app/views/shared/issuable/_form.html.haml
+++ b/app/views/shared/issuable/_form.html.haml
@@ -17,7 +17,7 @@
   = render 'shared/issuable/form/template_selector', issuable: issuable
   = render 'shared/issuable/form/title', issuable: issuable, form: form, has_wip_commits: commits && commits.detect(&:work_in_progress?)
 
-= render 'shared/issuable/form/description', issuable: issuable, form: form
+= render 'shared/issuable/form/description', issuable: issuable, form: form, project: project
 
 - if issuable.respond_to?(:confidential)
   .form-group
diff --git a/app/views/shared/issuable/form/_description.html.haml b/app/views/shared/issuable/form/_description.html.haml
index dbace9ce401de..cbc7125c0d56a 100644
--- a/app/views/shared/issuable/form/_description.html.haml
+++ b/app/views/shared/issuable/form/_description.html.haml
@@ -1,15 +1,22 @@
+- project = local_assigns.fetch(:project)
 - issuable = local_assigns.fetch(:issuable)
 - form = local_assigns.fetch(:form)
+- supports_slash_commands = issuable.new_record?
+
+- if supports_slash_commands
+  - preview_url = preview_markdown_path(project, slash_commands_target_type: issuable.class.name)
+- else
+  - preview_url = preview_markdown_path(project)
 
 .form-group.detail-page-description
   = form.label :description, 'Description', class: 'control-label'
   .col-sm-10
 
-    = render layout: 'projects/md_preview', locals: { preview_class: "md-preview", referenced_users: true } do
+    = render layout: 'projects/md_preview', locals: { url: preview_url, referenced_users: true } do
       = render 'projects/zen', f: form, attr: :description,
                                classes: 'note-textarea',
                                placeholder: "Write a comment or drag your files here...",
-                               supports_slash_commands: !issuable.persisted?
-      = render 'projects/notes/hints', supports_slash_commands: !issuable.persisted?
+                               supports_slash_commands: supports_slash_commands
+      = render 'projects/notes/hints', supports_slash_commands: supports_slash_commands
       .clearfix
       .error-alert
diff --git a/changelogs/unreleased/preview-separate-slash-commands.yml b/changelogs/unreleased/preview-separate-slash-commands.yml
new file mode 100644
index 0000000000000..6240ccc957c80
--- /dev/null
+++ b/changelogs/unreleased/preview-separate-slash-commands.yml
@@ -0,0 +1,4 @@
+---
+title: Display slash commands outcome when previewing Markdown
+merge_request: 10054
+author: Rares Sfirlogea
diff --git a/lib/gitlab/slash_commands/command_definition.rb b/lib/gitlab/slash_commands/command_definition.rb
index 60d35be259912..12a385f90fdf5 100644
--- a/lib/gitlab/slash_commands/command_definition.rb
+++ b/lib/gitlab/slash_commands/command_definition.rb
@@ -1,16 +1,19 @@
 module Gitlab
   module SlashCommands
     class CommandDefinition
-      attr_accessor :name, :aliases, :description, :params, :condition_block, :action_block
+      attr_accessor :name, :aliases, :description, :explanation, :params,
+        :condition_block, :parse_params_block, :action_block
 
       def initialize(name, attributes = {})
         @name = name
 
-        @aliases         = attributes[:aliases] || []
-        @description     = attributes[:description] || ''
-        @params          = attributes[:params] || []
+        @aliases = attributes[:aliases] || []
+        @description = attributes[:description] || ''
+        @explanation = attributes[:explanation] || ''
+        @params = attributes[:params] || []
         @condition_block = attributes[:condition_block]
-        @action_block    = attributes[:action_block]
+        @parse_params_block = attributes[:parse_params_block]
+        @action_block = attributes[:action_block]
       end
 
       def all_names
@@ -28,14 +31,20 @@ def available?(opts)
         context.instance_exec(&condition_block)
       end
 
+      def explain(context, opts, arg)
+        return unless available?(opts)
+
+        if explanation.respond_to?(:call)
+          execute_block(explanation, context, arg)
+        else
+          explanation
+        end
+      end
+
       def execute(context, opts, arg)
         return if noop? || !available?(opts)
 
-        if arg.present?
-          context.instance_exec(arg, &action_block)
-        elsif action_block.arity == 0
-          context.instance_exec(&action_block)
-        end
+        execute_block(action_block, context, arg)
       end
 
       def to_h(opts)
@@ -52,6 +61,23 @@ def to_h(opts)
           params: params
         }
       end
+
+      private
+
+      def execute_block(block, context, arg)
+        if arg.present?
+          parsed = parse_params(arg, context)
+          context.instance_exec(parsed, &block)
+        elsif block.arity == 0
+          context.instance_exec(&block)
+        end
+      end
+
+      def parse_params(arg, context)
+        return arg unless parse_params_block
+
+        context.instance_exec(arg, &parse_params_block)
+      end
     end
   end
 end
diff --git a/lib/gitlab/slash_commands/dsl.rb b/lib/gitlab/slash_commands/dsl.rb
index 50b0937d267cd..614bafbe1b213 100644
--- a/lib/gitlab/slash_commands/dsl.rb
+++ b/lib/gitlab/slash_commands/dsl.rb
@@ -44,6 +44,22 @@ def params(*params)
           @params = params
         end
 
+        # Allows to give an explanation of what the command will do when
+        # executed. This explanation is shown when rendering the Markdown
+        # preview.
+        #
+        # Example:
+        #
+        #   explanation do |arguments|
+        #     "Adds label(s) #{arguments.join(' ')}"
+        #   end
+        #   command :command_key do |arguments|
+        #     # Awesome code block
+        #   end
+        def explanation(text = '', &block)
+          @explanation = block_given? ? block : text
+        end
+
         # Allows to define conditions that must be met in order for the command
         # to be returned by `.command_names` & `.command_definitions`.
         # It accepts a block that will be evaluated with the context given to
@@ -61,6 +77,24 @@ def condition(&block)
           @condition_block = block
         end
 
+        # Allows to perform initial parsing of parameters. The result is passed
+        # both to `command` and `explanation` blocks, instead of the raw
+        # parameters.
+        # It accepts a block that will be evaluated with the context given to
+        # `CommandDefintion#to_h`.
+        #
+        # Example:
+        #
+        #   parse_params do |raw|
+        #     raw.strip
+        #   end
+        #   command :command_key do |parsed|
+        #     # Awesome code block
+        #   end
+        def parse_params(&block)
+          @parse_params_block = block
+        end
+
         # Registers a new command which is recognizeable from body of email or
         # comment.
         # It accepts aliases and takes a block.
@@ -75,11 +109,13 @@ def command(*command_names, &block)
 
           definition = CommandDefinition.new(
             name,
-            aliases:          aliases,
-            description:      @description,
-            params:           @params,
-            condition_block:  @condition_block,
-            action_block:     block
+            aliases: aliases,
+            description: @description,
+            explanation: @explanation,
+            params: @params,
+            condition_block: @condition_block,
+            parse_params_block: @parse_params_block,
+            action_block: block
           )
 
           self.command_definitions << definition
@@ -89,8 +125,14 @@ def command(*command_names, &block)
           end
 
           @description = nil
+          @explanation = nil
           @params = nil
           @condition_block = nil
+          @parse_params_block = nil
+        end
+
+        def definition_by_name(name)
+          command_definitions_by_name[name.to_sym]
         end
       end
     end
diff --git a/spec/lib/gitlab/slash_commands/command_definition_spec.rb b/spec/lib/gitlab/slash_commands/command_definition_spec.rb
index c9c2f314e576d..5b9173d3d3f51 100644
--- a/spec/lib/gitlab/slash_commands/command_definition_spec.rb
+++ b/spec/lib/gitlab/slash_commands/command_definition_spec.rb
@@ -167,6 +167,58 @@
             end
           end
         end
+
+        context 'when the command defines parse_params block' do
+          before do
+            subject.parse_params_block = ->(raw) { raw.strip }
+            subject.action_block = ->(parsed) { self.received_arg = parsed }
+          end
+
+          it 'executes the command passing the parsed param' do
+            subject.execute(context, {}, 'something   ')
+
+            expect(context.received_arg).to eq('something')
+          end
+        end
+      end
+    end
+  end
+
+  describe '#explain' do
+    context 'when the command is not available' do
+      before do
+        subject.condition_block = proc { false }
+        subject.explanation = 'Explanation'
+      end
+
+      it 'returns nil' do
+        result = subject.explain({}, {}, nil)
+
+        expect(result).to be_nil
+      end
+    end
+
+    context 'when the explanation is a static string' do
+      before do
+        subject.explanation = 'Explanation'
+      end
+
+      it 'returns this static string' do
+        result = subject.explain({}, {}, nil)
+
+        expect(result).to eq 'Explanation'
+      end
+    end
+
+    context 'when the explanation is dynamic' do
+      before do
+        subject.explanation = proc { |arg| "Dynamic #{arg}" }
+      end
+
+      it 'invokes the proc' do
+        result = subject.explain({}, {}, 'explanation')
+
+        expect(result).to eq 'Dynamic explanation'
       end
     end
   end
diff --git a/spec/lib/gitlab/slash_commands/dsl_spec.rb b/spec/lib/gitlab/slash_commands/dsl_spec.rb
index 2763d95071627..33b49a5ddf98b 100644
--- a/spec/lib/gitlab/slash_commands/dsl_spec.rb
+++ b/spec/lib/gitlab/slash_commands/dsl_spec.rb
@@ -11,67 +11,99 @@
       end
 
       params 'The first argument'
-      command :one_arg, :once, :first do |arg1|
-        arg1
+      explanation 'Static explanation'
+      command :explanation_with_aliases, :once, :first do |arg|
+        arg
       end
 
       desc do
         "A dynamic description for #{noteable.upcase}"
       end
       params 'The first argument', 'The second argument'
-      command :two_args do |arg1, arg2|
-        [arg1, arg2]
+      command :dynamic_description do |args|
+        args.split
       end
 
       command :cc
 
+      explanation do |arg|
+        "Action does something with #{arg}"
+      end
       condition do
         project == 'foo'
       end
       command :cond_action do |arg|
         arg
       end
+
+      parse_params do |raw_arg|
+        raw_arg.strip
+      end
+      command :with_params_parsing do |parsed|
+        parsed
+      end
     end
   end
 
   describe '.command_definitions' do
     it 'returns an array with commands definitions' do
-      no_args_def, one_arg_def, two_args_def, cc_def, cond_action_def = DummyClass.command_definitions
+      no_args_def, explanation_with_aliases_def, dynamic_description_def,
+      cc_def, cond_action_def, with_params_parsing_def =
+        DummyClass.command_definitions
 
       expect(no_args_def.name).to eq(:no_args)
       expect(no_args_def.aliases).to eq([:none])
       expect(no_args_def.description).to eq('A command with no args')
+      expect(no_args_def.explanation).to eq('')
       expect(no_args_def.params).to eq([])
       expect(no_args_def.condition_block).to be_nil
       expect(no_args_def.action_block).to be_a_kind_of(Proc)
+      expect(no_args_def.parse_params_block).to be_nil
 
-      expect(one_arg_def.name).to eq(:one_arg)
-      expect(one_arg_def.aliases).to eq([:once, :first])
-      expect(one_arg_def.description).to eq('')
-      expect(one_arg_def.params).to eq(['The first argument'])
-      expect(one_arg_def.condition_block).to be_nil
-      expect(one_arg_def.action_block).to be_a_kind_of(Proc)
+      expect(explanation_with_aliases_def.name).to eq(:explanation_with_aliases)
+      expect(explanation_with_aliases_def.aliases).to eq([:once, :first])
+      expect(explanation_with_aliases_def.description).to eq('')
+      expect(explanation_with_aliases_def.explanation).to eq('Static explanation')
+      expect(explanation_with_aliases_def.params).to eq(['The first argument'])
+      expect(explanation_with_aliases_def.condition_block).to be_nil
+      expect(explanation_with_aliases_def.action_block).to be_a_kind_of(Proc)
+      expect(explanation_with_aliases_def.parse_params_block).to be_nil
 
-      expect(two_args_def.name).to eq(:two_args)
-      expect(two_args_def.aliases).to eq([])
-      expect(two_args_def.to_h(noteable: "issue")[:description]).to eq('A dynamic description for ISSUE')
-      expect(two_args_def.params).to eq(['The first argument', 'The second argument'])
-      expect(two_args_def.condition_block).to be_nil
-      expect(two_args_def.action_block).to be_a_kind_of(Proc)
+      expect(dynamic_description_def.name).to eq(:dynamic_description)
+      expect(dynamic_description_def.aliases).to eq([])
+      expect(dynamic_description_def.to_h(noteable: 'issue')[:description]).to eq('A dynamic description for ISSUE')
+      expect(dynamic_description_def.explanation).to eq('')
+      expect(dynamic_description_def.params).to eq(['The first argument', 'The second argument'])
+      expect(dynamic_description_def.condition_block).to be_nil
+      expect(dynamic_description_def.action_block).to be_a_kind_of(Proc)
+      expect(dynamic_description_def.parse_params_block).to be_nil
 
       expect(cc_def.name).to eq(:cc)
       expect(cc_def.aliases).to eq([])
       expect(cc_def.description).to eq('')
+      expect(cc_def.explanation).to eq('')
       expect(cc_def.params).to eq([])
       expect(cc_def.condition_block).to be_nil
       expect(cc_def.action_block).to be_nil
+      expect(cc_def.parse_params_block).to be_nil
 
       expect(cond_action_def.name).to eq(:cond_action)
       expect(cond_action_def.aliases).to eq([])
       expect(cond_action_def.description).to eq('')
+      expect(cond_action_def.explanation).to be_a_kind_of(Proc)
       expect(cond_action_def.params).to eq([])
       expect(cond_action_def.condition_block).to be_a_kind_of(Proc)
       expect(cond_action_def.action_block).to be_a_kind_of(Proc)
+      expect(cond_action_def.parse_params_block).to be_nil
+
+      expect(with_params_parsing_def.name).to eq(:with_params_parsing)
+      expect(with_params_parsing_def.aliases).to eq([])
+      expect(with_params_parsing_def.description).to eq('')
+      expect(with_params_parsing_def.explanation).to eq('')
+      expect(with_params_parsing_def.params).to eq([])
+      expect(with_params_parsing_def.condition_block).to be_nil
+      expect(with_params_parsing_def.action_block).to be_a_kind_of(Proc)
+      expect(with_params_parsing_def.parse_params_block).to be_a_kind_of(Proc)
     end
   end
 end
diff --git a/spec/services/preview_markdown_service_spec.rb b/spec/services/preview_markdown_service_spec.rb
new file mode 100644
index 0000000000000..b2fb5c91313c0
--- /dev/null
+++ b/spec/services/preview_markdown_service_spec.rb
@@ -0,0 +1,67 @@
+require 'spec_helper'
+
+describe PreviewMarkdownService do
+  let(:user) { create(:user) }
+  let(:project) { create(:empty_project) }
+
+  before do
+    project.add_developer(user)
+  end
+
+  describe 'user references' do
+    let(:params) { { text: "Take a look #{user.to_reference}" } }
+    let(:service) { described_class.new(project, user, params) }
+
+    it 'returns users referenced in text' do
+      result = service.execute
+
+      expect(result[:users]).to eq [user.username]
+    end
+  end
+
+  context 'new note with slash commands' do
+    let(:issue) { create(:issue, project: project) }
+    let(:params) do
+      {
+        text: "Please do it\n/assign #{user.to_reference}",
+        slash_commands_target_type: 'Issue',
+        slash_commands_target_id: issue.id
+      }
+    end
+    let(:service) { described_class.new(project, user, params) }
+
+    it 'removes slash commands from text' do
+      result = service.execute
+
+      expect(result[:text]).to eq 'Please do it'
+    end
+
+    it 'explains slash commands effect' do
+      result = service.execute
+
+      expect(result[:commands]).to eq "Assigns #{user.to_reference}."
+    end
+  end
+
+  context 'merge request description' do
+    let(:params) do
+      {
+        text: "My work\n/estimate 2y",
+        slash_commands_target_type: 'MergeRequest'
+      }
+    end
+    let(:service) { described_class.new(project, user, params) }
+
+    it 'removes slash commands from text' do
+      result = service.execute
+
+      expect(result[:text]).to eq 'My work'
+    end
+
+    it 'explains slash commands effect' do
+      result = service.execute
+
+      expect(result[:commands]).to eq 'Sets time estimate to 2y.'
+    end    
+  end
+end
diff --git a/spec/services/slash_commands/interpret_service_spec.rb b/spec/services/slash_commands/interpret_service_spec.rb
index 29e65fe7ce6da..46cfac4a128fe 100644
--- a/spec/services/slash_commands/interpret_service_spec.rb
+++ b/spec/services/slash_commands/interpret_service_spec.rb
@@ -798,4 +798,211 @@
       end
     end
   end
+
+  describe '#explain' do
+    let(:service) { described_class.new(project, developer) }
+    let(:merge_request) { create(:merge_request, source_project: project) }
+
+    describe 'close command' do
+      let(:content) { '/close' }
+
+      it 'includes issuable name' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(['Closes this issue.'])
+      end
+    end
+
+    describe 'reopen command' do
+      let(:content) { '/reopen' }
+      let(:merge_request) { create(:merge_request, :closed, source_project: project) }
+
+      it 'includes issuable name' do
+        _, explanations = service.explain(content, merge_request)
+
+        expect(explanations).to eq(['Reopens this merge request.'])
+      end
+    end
+
+    describe 'title command' do
+      let(:content) { '/title This is new title' }
+
+      it 'includes new title' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(['Changes the title to "This is new title".'])
+      end
+    end
+
+    describe 'assign command' do
+      let(:content) { "/assign @#{developer.username} do it!" }
+
+      it 'includes only the user reference' do
+        _, explanations = service.explain(content, merge_request)
+
+        expect(explanations).to eq(["Assigns @#{developer.username}."])
+      end
+    end
+
+    describe 'unassign command' do
+      let(:content) { '/unassign' }
+      let(:issue) { create(:issue, project: project, assignee: developer) }
+
+      it 'includes current assignee reference' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(["Removes assignee @#{developer.username}."])
+      end
+    end
+
+    describe 'milestone command' do
+      let(:content) { '/milestone %wrong-milestone' }
+      let!(:milestone) { create(:milestone, project: project, title: '9.10') }
+
+      it 'is empty when milestone reference is wrong' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq([])
+      end
+    end
+
+    describe 'remove milestone command' do
+      let(:content) { '/remove_milestone' }
+      let(:merge_request) { create(:merge_request, source_project: project, milestone: milestone) }
+
+      it 'includes current milestone name' do
+        _, explanations = service.explain(content, merge_request)
+
+        expect(explanations).to eq(['Removes %"9.10" milestone.'])
+      end
+    end
+
+    describe 'label command' do
+      let(:content) { '/label ~missing' }
+      let!(:label) { create(:label, project: project) }
+
+      it 'is empty when there are no correct labels' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq([])
+      end
+    end
+
+    describe 'unlabel command' do
+      let(:content) { '/unlabel' }
+
+      it 'says all labels if no parameter provided' do
+        merge_request.update!(label_ids: [bug.id])
+        _, explanations = service.explain(content, merge_request)
+
+        expect(explanations).to eq(['Removes all labels.'])
+      end
+    end
+
+    describe 'relabel command' do
+      let(:content) { '/relabel Bug' }
+      let!(:bug) { create(:label, project: project, title: 'Bug') }
+      let(:feature) { create(:label, project: project, title: 'Feature') }
+
+      it 'includes label name' do
+        issue.update!(label_ids: [feature.id])
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(["Replaces all labels with ~#{bug.id} label."])
+      end
+    end
+
+    describe 'subscribe command' do
+      let(:content) { '/subscribe' }
+
+      it 'includes issuable name' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(['Subscribes to this issue.'])
+      end
+    end
+
+    describe 'unsubscribe command' do
+      let(:content) { '/unsubscribe' }
+
+      it 'includes issuable name' do
+        merge_request.subscribe(developer, project)
+        _, explanations = service.explain(content, merge_request)
+
+        expect(explanations).to eq(['Unsubscribes from this merge request.'])
+      end
+    end
+
+    describe 'due command' do
+      let(:content) { '/due April 1st 2016' }
+
+      it 'includes the date' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(['Sets the due date to Apr 1, 2016.'])
+      end
+    end
+
+    describe 'wip command' do
+      let(:content) { '/wip' }
+
+      it 'includes the new status' do
+        _, explanations = service.explain(content, merge_request)
+
+        expect(explanations).to eq(['Marks this merge request as Work In Progress.'])
+      end
+    end
+
+    describe 'award command' do
+      let(:content) { '/award :confetti_ball: ' }
+
+      it 'includes the emoji' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(['Toggles :confetti_ball: emoji award.'])
+      end
+    end
+
+    describe 'estimate command' do
+      let(:content) { '/estimate 79d' }
+
+      it 'includes the formatted duration' do
+        _, explanations = service.explain(content, merge_request)
+
+        expect(explanations).to eq(['Sets time estimate to 3mo 3w 4d.'])
+      end
+    end
+
+    describe 'spend command' do
+      let(:content) { '/spend -120m' }
+
+      it 'includes the formatted duration and proper verb' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(['Substracts 2h spent time.'])
+      end
+    end
+
+    describe 'target branch command' do
+      let(:content) { '/target_branch my-feature ' }
+
+      it 'includes the branch name' do
+        _, explanations = service.explain(content, merge_request)
+
+        expect(explanations).to eq(['Sets target branch to my-feature.'])
+      end
+    end
+
+    describe 'board move command' do
+      let(:content) { '/board_move ~bug' }
+      let!(:bug) { create(:label, project: project, title: 'bug') }
+      let!(:board) { create(:board, project: project) }
+
+      it 'includes the label name' do
+        _, explanations = service.explain(content, issue)
+
+        expect(explanations).to eq(["Moves issue to ~#{bug.id} column in the board."])
+      end
+    end
+  end
 end
diff --git a/spec/support/features/issuable_slash_commands_shared_examples.rb b/spec/support/features/issuable_slash_commands_shared_examples.rb
index 5bbe36d9b7fb8..6efcdd7a1ded9 100644
--- a/spec/support/features/issuable_slash_commands_shared_examples.rb
+++ b/spec/support/features/issuable_slash_commands_shared_examples.rb
@@ -257,4 +257,19 @@
       end
     end
   end
+
+  describe "preview of note on #{issuable_type}" do
+    it 'removes slash commands from note and explains them' do
+      visit public_send("namespace_project_#{issuable_type}_path", project.namespace, project, issuable)
+
+      page.within('.js-main-target-form') do
+        fill_in 'note[note]', with: "Awesome!\n/assign @bob "
+        click_on 'Preview'
+
+        expect(page).to have_content 'Awesome!'
+        expect(page).not_to have_content '/assign @bob'
+        expect(page).to have_content 'Assigns @bob.'
+      end
+    end
+  end
 end
-- 
GitLab


From f1ace97f8bdc69edc481b545b779721322cec95c Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=A9my=20Coutable?= <remy@rymai.me>
Date: Wed, 3 May 2017 17:25:31 +0200
Subject: [PATCH 220/363] Backport avatar-related spec changes from
 gitlab-org/gitlab-ee@4b464eaaee
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Signed-off-by: Rémy Coutable <remy@rymai.me>
---
 spec/factories/groups.rb    |  4 ++++
 spec/factories/users.rb     |  4 ++++
 spec/models/group_spec.rb   | 16 ++++++++++++++++
 spec/models/project_spec.rb | 11 +++--------
 spec/models/user_spec.rb    | 11 +++++++++++
 5 files changed, 38 insertions(+), 8 deletions(-)

diff --git a/spec/factories/groups.rb b/spec/factories/groups.rb
index 86f51ffca997f..52f76b094a399 100644
--- a/spec/factories/groups.rb
+++ b/spec/factories/groups.rb
@@ -17,6 +17,10 @@
       visibility_level Gitlab::VisibilityLevel::PRIVATE
     end
 
+    trait :with_avatar do
+      avatar { File.open(Rails.root.join('spec/fixtures/dk.png')) }
+    end
+
     trait :access_requestable do
       request_access_enabled true
     end
diff --git a/spec/factories/users.rb b/spec/factories/users.rb
index e1ae94a08e461..33fa80772ffcd 100644
--- a/spec/factories/users.rb
+++ b/spec/factories/users.rb
@@ -29,6 +29,10 @@
       after(:build) { |user, _| user.block! }
     end
 
+    trait :with_avatar do
+      avatar { File.open(Rails.root.join('spec/fixtures/dk.png')) }
+    end
+
     trait :two_factor_via_otp do
       before(:create) do |user|
         user.otp_required_for_login = true
diff --git a/spec/models/group_spec.rb b/spec/models/group_spec.rb
index a11805926cc53..3d60e52f23f4d 100644
--- a/spec/models/group_spec.rb
+++ b/spec/models/group_spec.rb
@@ -175,6 +175,22 @@
     end
   end
 
+  describe '#avatar_url' do
+    let!(:group) { create(:group, :access_requestable, :with_avatar) }
+    let(:user) { create(:user) }
+    subject { group.avatar_url }
+
+    context 'when avatar file is uploaded' do
+      before do
+        group.add_master(user)
+      end
+
+      let(:avatar_path) { "/uploads/group/avatar/#{group.id}/dk.png" }
+
+      it { should eq "http://#{Gitlab.config.gitlab.host}#{avatar_path}" }
+    end
+  end
+
   describe '.search' do
     it 'returns groups with a matching name' do
       expect(described_class.search(group.name)).to eq([group])
diff --git a/spec/models/project_spec.rb b/spec/models/project_spec.rb
index 36ce3070a6eb0..c3eeff6c43a04 100644
--- a/spec/models/project_spec.rb
+++ b/spec/models/project_spec.rb
@@ -811,12 +811,9 @@
 
     context 'when avatar file is uploaded' do
       let(:project) { create(:empty_project, :with_avatar) }
+      let(:avatar_path) { "/uploads/project/avatar/#{project.id}/dk.png" }
 
-      it 'creates a correct avatar path' do
-        avatar_path = "/uploads/project/avatar/#{project.id}/dk.png"
-
-        expect(project.avatar_url).to eq("http://#{Gitlab.config.gitlab.host}#{avatar_path}")
-      end
+      it { should eq "http://#{Gitlab.config.gitlab.host}#{avatar_path}" }
     end
 
     context 'When avatar file in git' do
@@ -824,9 +821,7 @@
         allow(project).to receive(:avatar_in_git) { true }
       end
 
-      let(:avatar_path) do
-        "/#{project.full_path}/avatar"
-      end
+      let(:avatar_path) { "/#{project.full_path}/avatar" }
 
       it { should eq "http://#{Gitlab.config.gitlab.host}#{avatar_path}" }
     end
diff --git a/spec/models/user_spec.rb b/spec/models/user_spec.rb
index 1c2df4c9d9741..5322b7178e3ef 100644
--- a/spec/models/user_spec.rb
+++ b/spec/models/user_spec.rb
@@ -874,6 +874,17 @@
     end
   end
 
+  describe '#avatar_url' do
+    let(:user) { create(:user, :with_avatar) }
+    subject { user.avatar_url }
+
+    context 'when avatar file is uploaded' do
+      let(:avatar_path) { "/uploads/user/avatar/#{user.id}/dk.png" }
+
+      it { should eq "http://#{Gitlab.config.gitlab.host}#{avatar_path}" }
+    end
+  end
+
   describe '#requires_ldap_check?' do
     let(:user) { User.new }
 
-- 
GitLab


From dde9b6698bb67dc34650a588ad3e644843067800 Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Thu, 4 May 2017 10:08:55 -0500
Subject: [PATCH 221/363] Fix webpack config conflict

---
 config/webpack.config.js | 5 -----
 1 file changed, 5 deletions(-)

diff --git a/config/webpack.config.js b/config/webpack.config.js
index 8cacafd611f64..97597397b75e7 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -33,11 +33,6 @@ var config = {
     graphs:               './graphs/graphs_bundle.js',
     group:                './group.js',
     groups_list:          './groups_list.js',
-<<<<<<< HEAD
-    issuable:             './issuable/issuable_bundle.js',
-=======
-    issues:               './issues/issues_bundle.js',
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
     issue_show:           './issue_show/index.js',
     main:                 './main.js',
     merge_conflicts:      './merge_conflicts/merge_conflicts_bundle.js',
-- 
GitLab


From e1d22a054e5ebac2846da1c2f5fbdf2bc22a7117 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Thu, 4 May 2017 16:16:00 +0100
Subject: [PATCH 222/363] Updated JS object for Vue methods

---
 app/assets/javascripts/vue_shared/translate.js | 6 +++---
 1 file changed, 3 insertions(+), 3 deletions(-)

diff --git a/app/assets/javascripts/vue_shared/translate.js b/app/assets/javascripts/vue_shared/translate.js
index ba9656d4e71e8..f83c4b00761d2 100644
--- a/app/assets/javascripts/vue_shared/translate.js
+++ b/app/assets/javascripts/vue_shared/translate.js
@@ -13,7 +13,7 @@ export default (Vue) => {
         @param text The text to be translated
         @returns {String} The translated text
       **/
-      __(text) { return __(text); },
+      __,
       /**
         Translate the text with a number
         if the number is more than 1 it will use the `pluralText` translation.
@@ -24,7 +24,7 @@ export default (Vue) => {
         @param count Number to decide which translation to use (eg. 2)
         @returns {String} Translated text with the number replaced (eg. '2 days')
       **/
-      n__(text, pluralText, count) { return n__(text, pluralText, count); },
+      n__,
       /**
         Translate context based text
         Either pass in the context translation like `Context|Text to translate`
@@ -36,7 +36,7 @@ export default (Vue) => {
         @param key Is the dynamic variable you want to be translated
         @returns {String} Translated context based text
       **/
-      s__(keyOrContext, key) { return s__(keyOrContext, key); },
+      s__,
     },
   });
 };
-- 
GitLab


From dd146a59c32de94a01dafb82279199310337ced8 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Thu, 4 May 2017 16:33:25 +0100
Subject: [PATCH 223/363] Changed how the default avatar is set

---
 .../javascripts/boards/boards_bundle.js       |  5 +++--
 app/assets/javascripts/boards/models/issue.js |  4 ++--
 app/assets/javascripts/boards/models/list.js  |  5 +++--
 app/assets/javascripts/boards/models/user.js  |  7 ++-----
 .../javascripts/boards/stores/boards_store.js |  4 ++--
 .../boards/utils/default_avatar.js            |  1 -
 spec/javascripts/boards/issue_card_spec.js    | 21 +++++++++++++++++++
 7 files changed, 33 insertions(+), 14 deletions(-)
 delete mode 100644 app/assets/javascripts/boards/utils/default_avatar.js

diff --git a/app/assets/javascripts/boards/boards_bundle.js b/app/assets/javascripts/boards/boards_bundle.js
index b6dee8177d276..8c08b2d4db38f 100644
--- a/app/assets/javascripts/boards/boards_bundle.js
+++ b/app/assets/javascripts/boards/boards_bundle.js
@@ -59,7 +59,8 @@ $(() => {
       issueLinkBase: $boardApp.dataset.issueLinkBase,
       rootPath: $boardApp.dataset.rootPath,
       bulkUpdatePath: $boardApp.dataset.bulkUpdatePath,
-      detailIssue: Store.detail
+      detailIssue: Store.detail,
+      defaultAvatar: $boardApp.dataset.defaultAvatar,
     },
     computed: {
       detailIssueVisible () {
@@ -82,7 +83,7 @@ $(() => {
       gl.boardService.all()
         .then((resp) => {
           resp.json().forEach((board) => {
-            const list = Store.addList(board);
+            const list = Store.addList(board, this.defaultAvatar);
 
             if (list.type === 'closed') {
               list.position = Infinity;
diff --git a/app/assets/javascripts/boards/models/issue.js b/app/assets/javascripts/boards/models/issue.js
index d6175069e370e..db783467f8772 100644
--- a/app/assets/javascripts/boards/models/issue.js
+++ b/app/assets/javascripts/boards/models/issue.js
@@ -6,7 +6,7 @@
 import Vue from 'vue';
 
 class ListIssue {
-  constructor (obj) {
+  constructor (obj, defaultAvatar) {
     this.globalId = obj.id;
     this.id = obj.iid;
     this.title = obj.title;
@@ -19,7 +19,7 @@ class ListIssue {
     this.position = obj.relative_position || Infinity;
 
     if (obj.assignee) {
-      this.assignee = new ListUser(obj.assignee);
+      this.assignee = new ListUser(obj.assignee, defaultAvatar);
     }
 
     if (obj.milestone) {
diff --git a/app/assets/javascripts/boards/models/list.js b/app/assets/javascripts/boards/models/list.js
index f2b79a88a4ad4..bd2f62bcc1aab 100644
--- a/app/assets/javascripts/boards/models/list.js
+++ b/app/assets/javascripts/boards/models/list.js
@@ -6,7 +6,7 @@ import queryData from '../utils/query_data';
 const PER_PAGE = 20;
 
 class List {
-  constructor (obj) {
+  constructor (obj, defaultAvatar) {
     this.id = obj.id;
     this._uid = this.guid();
     this.position = obj.position;
@@ -18,6 +18,7 @@ class List {
     this.loadingMore = false;
     this.issues = [];
     this.issuesSize = 0;
+    this.defaultAvatar = defaultAvatar;
 
     if (obj.label) {
       this.label = new ListLabel(obj.label);
@@ -106,7 +107,7 @@ class List {
 
   createIssues (data) {
     data.forEach((issueObj) => {
-      this.addIssue(new ListIssue(issueObj));
+      this.addIssue(new ListIssue(issueObj, this.defaultAvatar));
     });
   }
 
diff --git a/app/assets/javascripts/boards/models/user.js b/app/assets/javascripts/boards/models/user.js
index 875c21a37d362..2af583c3279f2 100644
--- a/app/assets/javascripts/boards/models/user.js
+++ b/app/assets/javascripts/boards/models/user.js
@@ -1,12 +1,9 @@
-/* eslint-disable no-unused-vars */
-import defaultAvatar from '../utils/default_avatar';
-
 class ListUser {
-  constructor(user) {
+  constructor(user, defaultAvatar) {
     this.id = user.id;
     this.name = user.name;
     this.username = user.username;
-    this.avatar = user.avatar_url || defaultAvatar();
+    this.avatar = user.avatar_url || defaultAvatar;
   }
 }
 
diff --git a/app/assets/javascripts/boards/stores/boards_store.js b/app/assets/javascripts/boards/stores/boards_store.js
index ccb000992155f..ad9997ac33436 100644
--- a/app/assets/javascripts/boards/stores/boards_store.js
+++ b/app/assets/javascripts/boards/stores/boards_store.js
@@ -23,8 +23,8 @@ gl.issueBoards.BoardsStore = {
     this.state.lists = [];
     this.filter.path = gl.utils.getUrlParamsArray().join('&');
   },
-  addList (listObj) {
-    const list = new List(listObj);
+  addList (listObj, defaultAvatar) {
+    const list = new List(listObj, defaultAvatar);
     this.state.lists.push(list);
 
     return list;
diff --git a/app/assets/javascripts/boards/utils/default_avatar.js b/app/assets/javascripts/boards/utils/default_avatar.js
deleted file mode 100644
index 062ffec6dce0d..0000000000000
--- a/app/assets/javascripts/boards/utils/default_avatar.js
+++ /dev/null
@@ -1 +0,0 @@
-export default () => document.getElementById('board-app').dataset.defaultAvatar;
diff --git a/spec/javascripts/boards/issue_card_spec.js b/spec/javascripts/boards/issue_card_spec.js
index 1a5e9e9fd0794..ef567635d48c1 100644
--- a/spec/javascripts/boards/issue_card_spec.js
+++ b/spec/javascripts/boards/issue_card_spec.js
@@ -146,6 +146,27 @@ describe('Issue card component', () => {
         ).not.toBeNull();
       });
     });
+
+    describe('assignee default avatar', () => {
+      beforeEach((done) => {
+        component.issue.assignee = new ListUser({
+          id: 1,
+          name: 'testing 123',
+          username: 'test',
+        }, 'default_avatar');
+
+        Vue.nextTick(done);
+      });
+
+      it('displays defaults avatar if users avatar is null', () => {
+        expect(
+          component.$el.querySelector('.card-assignee img'),
+        ).not.toBeNull();
+        expect(
+          component.$el.querySelector('.card-assignee img').getAttribute('src'),
+        ).toBe('default_avatar');
+      });
+    });
   });
 
   describe('labels', () => {
-- 
GitLab


From cf002738e766f977bdb0e857759f548a5c65c9bd Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Thu, 4 May 2017 18:11:28 +0200
Subject: [PATCH 224/363] refactor a few things based on feedback

---
 app/controllers/admin/services_controller.rb  |  4 ++-
 app/models/service.rb                         | 10 ------
 app/services/projects/propagate_service.rb    |  8 ++---
 .../propagate_project_service_worker.rb       |  2 +-
 .../admin/services_controller_spec.rb         | 32 +++++++++++++++++++
 spec/models/service_spec.rb                   | 27 ----------------
 .../projects/propagate_service_spec.rb        |  8 ++---
 .../propagate_project_service_worker_spec.rb  |  2 +-
 8 files changed, 45 insertions(+), 48 deletions(-)

diff --git a/app/controllers/admin/services_controller.rb b/app/controllers/admin/services_controller.rb
index 2b6f335cb2b1c..e335fbfffedfd 100644
--- a/app/controllers/admin/services_controller.rb
+++ b/app/controllers/admin/services_controller.rb
@@ -15,7 +15,9 @@ def edit
   end
 
   def update
-    if service.update_and_propagate(service_params[:service])
+    if service.update_attributes(service_params[:service])
+      PropagateProjectServiceWorker.perform_async(service.id) if  service.active?
+
       redirect_to admin_application_settings_services_path,
         notice: 'Application settings saved successfully'
     else
diff --git a/app/models/service.rb b/app/models/service.rb
index f85343877035e..c71a7d169eca7 100644
--- a/app/models/service.rb
+++ b/app/models/service.rb
@@ -254,16 +254,6 @@ def self.build_from_template(project_id, template)
     service
   end
 
-  def update_and_propagate(service_params)
-    return false unless update_attributes(service_params)
-
-    if service_params[:active]
-      PropagateProjectServiceWorker.perform_async(id)
-    end
-
-    true
-  end
-
   private
 
   def cache_project_has_external_issue_tracker
diff --git a/app/services/projects/propagate_service.rb b/app/services/projects/propagate_service.rb
index 3c05dcce07c7a..6e24a67d8b0b7 100644
--- a/app/services/projects/propagate_service.rb
+++ b/app/services/projects/propagate_service.rb
@@ -2,15 +2,15 @@ module Projects
   class PropagateService
     BATCH_SIZE = 100
 
-    def self.propagate!(*args)
-      new(*args).propagate!
+    def self.propagate(*args)
+      new(*args).propagate
     end
 
     def initialize(template)
       @template = template
     end
 
-    def propagate!
+    def propagate
       return unless @template&.active
 
       Rails.logger.info("Propagating services for template #{@template.id}")
@@ -28,7 +28,7 @@ def propagate_projects_with_template
 
         batch.each { |project_id| create_from_template(project_id) }
 
-        break if batch.count < BATCH_SIZE
+        break if batch.size < BATCH_SIZE
 
         offset += BATCH_SIZE
       end
diff --git a/app/workers/propagate_project_service_worker.rb b/app/workers/propagate_project_service_worker.rb
index 6cc3fe846356a..5eabe4eaecdbc 100644
--- a/app/workers/propagate_project_service_worker.rb
+++ b/app/workers/propagate_project_service_worker.rb
@@ -10,7 +10,7 @@ class PropagateProjectServiceWorker
   def perform(template_id)
     return unless try_obtain_lease_for(template_id)
 
-    Projects::PropagateService.propagate!(Service.find_by(id: template_id))
+    Projects::PropagateService.propagate(Service.find_by(id: template_id))
   end
 
   private
diff --git a/spec/controllers/admin/services_controller_spec.rb b/spec/controllers/admin/services_controller_spec.rb
index e5cdd52307ee4..808c98edb7f13 100644
--- a/spec/controllers/admin/services_controller_spec.rb
+++ b/spec/controllers/admin/services_controller_spec.rb
@@ -23,4 +23,36 @@
       end
     end
   end
+
+  describe "#update" do
+    let(:project) { create(:empty_project) }
+    let!(:service) do
+      RedmineService.create(
+        project: project,
+        active: false,
+        template: true,
+        properties: {
+          project_url: 'http://abc',
+          issues_url: 'http://abc',
+          new_issue_url: 'http://abc'
+        }
+      )
+    end
+
+    it 'updates the service params successfully and calls the propagation worker' do
+      expect(PropagateProjectServiceWorker).to receive(:perform_async).with(service.id)
+
+      put :update, id: service.id, service: { active: true }
+
+      expect(response).to have_http_status(302)
+    end
+
+    it 'updates the service params successfully' do
+      expect(PropagateProjectServiceWorker).not_to receive(:perform_async)
+
+      put :update, id: service.id, service: { properties: {} }
+
+      expect(response).to have_http_status(302)
+    end
+  end
 end
diff --git a/spec/models/service_spec.rb b/spec/models/service_spec.rb
index 3a7d8b729938b..134882648b9b4 100644
--- a/spec/models/service_spec.rb
+++ b/spec/models/service_spec.rb
@@ -254,31 +254,4 @@
       end
     end
   end
-
-  describe "#update_and_propagate" do
-    let(:project) { create(:empty_project) }
-    let!(:service) do
-      RedmineService.create(
-        project: project,
-        active: false,
-        properties: {
-          project_url: 'http://abc',
-          issues_url: 'http://abc',
-          new_issue_url: 'http://abc'
-        }
-      )
-    end
-
-    it 'updates the service params successfully and calls the propagation worker' do
-      expect(PropagateProjectServiceWorker).to receive(:perform_async).with(service.id)
-
-      expect(service.update_and_propagate(active: true)).to be true
-    end
-
-    it 'updates the service params successfully' do
-      expect(PropagateProjectServiceWorker).not_to receive(:perform_async)
-
-      expect(service.update_and_propagate(properties: {})).to be true
-    end
-  end
 end
diff --git a/spec/services/projects/propagate_service_spec.rb b/spec/services/projects/propagate_service_spec.rb
index 409c14655cd33..d4ec7c0b357bc 100644
--- a/spec/services/projects/propagate_service_spec.rb
+++ b/spec/services/projects/propagate_service_spec.rb
@@ -18,7 +18,7 @@
     let!(:project) { create(:empty_project) }
 
     it 'creates services for projects' do
-      expect { described_class.propagate!(service_template) }.
+      expect { described_class.propagate(service_template) }.
         to change { Service.count }.by(1)
     end
 
@@ -36,7 +36,7 @@
 
       Service.build_from_template(project.id, other_service).save!
 
-      expect { described_class.propagate!(service_template) }.
+      expect { described_class.propagate(service_template) }.
         to change { Service.count }.by(1)
     end
 
@@ -55,12 +55,12 @@
       Service.build_from_template(project.id, service_template).save!
       Service.build_from_template(project.id, other_service).save!
 
-      expect { described_class.propagate!(service_template) }.
+      expect { described_class.propagate(service_template) }.
         not_to change { Service.count }
     end
 
     it 'creates the service containing the template attributes' do
-      described_class.propagate!(service_template)
+      described_class.propagate(service_template)
 
       service = Service.find_by(type: service_template.type, template: false)
 
diff --git a/spec/workers/propagate_project_service_worker_spec.rb b/spec/workers/propagate_project_service_worker_spec.rb
index c16e95bd49b6e..4c7edbfcd3e4a 100644
--- a/spec/workers/propagate_project_service_worker_spec.rb
+++ b/spec/workers/propagate_project_service_worker_spec.rb
@@ -21,7 +21,7 @@
 
   describe '#perform' do
     it 'calls the propagate service with the template' do
-      expect(Projects::PropagateService).to receive(:propagate!).with(service_template)
+      expect(Projects::PropagateService).to receive(:propagate).with(service_template)
 
       subject.perform(service_template.id)
     end
-- 
GitLab


From 4b9eab02b8cc8bd443a802d1d73da26e5b3178d9 Mon Sep 17 00:00:00 2001
From: Bob Van Landuyt <bob@gitlab.com>
Date: Thu, 4 May 2017 18:11:31 +0200
Subject: [PATCH 225/363] Reject EE reserved namespace paths in CE as well

---
 app/validators/dynamic_path_validator.rb                   | 7 +++++++
 .../20170412174900_rename_reserved_dynamic_paths.rb        | 7 +++++++
 2 files changed, 14 insertions(+)

diff --git a/app/validators/dynamic_path_validator.rb b/app/validators/dynamic_path_validator.rb
index 226eb6b313c1f..d992b0c3725ac 100644
--- a/app/validators/dynamic_path_validator.rb
+++ b/app/validators/dynamic_path_validator.rb
@@ -115,13 +115,20 @@ class DynamicPathValidator < ActiveModel::EachValidator
   # this would map to the activity-page of it's parent.
   GROUP_ROUTES = %w[
     activity
+    analytics
+    audit_events
     avatar
     edit
     group_members
+    hooks
     issues
     labels
+    ldap
+    ldap_group_links
     merge_requests
     milestones
+    notification_setting
+    pipeline_quota
     projects
     subgroups
   ].freeze
diff --git a/db/post_migrate/20170412174900_rename_reserved_dynamic_paths.rb b/db/post_migrate/20170412174900_rename_reserved_dynamic_paths.rb
index a23f83205f148..08cf366f0a197 100644
--- a/db/post_migrate/20170412174900_rename_reserved_dynamic_paths.rb
+++ b/db/post_migrate/20170412174900_rename_reserved_dynamic_paths.rb
@@ -36,10 +36,17 @@ class RenameReservedDynamicPaths < ActiveRecord::Migration
 
   DISSALLOWED_GROUP_PATHS = %w[
     activity
+    analytics
+    audit_events
     avatar
     group_members
+    hooks
     labels
+    ldap
+    ldap_group_links
     milestones
+    notification_setting
+    pipeline_quota
     subgroups
   ]
 
-- 
GitLab


From 58e36044397b7fad093dfaea0dc385e3d9ce2f06 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Thu, 4 May 2017 17:19:41 +0100
Subject: [PATCH 226/363] Fixed bug where merge request JSON would show

Closes #28909
---
 app/views/projects/merge_requests/_show.html.haml          | 2 +-
 changelogs/unreleased/merge-request-poll-json-endpoint.yml | 4 ++++
 2 files changed, 5 insertions(+), 1 deletion(-)
 create mode 100644 changelogs/unreleased/merge-request-poll-json-endpoint.yml

diff --git a/app/views/projects/merge_requests/_show.html.haml b/app/views/projects/merge_requests/_show.html.haml
index 881ee9fd59656..9e306d4543ce9 100644
--- a/app/views/projects/merge_requests/_show.html.haml
+++ b/app/views/projects/merge_requests/_show.html.haml
@@ -6,7 +6,7 @@
   = page_specific_javascript_bundle_tag('common_vue')
   = page_specific_javascript_bundle_tag('diff_notes')
 
-.merge-request{ 'data-url' => merge_request_path(@merge_request), 'data-project-path' => project_path(@merge_request.project) }
+.merge-request{ 'data-url' => merge_request_path(@merge_request, format: :json), 'data-project-path' => project_path(@merge_request.project) }
   = render "projects/merge_requests/show/mr_title"
 
   .merge-request-details.issuable-details{ data: { id: @merge_request.project.id } }
diff --git a/changelogs/unreleased/merge-request-poll-json-endpoint.yml b/changelogs/unreleased/merge-request-poll-json-endpoint.yml
new file mode 100644
index 0000000000000..6c41984e9b702
--- /dev/null
+++ b/changelogs/unreleased/merge-request-poll-json-endpoint.yml
@@ -0,0 +1,4 @@
+---
+title: Fixed bug where merge request JSON would be displayed
+merge_request:
+author:
-- 
GitLab


From dac3b6a4164c77bb142ec1644ed7c697c5c69459 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 4 May 2017 17:21:27 +0100
Subject: [PATCH 227/363] Review changes, removed api/v3 updates as its frozen,
 removed the clientside_sentry properties from the sensitive data filter as
 they\'re both available publically

---
 config/application.rb  | 3 ---
 lib/api/v3/settings.rb | 4 ----
 2 files changed, 7 deletions(-)

diff --git a/config/application.rb b/config/application.rb
index 46652637c1f36..f2ecc4ce77c7e 100644
--- a/config/application.rb
+++ b/config/application.rb
@@ -55,7 +55,6 @@ class Application < Rails::Application
     # - Webhook URLs (:hook)
     # - GitLab-shell secret token (:secret_token)
     # - Sentry DSN (:sentry_dsn)
-    # - Clientside Sentry DSN (:clientside_sentry_dsn)
     # - Deploy keys (:key)
     config.filter_parameters += %i(
       authentication_token
@@ -72,8 +71,6 @@ class Application < Rails::Application
       runners_token
       secret_token
       sentry_dsn
-      clientside_sentry_enabled
-      clientside_sentry_dsn
       variables
     )
 
diff --git a/lib/api/v3/settings.rb b/lib/api/v3/settings.rb
index 67a85e13d8c48..aa31f01b62cc6 100644
--- a/lib/api/v3/settings.rb
+++ b/lib/api/v3/settings.rb
@@ -89,10 +89,6 @@ def current_settings
         given sentry_enabled: ->(val) { val } do
           requires :sentry_dsn, type: String, desc: 'Sentry Data Source Name'
         end
-        optional :clientside_sentry_enabled, type: Boolean, desc: 'Sentry can also be used for reporting and logging clientside exceptions. https://sentry.io/for/javascript/'
-        given clientside_sentry_enabled: ->(val) { val } do
-          requires :clientside_sentry_dsn, type: String, desc: 'Clientside Sentry Data Source Name'
-        end
         optional :repository_storage, type: String, desc: 'Storage paths for new projects'
         optional :repository_checks_enabled, type: Boolean, desc: "GitLab will periodically run 'git fsck' in all project and wiki repositories to look for silent disk corruption issues."
         optional :koding_enabled, type: Boolean, desc: 'Enable Koding'
-- 
GitLab


From 136baeda508ddf46f6d91c03d4128b2ee890d205 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Thu, 4 May 2017 17:31:59 +0100
Subject: [PATCH 228/363] Fixed Karma spec

---
 .../javascripts/deploy_keys/components/app.vue    |  2 +-
 .../deploy_keys/components/app_spec.js            | 15 ++++++++++++---
 2 files changed, 13 insertions(+), 4 deletions(-)

diff --git a/app/assets/javascripts/deploy_keys/components/app.vue b/app/assets/javascripts/deploy_keys/components/app.vue
index a6552125e6692..7315a9e11cb02 100644
--- a/app/assets/javascripts/deploy_keys/components/app.vue
+++ b/app/assets/javascripts/deploy_keys/components/app.vue
@@ -66,7 +66,7 @@
     },
     beforeDestroy() {
       eventHub.$off('enable.key', this.enableKey);
-      eventHub.$off('remove.key', this.removeKey);
+      eventHub.$off('remove.key', this.disableKey);
       eventHub.$off('disable.key', this.disableKey);
     },
   };
diff --git a/spec/javascripts/deploy_keys/components/app_spec.js b/spec/javascripts/deploy_keys/components/app_spec.js
index 43b8f71850883..700897f50b0ee 100644
--- a/spec/javascripts/deploy_keys/components/app_spec.js
+++ b/spec/javascripts/deploy_keys/components/app_spec.js
@@ -116,15 +116,24 @@ describe('Deploy keys app component', () => {
     expect(vm.service.disableKey).toHaveBeenCalledWith(key.id);
   });
 
-  it('calls disableKey when removing a key', () => {
+  it('calls disableKey when removing a key', (done) => {
     const key = data.public_keys[0];
 
     spyOn(window, 'confirm').and.returnValue(true);
-    spyOn(vm, 'disableKey');
+    spyOn(vm.service, 'getKeys');
+    spyOn(vm.service, 'disableKey').and.callFake(() => new Promise((resolve) => {
+      resolve();
+
+      setTimeout(() => {
+        expect(vm.service.getKeys).toHaveBeenCalled();
+
+        done();
+      });
+    }));
 
     eventHub.$emit('remove.key', key);
 
-    expect(vm.disableKey).toHaveBeenCalledWith(key);
+    expect(vm.service.disableKey).toHaveBeenCalledWith(key.id);
   });
 
   it('hasKeys returns true when there are keys', () => {
-- 
GitLab


From 4b1d580a84016397bc1dfd0d089b8ed7ace9228d Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Thu, 4 May 2017 11:43:45 -0500
Subject: [PATCH 229/363] Fix FE conflicts

---
 .../components/collapsed_state.js             |  42 -------
 .../components/comparison_pane.js             |  70 -----------
 .../components/estimate_only_pane.js          |  14 ---
 .../time_tracking/components/help_state.js    |  25 ----
 .../components/spent_only_pane.js             |  14 ---
 .../time_tracking/components/time_tracker.js  | 117 ------------------
 app/assets/javascripts/members.js             |  21 +---
 app/assets/javascripts/milestone_select.js    |  20 ---
 app/assets/javascripts/users_select.js        |  23 ----
 .../components/sidebar/_assignee.html.haml    |   4 -
 .../shared/issuable/_search_bar.html.haml     |   4 -
 app/views/shared/issuable/_sidebar.html.haml  |   5 -
 12 files changed, 1 insertion(+), 358 deletions(-)
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/collapsed_state.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/comparison_pane.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/estimate_only_pane.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/help_state.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/spent_only_pane.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/time_tracker.js

diff --git a/app/assets/javascripts/issuable/time_tracking/components/collapsed_state.js b/app/assets/javascripts/issuable/time_tracking/components/collapsed_state.js
deleted file mode 100644
index aec13e78f42ac..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/collapsed_state.js
+++ /dev/null
@@ -1,42 +0,0 @@
-import Vue from 'vue';
-import stopwatchSvg from 'icons/_icon_stopwatch.svg';
-
-require('../../../lib/utils/pretty_time');
-
-(() => {
-  Vue.component('time-tracking-collapsed-state', {
-    name: 'time-tracking-collapsed-state',
-    props: [
-      'showComparisonState',
-      'showSpentOnlyState',
-      'showEstimateOnlyState',
-      'showNoTimeTrackingState',
-      'timeSpentHumanReadable',
-      'timeEstimateHumanReadable',
-    ],
-    methods: {
-      abbreviateTime(timeStr) {
-        return gl.utils.prettyTime.abbreviateTime(timeStr);
-      },
-    },
-    template: `
-      <div class='sidebar-collapsed-icon'>
-        ${stopwatchSvg}
-        <div class='time-tracking-collapsed-summary'>
-          <div class='compare' v-if='showComparisonState'>
-            <span>{{ abbreviateTime(timeSpentHumanReadable) }} / {{ abbreviateTime(timeEstimateHumanReadable) }}</span>
-          </div>
-          <div class='estimate-only' v-if='showEstimateOnlyState'>
-            <span class='bold'>-- / {{ abbreviateTime(timeEstimateHumanReadable) }}</span>
-          </div>
-          <div class='spend-only' v-if='showSpentOnlyState'>
-            <span class='bold'>{{ abbreviateTime(timeSpentHumanReadable) }} / --</span>
-          </div>
-          <div class='no-tracking' v-if='showNoTimeTrackingState'>
-            <span class='no-value'>None</span>
-          </div>
-        </div>
-      </div>
-      `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/comparison_pane.js b/app/assets/javascripts/issuable/time_tracking/components/comparison_pane.js
deleted file mode 100644
index c55e263f6f475..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/comparison_pane.js
+++ /dev/null
@@ -1,70 +0,0 @@
-import Vue from 'vue';
-
-require('../../../lib/utils/pretty_time');
-
-(() => {
-  const prettyTime = gl.utils.prettyTime;
-
-  Vue.component('time-tracking-comparison-pane', {
-    name: 'time-tracking-comparison-pane',
-    props: [
-      'timeSpent',
-      'timeEstimate',
-      'timeSpentHumanReadable',
-      'timeEstimateHumanReadable',
-    ],
-    computed: {
-      parsedRemaining() {
-        const diffSeconds = this.timeEstimate - this.timeSpent;
-        return prettyTime.parseSeconds(diffSeconds);
-      },
-      timeRemainingHumanReadable() {
-        return prettyTime.stringifyTime(this.parsedRemaining);
-      },
-      timeRemainingTooltip() {
-        const prefix = this.timeRemainingMinutes < 0 ? 'Over by' : 'Time remaining:';
-        return `${prefix} ${this.timeRemainingHumanReadable}`;
-      },
-      /* Diff values for comparison meter */
-      timeRemainingMinutes() {
-        return this.timeEstimate - this.timeSpent;
-      },
-      timeRemainingPercent() {
-        return `${Math.floor((this.timeSpent / this.timeEstimate) * 100)}%`;
-      },
-      timeRemainingStatusClass() {
-        return this.timeEstimate >= this.timeSpent ? 'within_estimate' : 'over_estimate';
-      },
-      /* Parsed time values */
-      parsedEstimate() {
-        return prettyTime.parseSeconds(this.timeEstimate);
-      },
-      parsedSpent() {
-        return prettyTime.parseSeconds(this.timeSpent);
-      },
-    },
-    template: `
-      <div class='time-tracking-comparison-pane'>
-        <div class='compare-meter' data-toggle='tooltip' data-placement='top' role='timeRemainingDisplay'
-          :aria-valuenow='timeRemainingTooltip'
-          :title='timeRemainingTooltip'
-          :data-original-title='timeRemainingTooltip'
-          :class='timeRemainingStatusClass'>
-          <div class='meter-container' role='timeSpentPercent' :aria-valuenow='timeRemainingPercent'>
-            <div :style='{ width: timeRemainingPercent }' class='meter-fill'></div>
-          </div>
-          <div class='compare-display-container'>
-            <div class='compare-display pull-left'>
-              <span class='compare-label'>Spent</span>
-              <span class='compare-value spent'>{{ timeSpentHumanReadable }}</span>
-            </div>
-            <div class='compare-display estimated pull-right'>
-              <span class='compare-label'>Est</span>
-              <span class='compare-value'>{{ timeEstimateHumanReadable }}</span>
-            </div>
-          </div>
-        </div>
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/estimate_only_pane.js b/app/assets/javascripts/issuable/time_tracking/components/estimate_only_pane.js
deleted file mode 100644
index a7fbd704c40d3..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/estimate_only_pane.js
+++ /dev/null
@@ -1,14 +0,0 @@
-import Vue from 'vue';
-
-(() => {
-  Vue.component('time-tracking-estimate-only-pane', {
-    name: 'time-tracking-estimate-only-pane',
-    props: ['timeEstimateHumanReadable'],
-    template: `
-      <div class='time-tracking-estimate-only-pane'>
-        <span class='bold'>Estimated:</span>
-        {{ timeEstimateHumanReadable }}
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/help_state.js b/app/assets/javascripts/issuable/time_tracking/components/help_state.js
deleted file mode 100644
index 344b29ebea4d4..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/help_state.js
+++ /dev/null
@@ -1,25 +0,0 @@
-import Vue from 'vue';
-
-(() => {
-  Vue.component('time-tracking-help-state', {
-    name: 'time-tracking-help-state',
-    props: ['docsUrl'],
-    template: `
-      <div class='time-tracking-help-state'>
-        <div class='time-tracking-info'>
-          <h4>Track time with slash commands</h4>
-          <p>Slash commands can be used in the issues description and comment boxes.</p>
-          <p>
-            <code>/estimate</code>
-            will update the estimated time with the latest command.
-          </p>
-          <p>
-            <code>/spend</code>
-            will update the sum of the time spent.
-          </p>
-          <a class='btn btn-default learn-more-button' :href='docsUrl'>Learn more</a>
-        </div>
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/spent_only_pane.js b/app/assets/javascripts/issuable/time_tracking/components/spent_only_pane.js
deleted file mode 100644
index edb9169112ff6..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/spent_only_pane.js
+++ /dev/null
@@ -1,14 +0,0 @@
-import Vue from 'vue';
-
-(() => {
-  Vue.component('time-tracking-spent-only-pane', {
-    name: 'time-tracking-spent-only-pane',
-    props: ['timeSpentHumanReadable'],
-    template: `
-      <div class='time-tracking-spend-only-pane'>
-        <span class='bold'>Spent:</span>
-        {{ timeSpentHumanReadable }}
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/time_tracker.js b/app/assets/javascripts/issuable/time_tracking/components/time_tracker.js
deleted file mode 100644
index 0213522f5519b..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/time_tracker.js
+++ /dev/null
@@ -1,117 +0,0 @@
-import Vue from 'vue';
-
-require('./help_state');
-require('./collapsed_state');
-require('./spent_only_pane');
-require('./no_tracking_pane');
-require('./estimate_only_pane');
-require('./comparison_pane');
-
-(() => {
-  Vue.component('issuable-time-tracker', {
-    name: 'issuable-time-tracker',
-    props: [
-      'time_estimate',
-      'time_spent',
-      'human_time_estimate',
-      'human_time_spent',
-      'docsUrl',
-    ],
-    data() {
-      return {
-        showHelp: false,
-      };
-    },
-    computed: {
-      timeSpent() {
-        return this.time_spent;
-      },
-      timeEstimate() {
-        return this.time_estimate;
-      },
-      timeEstimateHumanReadable() {
-        return this.human_time_estimate;
-      },
-      timeSpentHumanReadable() {
-        return this.human_time_spent;
-      },
-      hasTimeSpent() {
-        return !!this.timeSpent;
-      },
-      hasTimeEstimate() {
-        return !!this.timeEstimate;
-      },
-      showComparisonState() {
-        return this.hasTimeEstimate && this.hasTimeSpent;
-      },
-      showEstimateOnlyState() {
-        return this.hasTimeEstimate && !this.hasTimeSpent;
-      },
-      showSpentOnlyState() {
-        return this.hasTimeSpent && !this.hasTimeEstimate;
-      },
-      showNoTimeTrackingState() {
-        return !this.hasTimeEstimate && !this.hasTimeSpent;
-      },
-      showHelpState() {
-        return !!this.showHelp;
-      },
-    },
-    methods: {
-      toggleHelpState(show) {
-        this.showHelp = show;
-      },
-    },
-    template: `
-      <div class='time_tracker time-tracking-component-wrap' v-cloak>
-        <time-tracking-collapsed-state
-          :show-comparison-state='showComparisonState'
-          :show-help-state='showHelpState'
-          :show-spent-only-state='showSpentOnlyState'
-          :show-estimate-only-state='showEstimateOnlyState'
-          :time-spent-human-readable='timeSpentHumanReadable'
-          :time-estimate-human-readable='timeEstimateHumanReadable'>
-        </time-tracking-collapsed-state>
-        <div class='title hide-collapsed'>
-          Time tracking
-          <div class='help-button pull-right'
-            v-if='!showHelpState'
-            @click='toggleHelpState(true)'>
-            <i class='fa fa-question-circle' aria-hidden='true'></i>
-          </div>
-          <div class='close-help-button pull-right'
-            v-if='showHelpState'
-            @click='toggleHelpState(false)'>
-            <i class='fa fa-close' aria-hidden='true'></i>
-          </div>
-        </div>
-        <div class='time-tracking-content hide-collapsed'>
-          <time-tracking-estimate-only-pane
-            v-if='showEstimateOnlyState'
-            :time-estimate-human-readable='timeEstimateHumanReadable'>
-          </time-tracking-estimate-only-pane>
-          <time-tracking-spent-only-pane
-            v-if='showSpentOnlyState'
-            :time-spent-human-readable='timeSpentHumanReadable'>
-          </time-tracking-spent-only-pane>
-          <time-tracking-no-tracking-pane
-            v-if='showNoTimeTrackingState'>
-          </time-tracking-no-tracking-pane>
-          <time-tracking-comparison-pane
-            v-if='showComparisonState'
-            :time-estimate='timeEstimate'
-            :time-spent='timeSpent'
-            :time-spent-human-readable='timeSpentHumanReadable'
-            :time-estimate-human-readable='timeEstimateHumanReadable'>
-          </time-tracking-comparison-pane>
-          <transition name='help-state-toggle'>
-            <time-tracking-help-state
-              v-if='showHelpState'
-              :docs-url='docsUrl'>
-            </time-tracking-help-state>
-          </transition>
-        </div>
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/members.js b/app/assets/javascripts/members.js
index bfc4e55116834..8291b8c4a709d 100644
--- a/app/assets/javascripts/members.js
+++ b/app/assets/javascripts/members.js
@@ -31,27 +31,8 @@
           toggleLabel(selected, $el) {
             return $el.text();
           },
-<<<<<<< HEAD
-          clicked: (selected, $link) => {
-            this.formSubmit(null, $link);
-=======
           clicked: (options) => {
-            const $link = options.$el;
-
-            if (!$link.data('revert')) {
-              this.formSubmit(null, $link);
-            } else {
-              const { $memberListItem, $toggle, $dateInput } = this.getMemberListItems($link);
-
-              $toggle.disable();
-              $dateInput.disable();
-
-              this.overrideLdap($memberListItem, $link.data('endpoint'), false).fail(() => {
-                $toggle.enable();
-                $dateInput.enable();
-              });
-            }
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
+            this.formSubmit(null, options.$el);
           },
         });
       });
diff --git a/app/assets/javascripts/milestone_select.js b/app/assets/javascripts/milestone_select.js
index 0ca7aedb5bd4b..11e68c0a3be80 100644
--- a/app/assets/javascripts/milestone_select.js
+++ b/app/assets/javascripts/milestone_select.js
@@ -121,30 +121,10 @@
             return $value.css('display', '');
           },
           vue: $dropdown.hasClass('js-issue-board-sidebar'),
-<<<<<<< HEAD
-          clicked: function(selected, $el, e) {
-=======
-          hideRow: function(milestone) {
-            if ($('html').hasClass('issue-boards-page') && !$dropdown.hasClass('js-issue-board-sidebar') &&
-              !$dropdown.closest('.add-issues-modal').length && gl.issueBoards.BoardsStore.state.currentBoard.milestone) {
-              return milestone !== gl.issueBoards.BoardsStore.state.currentBoard.milestone.title;
-            }
-
-            return false;
-          },
-          isSelectable: function() {
-            if ($('html').hasClass('issue-boards-page') && !$dropdown.hasClass('js-issue-board-sidebar') &&
-              !$dropdown.closest('.add-issues-modal').length && gl.issueBoards.BoardsStore.state.currentBoard.milestone_id) {
-              return false;
-            }
-
-            return true;
-          },
           clicked: function(options) {
             const { $el, e } = options;
             let selected = options.selectedObj;
 
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
             var data, isIssueIndex, isMRIndex, page, boardsStore;
             page = $('body').data('page');
             isIssueIndex = page === 'projects:issues:index';
diff --git a/app/assets/javascripts/users_select.js b/app/assets/javascripts/users_select.js
index 7828bf86f6028..cd1484fa857e1 100644
--- a/app/assets/javascripts/users_select.js
+++ b/app/assets/javascripts/users_select.js
@@ -320,11 +320,6 @@ import eventHub from './sidebar/event_hub';
                 $value.css('display', '');
               }
             },
-<<<<<<< HEAD
-            vue: $dropdown.hasClass('js-issue-board-sidebar'),
-            clicked: function(user, $el, e) {
-              var isIssueIndex, isMRIndex, page, selected, isSelecting;
-=======
             multiSelect: $dropdown.hasClass('js-multiselect'),
             inputMeta: $dropdown.data('input-meta'),
             clicked: function(options) {
@@ -390,7 +385,6 @@ import eventHub from './sidebar/event_hub';
               }
 
               var isIssueIndex, isMRIndex, page, selected;
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
               page = $('body').data('page');
               isIssueIndex = page === 'projects:issues:index';
               isMRIndex = (page === page && page === 'projects:merge_requests:index');
@@ -411,24 +405,7 @@ import eventHub from './sidebar/event_hub';
                 return Issuable.filterResults($dropdown.closest('form'));
               } else if ($dropdown.hasClass('js-filter-submit')) {
                 return $dropdown.closest('form').submit();
-<<<<<<< HEAD
-              } else if ($dropdown.hasClass('js-issue-board-sidebar')) {
-                if (user.id && isSelecting) {
-                  gl.issueBoards.boardStoreIssueSet('assignee', new ListUser({
-                    id: user.id,
-                    username: user.username,
-                    name: user.name,
-                    avatar_url: user.avatar_url
-                  }));
-                } else {
-                  gl.issueBoards.boardStoreIssueDelete('assignee');
-                }
-
-                updateIssueBoardsIssue();
-              } else {
-=======
               } else if (!$dropdown.hasClass('js-multiselect')) {
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
                 selected = $dropdown.closest('.selectbox').find("input[name='" + ($dropdown.data('field-name')) + "']").val();
                 return assignTo(selected);
               }
diff --git a/app/views/projects/boards/components/sidebar/_assignee.html.haml b/app/views/projects/boards/components/sidebar/_assignee.html.haml
index 96f7f12b1d74a..0c09e71feeed5 100644
--- a/app/views/projects/boards/components/sidebar/_assignee.html.haml
+++ b/app/views/projects/boards/components/sidebar/_assignee.html.haml
@@ -16,11 +16,7 @@
         "v-if" => "issue.assignees",
         "v-for" => "assignee in issue.assignees" }
       .dropdown
-<<<<<<< HEAD
-        %button.dropdown-menu-toggle.js-user-search.js-author-search.js-issue-board-sidebar{ type: "button", data: { toggle: "dropdown", field_name: "issue[assignee_id]", first_user: (current_user.username if current_user), current_user: "true", project_id: @project.id, null_user: "true", null_user_default: "true" },
-=======
         %button.dropdown-menu-toggle.js-user-search.js-author-search.js-multiselect.js-save-user-data.js-issue-board-sidebar{ type: "button", ref: "assigneeDropdown", data: { toggle: "dropdown", field_name: "issue[assignee_ids][]", first_user: (current_user.username if current_user), current_user: "true", project_id: @project.id, null_user: "true", multi_select: "true", dropdown: { header: 'Assignee(s)'} },
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
           ":data-issuable-id" => "issue.id",
           ":data-selected" => "assigneeId",
           ":data-issue-update" => "'#{namespace_project_issues_path(@project.namespace, @project)}/' + issue.id + '.json'" }
diff --git a/app/views/shared/issuable/_search_bar.html.haml b/app/views/shared/issuable/_search_bar.html.haml
index dd727f1abfb54..f7b87171573b0 100644
--- a/app/views/shared/issuable/_search_bar.html.haml
+++ b/app/views/shared/issuable/_search_bar.html.haml
@@ -130,11 +130,7 @@
               - field_name = "update[assignee_id]"
 
             = dropdown_tag("Assignee", options: { toggle_class: "js-user-search js-update-assignee js-filter-submit js-filter-bulk-update", title: "Assign to", filter: true, dropdown_class: "dropdown-menu-user dropdown-menu-selectable",
-<<<<<<< HEAD
-              placeholder: "Search authors", data: { first_user: (current_user.username if current_user), null_user: true, current_user: true, project_id: @project.id, field_name: "update[assignee_id]", default_label: "Assignee" } })
-=======
               placeholder: "Search authors", data: { first_user: (current_user.username if current_user), null_user: true, current_user: true, project_id: @project.id, field_name: field_name } })
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
           .filter-item.inline
             = dropdown_tag("Milestone", options: { title: "Assign milestone", toggle_class: 'js-milestone-select js-extra-options js-filter-submit js-filter-bulk-update', filter: true, dropdown_class: "dropdown-menu-selectable dropdown-menu-milestone", placeholder: "Search milestones", data: { show_no: true, field_name: "update[milestone_id]", project_id: @project.id, milestones: namespace_project_milestones_path(@project.namespace, @project, :json), use_id: true, default_label: "Milestone" } })
           .filter-item.inline.labels-filter
diff --git a/app/views/shared/issuable/_sidebar.html.haml b/app/views/shared/issuable/_sidebar.html.haml
index 9bf73c87c88aa..11b844476338d 100644
--- a/app/views/shared/issuable/_sidebar.html.haml
+++ b/app/views/shared/issuable/_sidebar.html.haml
@@ -50,13 +50,8 @@
                     assign yourself
 
         .selectbox.hide-collapsed
-<<<<<<< HEAD
-          = f.hidden_field 'assignee_id', value: issuable.assignee_id, id: 'issue_assignee_id'
-          = dropdown_tag('Select assignee', options: { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true, null_user_default: true, selected: issuable.assignee_id } })
-=======
           - issuable.assignees.each do |assignee|
             = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", assignee.id, id: nil
->>>>>>> b0a2435... Merge branch 'multiple_assignees_review_upstream' into ee_master
 
           - options = { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true } }
 
-- 
GitLab


From 6f040230d2307d980dc3ebd134caebb1945a3903 Mon Sep 17 00:00:00 2001
From: Filipa Lacerda <filipa@gitlab.com>
Date: Thu, 4 May 2017 17:56:46 +0100
Subject: [PATCH 230/363] Changes after review

---
 doc/development/fe_guide/style_guide_js.md | 32 +++++++++-------------
 1 file changed, 13 insertions(+), 19 deletions(-)

diff --git a/doc/development/fe_guide/style_guide_js.md b/doc/development/fe_guide/style_guide_js.md
index a3f806c171382..d2d895172410f 100644
--- a/doc/development/fe_guide/style_guide_js.md
+++ b/doc/development/fe_guide/style_guide_js.md
@@ -225,13 +225,6 @@ A forEach will cause side effects, it will be mutating the array being iterated.
 ### Vue.js
 
 #### Basic Rules
-1. Only include one Vue.js component per file.
-1. Export components as plain objects:
-  ```javascript
-    export default {
-      template: `<h1>I'm a component</h1>
-    }
-  ```
 1. The service has it's own file
 1. The store has it's own file
 1. Use a function in the bundle file to instantiate the Vue component:
@@ -274,22 +267,13 @@ A forEach will cause side effects, it will be mutating the array being iterated.
 
 #### Naming
 1. **Extensions**: Use `.vue` extension for Vue components.
-1. **Reference Naming**: Use PascalCase for Vue components and camelCase for their instances:
+1. **Reference Naming**: Use camelCase for their instances:
   ```javascript
-    // bad
-    import cardBoard from 'cardBoard';
-
     // good
-    import CardBoard from 'cardBoard'
+    import cardBoard from 'cardBoard'
 
-    // bad
     components: {
-      CardBoard: CardBoard
-    };
-
-    // good
-    components: {
-      cardBoard: CardBoard
+      cardBoard:
     };
   ```
 
@@ -495,6 +479,16 @@ A forEach will cause side effects, it will be mutating the array being iterated.
 1. Tooltips: When using a tooltip, include the tooltip mixin
 
 1. Don't change `data-original-title`.
+  ```javascript
+    // bad
+    <span data-original-title="tooltip text">Foo</span>
+
+    // good
+    <span title="tooltip text">Foo</span>
+
+    $('span').tooltip('fixTitle');
+  ```
+
 
 ## SCSS
 - [SCSS](style_guide_scss.md)
-- 
GitLab


From 0f2a9681a318e5d27ef4e45195f2ac4b75f351dc Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Thu, 4 May 2017 18:52:01 +0300
Subject: [PATCH 231/363] [Multiple issue assignees] Resolving conflicts

---
 app/models/issue_assignee.rb         | 16 ----------------
 app/models/user.rb                   |  1 -
 app/services/notification_service.rb |  4 +++-
 3 files changed, 3 insertions(+), 18 deletions(-)

diff --git a/app/models/issue_assignee.rb b/app/models/issue_assignee.rb
index b94b55bb1d1ec..0663d3aaef8b4 100644
--- a/app/models/issue_assignee.rb
+++ b/app/models/issue_assignee.rb
@@ -7,23 +7,7 @@ class IssueAssignee < ActiveRecord::Base
   after_create :update_assignee_cache_counts
   after_destroy :update_assignee_cache_counts
 
-  # EE-specific
-  after_create :update_elasticsearch_index
-  after_destroy :update_elasticsearch_index
-  # EE-specific
-
   def update_assignee_cache_counts
     assignee&.update_cache_counts
   end
-
-  def update_elasticsearch_index
-    if current_application_settings.elasticsearch_indexing?
-      ElasticIndexerWorker.perform_async(
-        :update,
-        'Issue',
-        issue.id,
-        changed_fields: ['assignee_ids']
-      )
-    end
-  end
 end
diff --git a/app/models/user.rb b/app/models/user.rb
index a3126cbb644a4..6015643b56507 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -89,7 +89,6 @@ class User < ActiveRecord::Base
   has_many :subscriptions,            dependent: :destroy
   has_many :recent_events, -> { order "id DESC" }, foreign_key: :author_id,   class_name: "Event"
   has_many :oauth_applications, class_name: 'Doorkeeper::Application', as: :owner, dependent: :destroy
-
   has_one  :abuse_report,             dependent: :destroy, foreign_key: :user_id
   has_many :reported_abuse_reports,   dependent: :destroy, foreign_key: :reporter_id, class_name: "AbuseReport"
   has_many :spam_logs,                dependent: :destroy
diff --git a/app/services/notification_service.rb b/app/services/notification_service.rb
index fe9f5ae2b3304..c65c66d715040 100644
--- a/app/services/notification_service.rb
+++ b/app/services/notification_service.rb
@@ -74,12 +74,14 @@ def reassigned_issue(issue, current_user, previous_assignees = [])
       previous_assignee: previous_assignees
     )
 
+    previous_assignee_ids = previous_assignees.map(&:id)
+
     recipients.each do |recipient|
       mailer.send(
         :reassigned_issue_email,
         recipient.id,
         issue.id,
-        previous_assignees.map(&:id),
+        previous_assignee_ids,
         current_user.id
       ).deliver_later
     end
-- 
GitLab


From fc464f1ff27734df81f7f85cbc23e93902c97cda Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Thu, 4 May 2017 20:22:03 +0300
Subject: [PATCH 232/363] Multiple issue assignee: fixed
 services/issues/update_service by using new NoteSummary

---
 app/services/system_note_service.rb  |  2 +-
 spec/features/boards/sidebar_spec.rb | 21 ---------------------
 2 files changed, 1 insertion(+), 22 deletions(-)

diff --git a/app/services/system_note_service.rb b/app/services/system_note_service.rb
index 898c69b3f8c2c..1dee791cfd615 100644
--- a/app/services/system_note_service.rb
+++ b/app/services/system_note_service.rb
@@ -84,7 +84,7 @@ def change_issue_assignees(issue, project, author, old_assignees)
         "assigned to #{issue.assignees.map(&:to_reference).to_sentence}"
       end
 
-    create_note(noteable: issue, project: project, author: author, note: body)
+    NoteSummary.new(issue, project, author, body, action: 'assignee')
   end
 
   # Called when one or more labels on a Noteable are added and/or removed
diff --git a/spec/features/boards/sidebar_spec.rb b/spec/features/boards/sidebar_spec.rb
index 02b6b5dc88865..7c53d2b47d9c8 100644
--- a/spec/features/boards/sidebar_spec.rb
+++ b/spec/features/boards/sidebar_spec.rb
@@ -21,7 +21,6 @@
     Timecop.freeze
 
     project.team << [user, :master]
-    project.team.add_developer(user2)
 
     login_as(user)
 
@@ -103,26 +102,6 @@
       expect(card).to have_selector('.avatar')
     end
 
-    it 'adds multiple assignees' do
-      click_card(card)
-
-      page.within('.assignee') do
-        click_link 'Edit'
-
-        wait_for_ajax
-
-        page.within('.dropdown-menu-user') do
-          click_link user.name
-          click_link user2.name
-        end
-
-        expect(page).to have_content(user.name)
-        expect(page).to have_content(user2.name)
-      end
-
-      expect(card.all('.avatar').length).to eq(2)
-    end
-
     it 'removes the assignee' do
       card_two = first('.board').find('.card:nth-child(2)')
       click_card(card_two)
-- 
GitLab


From cdcbe99aa749d4516f47ca9d2e7d510dc29d1c91 Mon Sep 17 00:00:00 2001
From: Douglas Barbosa Alexandre <dbalexandre@gmail.com>
Date: Wed, 3 May 2017 22:36:02 -0300
Subject: [PATCH 233/363] Add last_repository_updated_at to projects

---
 ...503004125_add_last_repository_updated_at_to_projects.rb | 7 +++++++
 db/schema.rb                                               | 3 ++-
 2 files changed, 9 insertions(+), 1 deletion(-)
 create mode 100644 db/migrate/20170503004125_add_last_repository_updated_at_to_projects.rb

diff --git a/db/migrate/20170503004125_add_last_repository_updated_at_to_projects.rb b/db/migrate/20170503004125_add_last_repository_updated_at_to_projects.rb
new file mode 100644
index 0000000000000..00c685cf342fe
--- /dev/null
+++ b/db/migrate/20170503004125_add_last_repository_updated_at_to_projects.rb
@@ -0,0 +1,7 @@
+class AddLastRepositoryUpdatedAtToProjects < ActiveRecord::Migration
+  DOWNTIME = false
+
+  def change
+    add_column :projects, :last_repository_updated_at, :datetime
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 01c0f00c92496..e12c6e30d8e0d 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170502091007) do
+ActiveRecord::Schema.define(version: 20170503004125) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -971,6 +971,7 @@
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
     t.string "import_jid"
     t.integer "cached_markdown_version"
+    t.datetime "last_repository_updated_at"
   end
 
   add_index "projects", ["ci_id"], name: "index_projects_on_ci_id", using: :btree
-- 
GitLab


From 341964da45460d5c6593d0e08178ecf61e726e91 Mon Sep 17 00:00:00 2001
From: Douglas Barbosa Alexandre <dbalexandre@gmail.com>
Date: Wed, 3 May 2017 22:37:43 -0300
Subject: [PATCH 234/363] Add index to last_repository_updated_at on projects

---
 ...x_to_last_repository_updated_at_on_projects.rb | 15 +++++++++++++++
 db/schema.rb                                      |  3 ++-
 2 files changed, 17 insertions(+), 1 deletion(-)
 create mode 100644 db/migrate/20170503004425_add_index_to_last_repository_updated_at_on_projects.rb

diff --git a/db/migrate/20170503004425_add_index_to_last_repository_updated_at_on_projects.rb b/db/migrate/20170503004425_add_index_to_last_repository_updated_at_on_projects.rb
new file mode 100644
index 0000000000000..6144d74745c8b
--- /dev/null
+++ b/db/migrate/20170503004425_add_index_to_last_repository_updated_at_on_projects.rb
@@ -0,0 +1,15 @@
+class AddIndexToLastRepositoryUpdatedAtOnProjects < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  disable_ddl_transaction!
+
+  def up
+    add_concurrent_index(:projects, :last_repository_updated_at)
+  end
+
+  def down
+    remove_concurrent_index(:projects, :last_repository_updated_at) if index_exists?(:projects, :last_repository_updated_at)
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index e12c6e30d8e0d..5d3e22d612304 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170503004125) do
+ActiveRecord::Schema.define(version: 20170503004425) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -980,6 +980,7 @@
   add_index "projects", ["description"], name: "index_projects_on_description_trigram", using: :gin, opclasses: {"description"=>"gin_trgm_ops"}
   add_index "projects", ["last_activity_at"], name: "index_projects_on_last_activity_at", using: :btree
   add_index "projects", ["last_repository_check_failed"], name: "index_projects_on_last_repository_check_failed", using: :btree
+  add_index "projects", ["last_repository_updated_at"], name: "index_projects_on_last_repository_updated_at", using: :btree
   add_index "projects", ["name"], name: "index_projects_on_name_trigram", using: :gin, opclasses: {"name"=>"gin_trgm_ops"}
   add_index "projects", ["namespace_id"], name: "index_projects_on_namespace_id", using: :btree
   add_index "projects", ["path"], name: "index_projects_on_path", using: :btree
-- 
GitLab


From 56db54d3e53f5bac4aa5190d3189f015e3fdbfa7 Mon Sep 17 00:00:00 2001
From: Douglas Barbosa Alexandre <dbalexandre@gmail.com>
Date: Wed, 3 May 2017 22:39:41 -0300
Subject: [PATCH 235/363] Set last_repository_updated_at to created_at upon
 project creation

---
 app/models/project.rb       | 5 +++++
 spec/models/project_spec.rb | 8 ++++++++
 2 files changed, 13 insertions(+)

diff --git a/app/models/project.rb b/app/models/project.rb
index 025db89ebfd82..edbca3b537b33 100644
--- a/app/models/project.rb
+++ b/app/models/project.rb
@@ -53,6 +53,11 @@ def set_last_activity_at
     update_column(:last_activity_at, self.created_at)
   end
 
+  after_create :set_last_repository_updated_at
+  def set_last_repository_updated_at
+    update_column(:last_repository_updated_at, self.created_at)
+  end
+
   after_destroy :remove_pages
 
   # update visibility_level of forks
diff --git a/spec/models/project_spec.rb b/spec/models/project_spec.rb
index 36ce3070a6eb0..316ece87faac5 100644
--- a/spec/models/project_spec.rb
+++ b/spec/models/project_spec.rb
@@ -1925,4 +1925,12 @@ def enable_lfs
         not_to raise_error
     end
   end
+
+  describe '#last_repository_updated_at' do
+    it 'sets to created_at upon creation' do
+      project = create(:empty_project, created_at: 2.hours.ago)
+
+      expect(project.last_repository_updated_at.to_i).to eq(project.created_at.to_i)
+    end
+  end
 end
-- 
GitLab


From 91b5aaf770836b7101414527a4650db5fa669ce2 Mon Sep 17 00:00:00 2001
From: Douglas Barbosa Alexandre <dbalexandre@gmail.com>
Date: Wed, 3 May 2017 22:44:39 -0300
Subject: [PATCH 236/363] Update last_repository_updated_at when a push event
 is created

---
 app/models/event.rb       |  6 ++++++
 spec/models/event_spec.rb | 42 +++++++++++++++++++++++++++++++--------
 2 files changed, 40 insertions(+), 8 deletions(-)

diff --git a/app/models/event.rb b/app/models/event.rb
index b780c1faf81d9..e6fad46077aa1 100644
--- a/app/models/event.rb
+++ b/app/models/event.rb
@@ -30,6 +30,7 @@ class Event < ActiveRecord::Base
 
   # Callbacks
   after_create :reset_project_activity
+  after_create :set_last_repository_updated_at, if: :push?
 
   # Scopes
   scope :recent, -> { reorder(id: :desc) }
@@ -357,4 +358,9 @@ def authored_by?(user)
   def recent_update?
     project.last_activity_at > RESET_PROJECT_ACTIVITY_INTERVAL.ago
   end
+
+  def set_last_repository_updated_at
+    Project.unscoped.where(id: project_id).
+      update_all(last_repository_updated_at: created_at)
+  end
 end
diff --git a/spec/models/event_spec.rb b/spec/models/event_spec.rb
index 8c90a538f5723..a9c5b604268cd 100644
--- a/spec/models/event_spec.rb
+++ b/spec/models/event_spec.rb
@@ -15,13 +15,39 @@
   end
 
   describe 'Callbacks' do
-    describe 'after_create :reset_project_activity' do
-      let(:project) { create(:empty_project) }
+    let(:project) { create(:empty_project) }
 
+    describe 'after_create :reset_project_activity' do
       it 'calls the reset_project_activity method' do
         expect_any_instance_of(described_class).to receive(:reset_project_activity)
 
-        create_event(project, project.owner)
+        create_push_event(project, project.owner)
+      end
+    end
+
+    describe 'after_create :set_last_repository_updated_at' do
+      context 'with a push event' do
+        it 'updates the project last_repository_updated_at' do
+          project.update(last_repository_updated_at: 1.year.ago)
+
+          create_push_event(project, project.owner)
+
+          project.reload
+
+          expect(project.last_repository_updated_at).to be_within(1.minute).of(Time.now)
+        end
+      end
+
+      context 'without a push event' do
+        it 'does not update the project last_repository_updated_at' do
+          project.update(last_repository_updated_at: 1.year.ago)
+
+          create(:closed_issue_event, project: project, author: project.owner)
+
+          project.reload
+
+          expect(project.last_repository_updated_at).to be_within(1.minute).of(1.year.ago)
+        end
       end
     end
   end
@@ -29,7 +55,7 @@
   describe "Push event" do
     let(:project) { create(:empty_project, :private) }
     let(:user) { project.owner }
-    let(:event) { create_event(project, user) }
+    let(:event) { create_push_event(project, user) }
 
     it do
       expect(event.push?).to be_truthy
@@ -243,7 +269,7 @@
         expect(project).not_to receive(:update_column).
           with(:last_activity_at, a_kind_of(Time))
 
-        create_event(project, project.owner)
+        create_push_event(project, project.owner)
       end
     end
 
@@ -251,11 +277,11 @@
       it 'updates the project' do
         project.update(last_activity_at: 1.year.ago)
 
-        create_event(project, project.owner)
+        create_push_event(project, project.owner)
 
         project.reload
 
-        project.last_activity_at <= 1.minute.ago
+        expect(project.last_activity_at).to be_within(1.minute).of(Time.now)
       end
     end
   end
@@ -278,7 +304,7 @@
     end
   end
 
-  def create_event(project, user, attrs = {})
+  def create_push_event(project, user, attrs = {})
     data = {
       before: Gitlab::Git::BLANK_SHA,
       after: "0220c11b9a3e6c69dc8fd35321254ca9a7b98f7e",
-- 
GitLab


From dbb50ed890e9a81dbba288fac033dec7f0b3efeb Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Thu, 4 May 2017 13:36:18 -0500
Subject: [PATCH 237/363] Restrict FE assignees to 1

---
 app/assets/javascripts/users_select.js                     | 6 ++++--
 .../projects/boards/components/sidebar/_assignee.html.haml | 4 ++--
 app/views/shared/issuable/_sidebar.html.haml               | 7 ++++---
 3 files changed, 10 insertions(+), 7 deletions(-)

diff --git a/app/assets/javascripts/users_select.js b/app/assets/javascripts/users_select.js
index cd1484fa857e1..64171e491e576 100644
--- a/app/assets/javascripts/users_select.js
+++ b/app/assets/javascripts/users_select.js
@@ -388,10 +388,12 @@ import eventHub from './sidebar/event_hub';
               page = $('body').data('page');
               isIssueIndex = page === 'projects:issues:index';
               isMRIndex = (page === page && page === 'projects:merge_requests:index');
-              isSelecting = (user.id !== selectedId);
-              selectedId = isSelecting ? user.id : selectedIdDefault;
               if ($dropdown.hasClass('js-filter-bulk-update') || $dropdown.hasClass('js-issuable-form-dropdown')) {
                 e.preventDefault();
+
+                let isSelecting = (user.id !== selectedId);
+                selectedId = isSelecting ? user.id : selectedIdDefault;
+
                 if (selectedId === gon.current_user_id) {
                   $('.assign-to-me-link').hide();
                 } else {
diff --git a/app/views/projects/boards/components/sidebar/_assignee.html.haml b/app/views/projects/boards/components/sidebar/_assignee.html.haml
index 0c09e71feeed5..642da679f97be 100644
--- a/app/views/projects/boards/components/sidebar/_assignee.html.haml
+++ b/app/views/projects/boards/components/sidebar/_assignee.html.haml
@@ -16,11 +16,11 @@
         "v-if" => "issue.assignees",
         "v-for" => "assignee in issue.assignees" }
       .dropdown
-        %button.dropdown-menu-toggle.js-user-search.js-author-search.js-multiselect.js-save-user-data.js-issue-board-sidebar{ type: "button", ref: "assigneeDropdown", data: { toggle: "dropdown", field_name: "issue[assignee_ids][]", first_user: (current_user.username if current_user), current_user: "true", project_id: @project.id, null_user: "true", multi_select: "true", dropdown: { header: 'Assignee(s)'} },
+        %button.dropdown-menu-toggle.js-user-search.js-author-search.js-multiselect.js-save-user-data.js-issue-board-sidebar{ type: "button", ref: "assigneeDropdown", data: { toggle: "dropdown", field_name: "issue[assignee_ids][]", first_user: (current_user.username if current_user), current_user: "true", project_id: @project.id, null_user: "true", multi_select: "true", 'max-select' => 1, dropdown: { header: 'Assignee' } },
           ":data-issuable-id" => "issue.id",
           ":data-selected" => "assigneeId",
           ":data-issue-update" => "'#{namespace_project_issues_path(@project.namespace, @project)}/' + issue.id + '.json'" }
-          Select assignee(s)
+          Select assignee
           = icon("chevron-down")
         .dropdown-menu.dropdown-select.dropdown-menu-user.dropdown-menu-selectable.dropdown-menu-author
           = dropdown_title("Assign to")
diff --git a/app/views/shared/issuable/_sidebar.html.haml b/app/views/shared/issuable/_sidebar.html.haml
index 11b844476338d..44e624c15a78d 100644
--- a/app/views/shared/issuable/_sidebar.html.haml
+++ b/app/views/shared/issuable/_sidebar.html.haml
@@ -53,17 +53,18 @@
           - issuable.assignees.each do |assignee|
             = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", assignee.id, id: nil
 
-          - options = { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true } }
+          - options = { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_ids][]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true } }
 
           - if issuable.instance_of?(Issue)
             - if issuable.assignees.length == 0
               = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", 0, id: nil
-            - title = 'Select assignee(s)'
+            - title = 'Select assignee'
             - options[:toggle_class] += ' js-multiselect js-save-user-data'
             - options[:data][:field_name] = "#{issuable.to_ability_name}[assignee_ids][]"
             - options[:data][:multi_select] = true
             - options[:data]['dropdown-title'] = title
-            - options[:data]['dropdown-header'] = 'Assignee(s)'
+            - options[:data]['dropdown-header'] = 'Assignee'
+            - options[:data]['max-select'] = 1
           - else
             - title = 'Select assignee'
 
-- 
GitLab


From 9f2edaa84642113d33fee37eee1a8daca2be54c0 Mon Sep 17 00:00:00 2001
From: Douglas Barbosa Alexandre <dbalexandre@gmail.com>
Date: Wed, 3 May 2017 22:47:10 -0300
Subject: [PATCH 238/363] Update last_repository_updated_at when the wiki is
 updated

---
 app/models/project_wiki.rb       |  2 +-
 spec/models/project_wiki_spec.rb | 21 +++++++++++++++------
 2 files changed, 16 insertions(+), 7 deletions(-)

diff --git a/app/models/project_wiki.rb b/app/models/project_wiki.rb
index 70eef359cdd09..189c106b70b67 100644
--- a/app/models/project_wiki.rb
+++ b/app/models/project_wiki.rb
@@ -183,6 +183,6 @@ def path_to_repo
   end
 
   def update_project_activity
-    @project.touch(:last_activity_at)
+    @project.touch(:last_activity_at, :last_repository_updated_at)
   end
 end
diff --git a/spec/models/project_wiki_spec.rb b/spec/models/project_wiki_spec.rb
index b5b9cd024b056..969e9f7a130de 100644
--- a/spec/models/project_wiki_spec.rb
+++ b/spec/models/project_wiki_spec.rb
@@ -213,9 +213,12 @@
     end
 
     it 'updates project activity' do
-      expect(subject).to receive(:update_project_activity)
-
       subject.create_page('Test Page', 'This is content')
+
+      project.reload
+
+      expect(project.last_activity_at).to be_within(1.minute).of(Time.now)
+      expect(project.last_repository_updated_at).to be_within(1.minute).of(Time.now)
     end
   end
 
@@ -240,9 +243,12 @@
     end
 
     it 'updates project activity' do
-      expect(subject).to receive(:update_project_activity)
-
       subject.update_page(@gollum_page, 'Yet more content', :markdown, 'Updated page again')
+
+      project.reload
+
+      expect(project.last_activity_at).to be_within(1.minute).of(Time.now)
+      expect(project.last_repository_updated_at).to be_within(1.minute).of(Time.now)
     end
   end
 
@@ -258,9 +264,12 @@
     end
 
     it 'updates project activity' do
-      expect(subject).to receive(:update_project_activity)
-
       subject.delete_page(@page)
+
+      project.reload
+
+      expect(project.last_activity_at).to be_within(1.minute).of(Time.now)
+      expect(project.last_repository_updated_at).to be_within(1.minute).of(Time.now)
     end
   end
 
-- 
GitLab


From 41d0f23108ddce3f9285898de75508564edd80b3 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 4 May 2017 18:49:56 +0000
Subject: [PATCH 239/363] Resolve "Web GUI: Unable to create new branch from
 commit SHA"

---
 app/assets/javascripts/new_branch_form.js     | 16 +++++++
 app/views/projects/branches/new.html.haml     | 12 ++---
 app/views/projects/compare/_form.html.haml    |  4 +-
 .../_ref_dropdown.html.haml                   |  4 +-
 .../branches/new_branch_ref_dropdown_spec.rb  | 48 +++++++++++++++++++
 5 files changed, 75 insertions(+), 9 deletions(-)
 rename app/views/{projects/compare => shared}/_ref_dropdown.html.haml (50%)
 create mode 100644 spec/features/projects/branches/new_branch_ref_dropdown_spec.rb

diff --git a/app/assets/javascripts/new_branch_form.js b/app/assets/javascripts/new_branch_form.js
index 5828f460a235a..67046d52a653c 100644
--- a/app/assets/javascripts/new_branch_form.js
+++ b/app/assets/javascripts/new_branch_form.js
@@ -34,6 +34,7 @@
         filterByText: true,
         remote: false,
         fieldName: $branchSelect.data('field-name'),
+        filterInput: 'input[type="search"]',
         selectable: true,
         isSelectable: function(branch, $el) {
           return !$el.hasClass('is-active');
@@ -50,6 +51,21 @@
           }
         }
       });
+
+      const $dropdownContainer = $branchSelect.closest('.dropdown');
+      const $fieldInput = $(`input[name="${$branchSelect.data('field-name')}"]`, $dropdownContainer);
+      const $filterInput = $('input[type="search"]', $dropdownContainer);
+
+      $filterInput.on('keyup', (e) => {
+        const keyCode = e.keyCode || e.which;
+        if (keyCode !== 13) return;
+
+        const text = $filterInput.val();
+        $fieldInput.val(text);
+        $('.dropdown-toggle-text', $branchSelect).text(text);
+
+        $dropdownContainer.removeClass('open');
+      });
     };
 
     NewBranchForm.prototype.setupRestrictions = function() {
diff --git a/app/views/projects/branches/new.html.haml b/app/views/projects/branches/new.html.haml
index d3c3e40d5185a..796ecdfd0145a 100644
--- a/app/views/projects/branches/new.html.haml
+++ b/app/views/projects/branches/new.html.haml
@@ -1,4 +1,5 @@
 - page_title "New Branch"
+- default_ref = params[:ref] || @project.default_branch
 
 - if @error
   .alert.alert-danger
@@ -16,12 +17,11 @@
       .help-block.text-danger.js-branch-name-error
   .form-group
     = label_tag :ref, 'Create from', class: 'control-label'
-    .col-sm-10
-      = hidden_field_tag :ref, params[:ref] || @project.default_branch
-      = dropdown_tag(params[:ref] || @project.default_branch,
-                     options: { toggle_class: 'js-branch-select wide',
-                                filter: true, dropdown_class: "dropdown-menu-selectable", placeholder: "Search branches",
-                                data: { selected: params[:ref] || @project.default_branch, field_name: 'ref' } })
+    .col-sm-10.dropdown.create-from
+      = hidden_field_tag :ref, default_ref
+      = button_tag type: 'button', title: default_ref, class: 'dropdown-toggle form-control js-branch-select', required: true, data: { toggle: 'dropdown', selected: default_ref, field_name: 'ref' } do
+        .text-left.dropdown-toggle-text= default_ref
+      = render 'shared/ref_dropdown', dropdown_class: 'wide'
       .help-block Existing branch name, tag, or commit SHA
   .form-actions
     = button_tag 'Create branch', class: 'btn btn-create', tabindex: 3
diff --git a/app/views/projects/compare/_form.html.haml b/app/views/projects/compare/_form.html.haml
index 0f080b6aceefb..1f4c9fac54c73 100644
--- a/app/views/projects/compare/_form.html.haml
+++ b/app/views/projects/compare/_form.html.haml
@@ -9,7 +9,7 @@
         = hidden_field_tag :from, params[:from]
         = button_tag type: 'button', title: params[:from], class: "form-control compare-dropdown-toggle js-compare-dropdown has-tooltip", required: true, data: { refs_url: refs_namespace_project_path(@project.namespace, @project), toggle: "dropdown", target: ".js-compare-from-dropdown", selected: params[:from], field_name: :from } do
           .dropdown-toggle-text.str-truncated= params[:from] || 'Select branch/tag'
-      = render "ref_dropdown"
+      = render 'shared/ref_dropdown'
     .compare-ellipsis.inline ...
     .form-group.dropdown.compare-form-group.to.js-compare-to-dropdown
       .input-group.inline-input-group
@@ -17,7 +17,7 @@
         = hidden_field_tag :to, params[:to]
         = button_tag type: 'button', title: params[:to], class: "form-control compare-dropdown-toggle js-compare-dropdown has-tooltip", required: true, data: { refs_url: refs_namespace_project_path(@project.namespace, @project), toggle: "dropdown", target: ".js-compare-to-dropdown", selected: params[:to], field_name: :to } do
           .dropdown-toggle-text.str-truncated= params[:to] || 'Select branch/tag'
-      = render "ref_dropdown"
+      = render 'shared/ref_dropdown'
     &nbsp;
     = button_tag "Compare", class: "btn btn-create commits-compare-btn"
     - if @merge_request.present?
diff --git a/app/views/projects/compare/_ref_dropdown.html.haml b/app/views/shared/_ref_dropdown.html.haml
similarity index 50%
rename from app/views/projects/compare/_ref_dropdown.html.haml
rename to app/views/shared/_ref_dropdown.html.haml
index 05fb37cdc0f52..96f68c80c48cd 100644
--- a/app/views/projects/compare/_ref_dropdown.html.haml
+++ b/app/views/shared/_ref_dropdown.html.haml
@@ -1,4 +1,6 @@
-.dropdown-menu.dropdown-menu-selectable
+- dropdown_class = local_assigns.fetch(:dropdown_class, '')
+
+.dropdown-menu.dropdown-menu-selectable{ class: dropdown_class }
   = dropdown_title "Select Git revision"
   = dropdown_filter "Filter by Git revision"
   = dropdown_content
diff --git a/spec/features/projects/branches/new_branch_ref_dropdown_spec.rb b/spec/features/projects/branches/new_branch_ref_dropdown_spec.rb
new file mode 100644
index 0000000000000..cfc782c98ad42
--- /dev/null
+++ b/spec/features/projects/branches/new_branch_ref_dropdown_spec.rb
@@ -0,0 +1,48 @@
+require 'spec_helper'
+
+describe 'New Branch Ref Dropdown', :js, :feature do
+  let(:user) { create(:user) }
+  let(:project) { create(:project, :public) }
+  let(:toggle) { find('.create-from .dropdown-toggle') }
+
+  before do
+    project.add_master(user)
+
+    login_as(user)
+    visit new_namespace_project_branch_path(project.namespace, project)
+  end
+
+  it 'filters a list of branches and tags' do
+    toggle.click
+
+    filter_by('v1.0.0')
+
+    expect(items_count).to be(1)
+
+    filter_by('video')
+
+    expect(items_count).to be(1)
+
+    find('.create-from .dropdown-content li').click
+
+    expect(toggle).to have_content 'video'
+  end
+
+  it 'accepts a manually entered commit SHA' do
+    toggle.click
+
+    filter_by('somecommitsha')
+
+    find('.create-from input[type=search]').send_keys(:enter)
+
+    expect(toggle).to have_content 'somecommitsha'
+  end
+
+  def items_count
+    all('.create-from .dropdown-content li').length
+  end
+
+  def filter_by(filter_text)
+    fill_in 'Filter by Git revision', with: filter_text
+  end
+end
-- 
GitLab


From 9c4ee8ee64dcba297de13d453f6eea18304d50db Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Thu, 4 May 2017 13:55:13 -0500
Subject: [PATCH 240/363] Fix eslint and rubocop

---
 .../protected_branches/protected_branch_access_dropdown.js     | 3 +--
 app/assets/javascripts/users_select.js                         | 2 +-
 spec/features/issues/filtered_search/filter_issues_spec.rb     | 1 -
 spec/javascripts/boards/board_card_spec.js                     | 2 +-
 4 files changed, 3 insertions(+), 5 deletions(-)

diff --git a/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js b/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
index 1266d70f073aa..42993a252c33d 100644
--- a/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
+++ b/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
@@ -20,8 +20,7 @@
           }
         },
         clicked(opts) {
-          const { $el, e } = opts;
-          const item = opts.selectedObj;
+          const { e } = opts;
 
           e.preventDefault();
           onSelect();
diff --git a/app/assets/javascripts/users_select.js b/app/assets/javascripts/users_select.js
index 64171e491e576..2dcaa24e93a1f 100644
--- a/app/assets/javascripts/users_select.js
+++ b/app/assets/javascripts/users_select.js
@@ -391,7 +391,7 @@ import eventHub from './sidebar/event_hub';
               if ($dropdown.hasClass('js-filter-bulk-update') || $dropdown.hasClass('js-issuable-form-dropdown')) {
                 e.preventDefault();
 
-                let isSelecting = (user.id !== selectedId);
+                const isSelecting = (user.id !== selectedId);
                 selectedId = isSelecting ? user.id : selectedIdDefault;
 
                 if (selectedId === gon.current_user_id) {
diff --git a/spec/features/issues/filtered_search/filter_issues_spec.rb b/spec/features/issues/filtered_search/filter_issues_spec.rb
index ece62c8da4131..a8f4e2d7e10ad 100644
--- a/spec/features/issues/filtered_search/filter_issues_spec.rb
+++ b/spec/features/issues/filtered_search/filter_issues_spec.rb
@@ -54,7 +54,6 @@ def select_search_at_index(pos)
     create(:issue, project: project, title: "issue by assignee", milestone: milestone, author: user, assignees: [user])
     create(:issue, project: project, title: "issue by assignee with searchTerm", milestone: milestone, author: user, assignees: [user])
 
-
     issue = create(:issue,
       title: "Bug 2",
       project: project,
diff --git a/spec/javascripts/boards/board_card_spec.js b/spec/javascripts/boards/board_card_spec.js
index de072e7e470bf..2064ca2632bbf 100644
--- a/spec/javascripts/boards/board_card_spec.js
+++ b/spec/javascripts/boards/board_card_spec.js
@@ -6,7 +6,7 @@
 /* global BoardService */
 
 import Vue from 'vue';
-import '~/boards/models/user';
+import '~/boards/models/assignee';
 
 require('~/boards/models/list');
 require('~/boards/models/label');
-- 
GitLab


From ad6ac17c5434f7eb87005dc3603b4ae9409c333f Mon Sep 17 00:00:00 2001
From: Jose Ivan Vargas <jvargas@gitlab.com>
Date: Fri, 28 Apr 2017 15:20:58 -0500
Subject: [PATCH 241/363] Added rescue block for the test method for the
 prometheus service

---
 .../unreleased/prometheus-integration-test-setting-fix.yml  | 4 ++++
 lib/gitlab/prometheus.rb                                    | 6 +++++-
 spec/models/project_services/prometheus_service_spec.rb     | 5 +++--
 3 files changed, 12 insertions(+), 3 deletions(-)
 create mode 100644 changelogs/unreleased/prometheus-integration-test-setting-fix.yml

diff --git a/changelogs/unreleased/prometheus-integration-test-setting-fix.yml b/changelogs/unreleased/prometheus-integration-test-setting-fix.yml
new file mode 100644
index 0000000000000..c65c682a11442
--- /dev/null
+++ b/changelogs/unreleased/prometheus-integration-test-setting-fix.yml
@@ -0,0 +1,4 @@
+---
+title: Added rescue block for the test method
+merge_request: 10994
+author:
diff --git a/lib/gitlab/prometheus.rb b/lib/gitlab/prometheus.rb
index 62239779454f6..37afec5e7df4c 100644
--- a/lib/gitlab/prometheus.rb
+++ b/lib/gitlab/prometheus.rb
@@ -49,7 +49,11 @@ def join_api_url(type, args = {})
     end
 
     def get(url)
-      handle_response(HTTParty.get(url))
+      begin
+        handle_response(HTTParty.get(url))
+      rescue SocketError
+        raise PrometheusError, "Can't connect to #{url}"
+      end
     end
 
     def handle_response(response)
diff --git a/spec/models/project_services/prometheus_service_spec.rb b/spec/models/project_services/prometheus_service_spec.rb
index d15079b686be9..5ef1b53be1570 100644
--- a/spec/models/project_services/prometheus_service_spec.rb
+++ b/spec/models/project_services/prometheus_service_spec.rb
@@ -93,11 +93,12 @@
 
     [404, 500].each do |status|
       context "when Prometheus responds with #{status}" do
+        body_response = 'QUERY_FAILED'
         before do
-          stub_all_prometheus_requests(environment.slug, status: status, body: 'QUERY FAILED!')
+          stub_all_prometheus_requests(environment.slug, status: status, body: body_response)
         end
 
-        it { is_expected.to eq(success: false, result: %(#{status} - "QUERY FAILED!")) }
+        it { is_expected.to eq(success: false, result: %(#{status} - \"#{body_response}\")) }
       end
     end
   end
-- 
GitLab


From 63a5d98a7c921289e9b43f4b54c03614427f7eda Mon Sep 17 00:00:00 2001
From: Jose Ivan Vargas <jvargas@gitlab.com>
Date: Tue, 2 May 2017 11:16:59 -0500
Subject: [PATCH 242/363] Added specs

---
 lib/gitlab/prometheus.rb           | 10 +++++-----
 spec/lib/gitlab/prometheus_spec.rb | 18 ++++++++++++++++++
 spec/support/prometheus_helpers.rb | 10 ++++++++++
 3 files changed, 33 insertions(+), 5 deletions(-)

diff --git a/lib/gitlab/prometheus.rb b/lib/gitlab/prometheus.rb
index 37afec5e7df4c..94541009100a4 100644
--- a/lib/gitlab/prometheus.rb
+++ b/lib/gitlab/prometheus.rb
@@ -49,11 +49,11 @@ def join_api_url(type, args = {})
     end
 
     def get(url)
-      begin
-        handle_response(HTTParty.get(url))
-      rescue SocketError
-        raise PrometheusError, "Can't connect to #{url}"
-      end
+      handle_response(HTTParty.get(url))
+    rescue SocketError
+      raise PrometheusError, "Can't connect to #{url}"
+    rescue OpenSSL::SSL::SSLError
+      raise PrometheusError, "#{url} contains invalid SSL data"
     end
 
     def handle_response(response)
diff --git a/spec/lib/gitlab/prometheus_spec.rb b/spec/lib/gitlab/prometheus_spec.rb
index 280264188e2e2..d868366951829 100644
--- a/spec/lib/gitlab/prometheus_spec.rb
+++ b/spec/lib/gitlab/prometheus_spec.rb
@@ -49,6 +49,24 @@
     end
   end
 
+  describe 'failure to reach a prometheus url' do
+    prometheus_invalid_url = 'https://prometheus.invalid.example.com'
+
+    it 'raises a Gitlab::PrometheusError error when a SocketError is rescued' do
+      req_stub = stub_prometheus_request_with_socket_exception(prometheus_invalid_url)
+
+      expect { subject.send(:get, prometheus_invalid_url) }.to raise_error(Gitlab::PrometheusError, "Can't connect to #{prometheus_invalid_url}")
+      expect(req_stub).to have_been_requested
+    end
+
+    it 'raises a Gitlab::PrometheusError error when a SSLError is rescued' do
+      req_stub = stub_prometheus_request_with_ssl_exception(prometheus_invalid_url)
+
+      expect { subject.send(:get, prometheus_invalid_url) }.to raise_error(Gitlab::PrometheusError, "#{prometheus_invalid_url} contains invalid SSL data")
+      expect(req_stub).to have_been_requested
+    end
+  end
+
   describe '#query' do
     let(:prometheus_query) { prometheus_cpu_query('env-slug') }
     let(:query_url) { prometheus_query_url(prometheus_query) }
diff --git a/spec/support/prometheus_helpers.rb b/spec/support/prometheus_helpers.rb
index cc79b11616aca..625a40e1154c8 100644
--- a/spec/support/prometheus_helpers.rb
+++ b/spec/support/prometheus_helpers.rb
@@ -33,6 +33,16 @@ def stub_prometheus_request(url, body: {}, status: 200)
       })
   end
 
+  def stub_prometheus_request_with_socket_exception(url)
+    WebMock.stub_request(:get, url)
+      .to_raise(SocketError)
+  end
+
+  def stub_prometheus_request_with_ssl_exception(url)
+    WebMock.stub_request(:get, url)
+      .to_raise(OpenSSL::SSL::SSLError)
+  end
+
   def stub_all_prometheus_requests(environment_slug, body: nil, status: 200)
     stub_prometheus_request(
       prometheus_query_url(prometheus_memory_query(environment_slug)),
-- 
GitLab


From 64e811957795293b647bda7aef23ffbd92082614 Mon Sep 17 00:00:00 2001
From: Jose Ivan Vargas <jvargas@gitlab.com>
Date: Tue, 2 May 2017 16:53:49 -0500
Subject: [PATCH 243/363] Improved code styling and added a HTTParty rescue
 block

---
 lib/gitlab/prometheus.rb                      |  2 ++
 spec/lib/gitlab/prometheus_spec.rb            | 34 +++++++++++++------
 .../prometheus_service_spec.rb                |  5 ++-
 spec/support/prometheus_helpers.rb            | 10 ++----
 4 files changed, 29 insertions(+), 22 deletions(-)

diff --git a/lib/gitlab/prometheus.rb b/lib/gitlab/prometheus.rb
index 94541009100a4..aa9810c5c4171 100644
--- a/lib/gitlab/prometheus.rb
+++ b/lib/gitlab/prometheus.rb
@@ -54,6 +54,8 @@ def get(url)
       raise PrometheusError, "Can't connect to #{url}"
     rescue OpenSSL::SSL::SSLError
       raise PrometheusError, "#{url} contains invalid SSL data"
+    rescue HTTParty::Error
+      raise PrometheusError, "An error has ocurred"
     end
 
     def handle_response(response)
diff --git a/spec/lib/gitlab/prometheus_spec.rb b/spec/lib/gitlab/prometheus_spec.rb
index d868366951829..8e187df6ca4fc 100644
--- a/spec/lib/gitlab/prometheus_spec.rb
+++ b/spec/lib/gitlab/prometheus_spec.rb
@@ -49,21 +49,33 @@
     end
   end
 
-  describe 'failure to reach a prometheus url' do
-    prometheus_invalid_url = 'https://prometheus.invalid.example.com'
+  describe 'failure to reach a provided prometheus url' do
+    let(:prometheus_url) {"https://prometheus.invalid.example.com"}
 
-    it 'raises a Gitlab::PrometheusError error when a SocketError is rescued' do
-      req_stub = stub_prometheus_request_with_socket_exception(prometheus_invalid_url)
+    context 'exceptions are raised' do
+      it 'raises a Gitlab::PrometheusError error when a SocketError is rescued' do
+        req_stub = stub_prometheus_request_with_exception(prometheus_url, SocketError)
 
-      expect { subject.send(:get, prometheus_invalid_url) }.to raise_error(Gitlab::PrometheusError, "Can't connect to #{prometheus_invalid_url}")
-      expect(req_stub).to have_been_requested
-    end
+        expect { subject.send(:get, prometheus_url) }
+          .to raise_error(Gitlab::PrometheusError, "Can't connect to #{prometheus_url}")
+        expect(req_stub).to have_been_requested
+      end
 
-    it 'raises a Gitlab::PrometheusError error when a SSLError is rescued' do
-      req_stub = stub_prometheus_request_with_ssl_exception(prometheus_invalid_url)
+      it 'raises a Gitlab::PrometheusError error when a SSLError is rescued' do
+        req_stub = stub_prometheus_request_with_exception(prometheus_url, OpenSSL::SSL::SSLError)
 
-      expect { subject.send(:get, prometheus_invalid_url) }.to raise_error(Gitlab::PrometheusError, "#{prometheus_invalid_url} contains invalid SSL data")
-      expect(req_stub).to have_been_requested
+        expect { subject.send(:get, prometheus_url) }
+          .to raise_error(Gitlab::PrometheusError, "#{prometheus_url} contains invalid SSL data")
+        expect(req_stub).to have_been_requested
+      end
+
+      it 'raises a Gitlab::PrometheusError error when a HTTParty::Error is rescued' do
+        req_stub = stub_prometheus_request_with_exception(prometheus_url, HTTParty::Error)
+
+        expect { subject.send(:get, prometheus_url) }
+          .to raise_error(Gitlab::PrometheusError, "An error has ocurred")
+        expect(req_stub).to have_been_requested
+      end
     end
   end
 
diff --git a/spec/models/project_services/prometheus_service_spec.rb b/spec/models/project_services/prometheus_service_spec.rb
index 5ef1b53be1570..f3126bc1e570b 100644
--- a/spec/models/project_services/prometheus_service_spec.rb
+++ b/spec/models/project_services/prometheus_service_spec.rb
@@ -93,12 +93,11 @@
 
     [404, 500].each do |status|
       context "when Prometheus responds with #{status}" do
-        body_response = 'QUERY_FAILED'
         before do
-          stub_all_prometheus_requests(environment.slug, status: status, body: body_response)
+          stub_all_prometheus_requests(environment.slug, status: status, body: "QUERY FAILED!")
         end
 
-        it { is_expected.to eq(success: false, result: %(#{status} - \"#{body_response}\")) }
+        it { is_expected.to eq(success: false, result: %(#{status} - "QUERY FAILED!")) }
       end
     end
   end
diff --git a/spec/support/prometheus_helpers.rb b/spec/support/prometheus_helpers.rb
index 625a40e1154c8..a204365431b1e 100644
--- a/spec/support/prometheus_helpers.rb
+++ b/spec/support/prometheus_helpers.rb
@@ -33,14 +33,8 @@ def stub_prometheus_request(url, body: {}, status: 200)
       })
   end
 
-  def stub_prometheus_request_with_socket_exception(url)
-    WebMock.stub_request(:get, url)
-      .to_raise(SocketError)
-  end
-
-  def stub_prometheus_request_with_ssl_exception(url)
-    WebMock.stub_request(:get, url)
-      .to_raise(OpenSSL::SSL::SSLError)
+  def stub_prometheus_request_with_exception(url, exception_type)
+    WebMock.stub_request(:get, url).to_raise(exception_type)
   end
 
   def stub_all_prometheus_requests(environment_slug, body: nil, status: 200)
-- 
GitLab


From 3f6477fd1f6fc83e5ef8d03e7c2fe960294c8a98 Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Thu, 4 May 2017 14:35:02 -0500
Subject: [PATCH 244/363] Fix karma

---
 spec/javascripts/boards/board_card_spec.js | 6 +++---
 spec/javascripts/boards/board_list_spec.js | 1 +
 spec/javascripts/boards/list_spec.js       | 6 ++++--
 3 files changed, 8 insertions(+), 5 deletions(-)

diff --git a/spec/javascripts/boards/board_card_spec.js b/spec/javascripts/boards/board_card_spec.js
index 2064ca2632bbf..376e706d1db82 100644
--- a/spec/javascripts/boards/board_card_spec.js
+++ b/spec/javascripts/boards/board_card_spec.js
@@ -1,5 +1,5 @@
 /* global List */
-/* global ListUser */
+/* global ListAssignee */
 /* global ListLabel */
 /* global listObj */
 /* global boardsMockInterceptor */
@@ -133,12 +133,12 @@ describe('Issue card', () => {
     });
 
     it('does not set detail issue if img is clicked', (done) => {
-      vm.issue.assignee = new ListUser({
+      vm.issue.assignees = [new ListAssignee({
         id: 1,
         name: 'testing 123',
         username: 'test',
         avatar: 'test_image',
-      });
+      })];
 
       Vue.nextTick(() => {
         triggerEvent('mouseup', vm.$el.querySelector('img'));
diff --git a/spec/javascripts/boards/board_list_spec.js b/spec/javascripts/boards/board_list_spec.js
index 3f598887603af..a89be91166700 100644
--- a/spec/javascripts/boards/board_list_spec.js
+++ b/spec/javascripts/boards/board_list_spec.js
@@ -35,6 +35,7 @@ describe('Board list component', () => {
       iid: 1,
       confidential: false,
       labels: [],
+      assignees: [],
     });
     list.issuesSize = 1;
     list.issues.push(issue);
diff --git a/spec/javascripts/boards/list_spec.js b/spec/javascripts/boards/list_spec.js
index 88a0e6855f4fc..8e3d9fd77a073 100644
--- a/spec/javascripts/boards/list_spec.js
+++ b/spec/javascripts/boards/list_spec.js
@@ -120,7 +120,8 @@ describe('List model', () => {
           title: 'Testing',
           iid: _.random(10000) + i,
           confidential: false,
-          labels: [list.label]
+          labels: [list.label],
+          assignees: [],
         }));
       }
       list.issuesSize = 50;
@@ -138,7 +139,8 @@ describe('List model', () => {
         title: 'Testing',
         iid: _.random(10000),
         confidential: false,
-        labels: [list.label]
+        labels: [list.label],
+        assignees: [],
       }));
       list.issuesSize = 2;
 
-- 
GitLab


From 0c46b3b8ad1bd251125773649138231482397614 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Thu, 4 May 2017 14:20:23 -0600
Subject: [PATCH 245/363] use computed css class for animations

---
 .../issue_show/issue_title_description.vue    | 58 ++++++++++---------
 1 file changed, 30 insertions(+), 28 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index d74d59531ddd7..838dccd6a79eb 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -29,15 +29,22 @@ export default {
       },
     });
 
+    const defaultFlags = {
+      pre: true,
+      pulse: false,
+    };
+
     return {
       poll,
       apiData: {},
+      tasks: '0 of 0',
       title: null,
       titleText: '',
-      tasks: '0 of 0',
+      titleFlag: defaultFlags,
       description: null,
       descriptionText: '',
       descriptionChange: false,
+      descriptionFlag: defaultFlags,
       timeAgoEl: $('.issue_edited_ago'),
       titleEl: document.querySelector('title'),
     };
@@ -51,11 +58,9 @@ export default {
       tasks(this.apiData, this.tasks);
     },
     elementsToVisualize(noTitleChange, noDescriptionChange) {
-      const elementStack = [];
-
       if (!noTitleChange) {
         this.titleText = this.apiData.title_text;
-        elementStack.push(this.$refs['issue-title']);
+        this.titleFlag = { pre: true, pulse: false };
       }
 
       if (!noDescriptionChange) {
@@ -63,35 +68,24 @@ export default {
         this.descriptionChange = true;
         this.updateTaskHTML();
         this.tasks = this.apiData.task_status;
-        elementStack.push(this.$refs['issue-content-container-gfm-entry']);
+        this.descriptionFlag = { pre: true, pulse: false };
       }
 
-      elementStack.forEach((element) => {
-        if (element) {
-          element.classList.remove('issue-realtime-trigger-pulse');
-          element.classList.add('issue-realtime-pre-pulse');
-        }
-      });
-
-      return elementStack;
+      return { noTitleChange, noDescriptionChange };
     },
     setTabTitle() {
       const currentTabTitleScope = this.titleEl.innerText.split('·');
       currentTabTitleScope[0] = `${this.titleText} (#${this.apiData.issue_number}) `;
       this.titleEl.innerText = currentTabTitleScope.join('·');
     },
-    animate(title, description, elementsToVisualize) {
+    animate(title, description) {
       this.title = title;
       this.description = description;
       this.setTabTitle();
 
       this.$nextTick(() => {
-        elementsToVisualize.forEach((element) => {
-          if (element) {
-            element.classList.remove('issue-realtime-pre-pulse');
-            element.classList.add('issue-realtime-trigger-pulse');
-          }
-        });
+        this.titleFlag = { pre: false, pulse: true };
+        this.descriptionFlag = { pre: false, pulse: true };
       });
     },
     triggerAnimation() {
@@ -110,12 +104,8 @@ export default {
       */
       if (noTitleChange && noDescriptionChange) return;
 
-      const elementsToVisualize = this.elementsToVisualize(
-        noTitleChange,
-        noDescriptionChange,
-      );
-
-      this.animate(title, description, elementsToVisualize);
+      this.elementsToVisualize(noTitleChange, noDescriptionChange);
+      this.animate(title, description);
     },
     updateEditedTimeAgo() {
       const toolTipTime = gl.utils.formatDate(this.apiData.updated_at);
@@ -128,6 +118,18 @@ export default {
     descriptionClass() {
       return `description ${this.canUpdateIssue} is-task-list-enabled`;
     },
+    titleAnimationCss() {
+      return {
+        'title issue-realtime-pre-pulse': this.titleFlag.pre,
+        'title issue-realtime-trigger-pulse': this.titleFlag.pulse,
+      };
+    },
+    descriptionAnimationCss() {
+      return {
+        'wiki issue-realtime-pre-pulse': this.descriptionFlag.pre,
+        'wiki issue-realtime-trigger-pulse': this.descriptionFlag.pulse,
+      };
+    },
   },
   created() {
     if (!Visibility.hidden()) {
@@ -166,7 +168,7 @@ export default {
 <template>
   <div>
     <h2
-      class="title issue-realtime-trigger-pulse"
+      :class="titleAnimationCss"
       ref="issue-title"
       v-html="title"
     >
@@ -176,7 +178,7 @@ export default {
       v-if="description"
     >
       <div
-        class="wiki issue-realtime-trigger-pulse"
+        :class="descriptionAnimationCss"
         v-html="description"
         ref="issue-content-container-gfm-entry"
       >
-- 
GitLab


From 6911baa13a082fbf6adf3695f16d859d6758a03c Mon Sep 17 00:00:00 2001
From: Douglas Barbosa Alexandre <dbalexandre@gmail.com>
Date: Thu, 4 May 2017 17:27:46 -0300
Subject: [PATCH 246/363] Fix Import/Export configuration spec

---
 spec/lib/gitlab/import_export/safe_model_attributes.yml | 1 +
 1 file changed, 1 insertion(+)

diff --git a/spec/lib/gitlab/import_export/safe_model_attributes.yml b/spec/lib/gitlab/import_export/safe_model_attributes.yml
index ebfaab4eacd78..59c8b48a2beed 100644
--- a/spec/lib/gitlab/import_export/safe_model_attributes.yml
+++ b/spec/lib/gitlab/import_export/safe_model_attributes.yml
@@ -351,6 +351,7 @@ Project:
 - auto_cancel_pending_pipelines
 - printing_merge_request_link_enabled
 - build_allow_git_fetch
+- last_repository_updated_at
 Author:
 - name
 ProjectFeature:
-- 
GitLab


From 1bf694fcb95c978e8cf32664b3ced186e7a7d850 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Thu, 4 May 2017 20:49:39 +0000
Subject: [PATCH 247/363] Removed clientside_sentry ref in settings.rb

---
 lib/api/v3/settings.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/lib/api/v3/settings.rb b/lib/api/v3/settings.rb
index aa31f01b62cc6..748d6b97d4f94 100644
--- a/lib/api/v3/settings.rb
+++ b/lib/api/v3/settings.rb
@@ -120,7 +120,7 @@ def current_settings
                         :home_page_url, :after_sign_out_path, :sign_in_text, :help_page_text,
                         :shared_runners_enabled, :max_artifacts_size, :max_pages_size, :container_registry_token_expire_delay,
                         :metrics_enabled, :sidekiq_throttling_enabled, :recaptcha_enabled,
-                        :akismet_enabled, :admin_notification_email, :sentry_enabled, :clientside_sentry_enabled,
+                        :akismet_enabled, :admin_notification_email, :sentry_enabled,
                         :repository_storage, :repository_checks_enabled, :koding_enabled, :plantuml_enabled,
                         :version_check_enabled, :email_author_in_body, :html_emails_enabled,
                         :housekeeping_enabled, :terminal_max_session_time
-- 
GitLab


From 5fb98734d5b91ee6eb57b5fbe098e401d490d946 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Thu, 4 May 2017 15:18:59 -0600
Subject: [PATCH 248/363] use more subtle datetime and title change for
 timeagoEl - use SCSS variable for timing on animation

---
 .../javascripts/issue_show/issue_title_description.vue       | 5 +----
 app/assets/stylesheets/pages/issues.scss                     | 2 +-
 2 files changed, 2 insertions(+), 5 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 838dccd6a79eb..e88ab69455b54 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -70,8 +70,6 @@ export default {
         this.tasks = this.apiData.task_status;
         this.descriptionFlag = { pre: true, pulse: false };
       }
-
-      return { noTitleChange, noDescriptionChange };
     },
     setTabTitle() {
       const currentTabTitleScope = this.titleEl.innerText.split('·');
@@ -109,9 +107,8 @@ export default {
     },
     updateEditedTimeAgo() {
       const toolTipTime = gl.utils.formatDate(this.apiData.updated_at);
-
       this.timeAgoEl.attr('datetime', this.apiData.updated_at);
-      this.timeAgoEl.attr('data-original-title', toolTipTime);
+      this.timeAgoEl.attr('title', toolTipTime).tooltip('fixTitle');
     },
   },
   computed: {
diff --git a/app/assets/stylesheets/pages/issues.scss b/app/assets/stylesheets/pages/issues.scss
index f109a01cfa0e3..eda1f96838837 100644
--- a/app/assets/stylesheets/pages/issues.scss
+++ b/app/assets/stylesheets/pages/issues.scss
@@ -23,7 +23,7 @@
 }
 
 .issue-realtime-trigger-pulse {
-  transition: opacity 0.2s ease;
+  transition: opacity $fade-in-duration linear;
   opacity: 1;
 }
 
-- 
GitLab


From f7b7a520920204d1a2864af6d911ef9dc8c2df6b Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 4 May 2017 17:05:14 -0500
Subject: [PATCH 249/363] More updates for translations plus small tweaks.

---
 .../components/stage_issue_component.js       |   2 +-
 .../cycle_analytics_service.js                |   2 +-
 app/assets/javascripts/locale/de/app.js       |   2 +-
 app/assets/javascripts/locale/en/app.js       |   2 +-
 app/assets/javascripts/locale/es/app.js       |   2 +-
 app/views/profiles/show.html.haml             |   1 +
 .../projects/cycle_analytics/show.html.haml   |   2 +-
 config/locales/es.yml                         | 217 ++++++++++++++++++
 db/fixtures/development/17_cycle_analytics.rb |   2 +-
 locale/de/gitlab.po                           |   3 -
 locale/en/gitlab.po                           |   3 -
 locale/es/gitlab.po                           |  31 ++-
 locale/gitlab.pot                             |   7 +-
 13 files changed, 241 insertions(+), 35 deletions(-)
 create mode 100644 config/locales/es.yml

diff --git a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
index c4018c8fc36e5..ad285874643fd 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
@@ -28,7 +28,7 @@ global.cycleAnalytics.StageIssueComponent = Vue.extend({
             <a :href="issue.url" class="issue-link">#{{ issue.iid }}</a>
             &middot;
             <span>
-              {{ __('Opened') }}
+              {{ __('OpenedNDaysAgo|Opened') }}
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
index c176376e8cf89..6504d7db2f2ae 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
@@ -30,7 +30,7 @@ class CycleAnalyticsService {
       startDate,
     } = options;
 
-    return $.get(`${this.requestPath}/events/${stage.name.toLowerCase()}.json`, {
+    return $.get(`${this.requestPath}/events/${stage.name}.json`, {
       cycle_analytics: {
         start_date: startDate,
       },
diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index aee25c3439249..7dfa38dab8c90 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"Opened":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index ed32f7e49c0ea..d6dcf739aadc1 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"Opened":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index 0d8152bb8635e..2bb5145e67b65 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-03 21:03-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":["por"],"Commit":["Cambio","Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"CycleAnalyticsStage|Code":["Código"],"CycleAnalyticsStage|Issue":["Incidencia"],"CycleAnalyticsStage|Plan":["Planificación"],"CycleAnalyticsStage|Production":["Producción"],"CycleAnalyticsStage|Review":["Revisión"],"CycleAnalyticsStage|Staging":["Puesta en escena"],"CycleAnalyticsStage|Test":["Pruebas"],"Deploy":["Despliegue","Despliegues"],"FirstPushedBy|First":["Primer"],"FirstPushedBy|pushed by":["enviado por"],"From issue creation until deploy to production":["Desde la creación de la incidencia hasta el despliegue a producción"],"From merge request merge until deploy to production":["Desde la integración de la solicitud de fusión hasta el despliegue a producción"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Último %d día","Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d evento","Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issue":["Nueva incidencia","Nuevas incidencias"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"Opened":["Abiertos"],"OpenedNDaysAgo|Opened":["Abierto"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Incidencias Relacionadas"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relacionados Originados por Cambios"],"Relative Deployed Builds":["Builds Desplegados Relacionados"],"Relative Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida del desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time before an issue gets scheduled":["Tiempo antes de que una incidencia sea programada"],"Time before an issue starts implementation":["Tiempo antes de que empieze la implementación de una incidencia"],"Time between merge request creation and merge/close":["Tiempo entre la creación de la solicitud de fusión y la integración o cierre de esta"],"Time until first merge request":["Tiempo hasta la primera solicitud de fusión"],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":["Tiempo Total"],"Total test time for all commits/merges":["Tiempo total de pruebas para todos los cambios o integraciones"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pedir acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-04 14:52-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":["por"],"Commit":["Cambio","Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"CycleAnalyticsStage|Code":["Código"],"CycleAnalyticsStage|Issue":["Incidencia"],"CycleAnalyticsStage|Plan":["Planificación"],"CycleAnalyticsStage|Production":["Producción"],"CycleAnalyticsStage|Review":["Revisión"],"CycleAnalyticsStage|Staging":["Puesta en escena"],"CycleAnalyticsStage|Test":["Pruebas"],"Deploy":["Despliegue","Despliegues"],"FirstPushedBy|First":["Primer"],"FirstPushedBy|pushed by":["enviado por"],"From issue creation until deploy to production":["Desde la creación de la incidencia hasta el despliegue a producción"],"From merge request merge until deploy to production":["Desde la integración de la solicitud de fusión hasta el despliegue a producción"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Último %d día","Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d evento","Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issue":["Nueva incidencia","Nuevas incidencias"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OpenedNDaysAgo|Opened":["Abierto"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Incidencias Relacionadas"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relativos Originados por Cambios"],"Relative Deployed Builds":["Builds Desplegados Relativos"],"Relative Merged Requests":["Solicitudes de fusión Relativas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida de desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe el primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time before an issue gets scheduled":["Tiempo antes de que una incidencia sea programada"],"Time before an issue starts implementation":["Tiempo antes de que empieze la implementación de una incidencia"],"Time between merge request creation and merge/close":["Tiempo entre la creación de la solicitud de fusión y la integración o cierre de ésta"],"Time until first merge request":["Tiempo hasta la primera solicitud de fusión"],"Time|hr":["hr","hrs"],"Time|min":["min","mins"],"Time|s":["s"],"Total Time":["Tiempo Total"],"Total test time for all commits/merges":["Tiempo total de pruebas para todos los cambios o integraciones"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pide acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"]}}};
\ No newline at end of file
diff --git a/app/views/profiles/show.html.haml b/app/views/profiles/show.html.haml
index 6e14be62e32e3..4a1438aa68e92 100644
--- a/app/views/profiles/show.html.haml
+++ b/app/views/profiles/show.html.haml
@@ -76,6 +76,7 @@
         = f.label :preferred_language, class: "label-light"
         = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES.map { |value, label| [label, value] },
           {}, class: "select2"
+        %span.help-block This feature is experimental and translations are not complete yet.
       .form-group
         = f.label :skype, class: "label-light"
         = f.text_field :skype, class: "form-control"
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index eadaff70e0469..819f29d3ca594 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -31,7 +31,7 @@
           .row
             .col-sm-3.col-xs-12.column{ "v-for" => "item in state.summary" }
               %h3.header {{ item.value }}
-              %p.text {{ __(item.title) }}
+              %p.text {{ item.title }}
             .col-sm-3.col-xs-12.column
               .dropdown.inline.js-ca-dropdown
                 %button.dropdown-menu-toggle{ "data-toggle" => "dropdown", :type => "button" }
diff --git a/config/locales/es.yml b/config/locales/es.yml
new file mode 100644
index 0000000000000..87e79beee7460
--- /dev/null
+++ b/config/locales/es.yml
@@ -0,0 +1,217 @@
+---
+es:
+  activerecord:
+    errors:
+      messages:
+        record_invalid: "La validación falló: %{errors}"
+        restrict_dependent_destroy:
+          has_one: No se puede eliminar el registro porque existe un %{record} dependiente
+          has_many: No se puede eliminar el registro porque existen %{record} dependientes
+  date:
+    abbr_day_names:
+    - dom
+    - lun
+    - mar
+    - mié
+    - jue
+    - vie
+    - sáb
+    abbr_month_names:
+    -
+    - ene
+    - feb
+    - mar
+    - abr
+    - may
+    - jun
+    - jul
+    - ago
+    - sep
+    - oct
+    - nov
+    - dic
+    day_names:
+    - domingo
+    - lunes
+    - martes
+    - miércoles
+    - jueves
+    - viernes
+    - sábado
+    formats:
+      default: "%d/%m/%Y"
+      long: "%d de %B de %Y"
+      short: "%d de %b"
+    month_names:
+    -
+    - enero
+    - febrero
+    - marzo
+    - abril
+    - mayo
+    - junio
+    - julio
+    - agosto
+    - septiembre
+    - octubre
+    - noviembre
+    - diciembre
+    order:
+    - :day
+    - :month
+    - :year
+  datetime:
+    distance_in_words:
+      about_x_hours:
+        one: alrededor de 1 hora
+        other: alrededor de %{count} horas
+      about_x_months:
+        one: alrededor de 1 mes
+        other: alrededor de %{count} meses
+      about_x_years:
+        one: alrededor de 1 año
+        other: alrededor de %{count} años
+      almost_x_years:
+        one: casi 1 año
+        other: casi %{count} años
+      half_a_minute: medio minuto
+      less_than_x_minutes:
+        one: menos de 1 minuto
+        other: menos de %{count} minutos
+      less_than_x_seconds:
+        one: menos de 1 segundo
+        other: menos de %{count} segundos
+      over_x_years:
+        one: más de 1 año
+        other: más de %{count} años
+      x_days:
+        one: 1 día
+        other: "%{count} días"
+      x_minutes:
+        one: 1 minuto
+        other: "%{count} minutos"
+      x_months:
+        one: 1 mes
+        other: "%{count} meses"
+      x_years:
+        one: 1 año
+        other: "%{count} años"
+      x_seconds:
+        one: 1 segundo
+        other: "%{count} segundos"
+    prompts:
+      day: Día
+      hour: Hora
+      minute: Minutos
+      month: Mes
+      second: Segundos
+      year: Año
+  errors:
+    format: "%{attribute} %{message}"
+    messages:
+      accepted: debe ser aceptado
+      blank: no puede estar en blanco
+      present: debe estar en blanco
+      confirmation: no coincide
+      empty: no puede estar vacío
+      equal_to: debe ser igual a %{count}
+      even: debe ser par
+      exclusion: está reservado
+      greater_than: debe ser mayor que %{count}
+      greater_than_or_equal_to: debe ser mayor que o igual a %{count}
+      inclusion: no está incluido en la lista
+      invalid: no es válido
+      less_than: debe ser menor que %{count}
+      less_than_or_equal_to: debe ser menor que o igual a %{count}
+      model_invalid: "La validación falló: %{errors}"
+      not_a_number: no es un número
+      not_an_integer: debe ser un entero
+      odd: debe ser impar
+      required: debe existir
+      taken: ya está en uso
+      too_long:
+        one: "es demasiado largo (1 carácter máximo)"
+        other: "es demasiado largo (%{count} caracteres máximo)"
+      too_short:
+        one: "es demasiado corto (1 carácter mínimo)"
+        other: "es demasiado corto (%{count} caracteres mínimo)"
+      wrong_length:
+        one: "no tiene la longitud correcta (1 carácter exactos)"
+        other: "no tiene la longitud correcta (%{count} caracteres exactos)"
+      other_than: debe ser distinto de %{count}
+    template:
+      body: 'Se encontraron problemas con los siguientes campos:'
+      header:
+        one: No se pudo guardar este/a %{model} porque se encontró 1 error
+        other: No se pudo guardar este/a %{model} porque se encontraron %{count} errores
+  helpers:
+    select:
+      prompt: Por favor seleccione
+    submit:
+      create: Crear %{model}
+      submit: Guardar %{model}
+      update: Actualizar %{model}
+  number:
+    currency:
+      format:
+        delimiter: "."
+        format: "%n %u"
+        precision: 2
+        separator: ","
+        significant: false
+        strip_insignificant_zeros: false
+        unit: "€"
+    format:
+      delimiter: "."
+      precision: 3
+      separator: ","
+      significant: false
+      strip_insignificant_zeros: false
+    human:
+      decimal_units:
+        format: "%n %u"
+        units:
+          billion: mil millones
+          million:
+            one: millón
+            other: millones
+          quadrillion: mil billones
+          thousand: mil
+          trillion:
+            one: billón
+            other: billones
+          unit: ''
+      format:
+        delimiter: ''
+        precision: 1
+        significant: true
+        strip_insignificant_zeros: true
+      storage_units:
+        format: "%n %u"
+        units:
+          byte:
+            one: Byte
+            other: Bytes
+          gb: GB
+          kb: KB
+          mb: MB
+          tb: TB
+    percentage:
+      format:
+        delimiter: ''
+        format: "%n %"
+    precision:
+      format:
+        delimiter: ''
+  support:
+    array:
+      last_word_connector: " y "
+      two_words_connector: " y "
+      words_connector: ", "
+  time:
+    am: am
+    formats:
+      default: "%A, %d de %B de %Y %H:%M:%S %z"
+      long: "%d de %B de %Y %H:%M"
+      short: "%d de %b %H:%M"
+    pm: pm
diff --git a/db/fixtures/development/17_cycle_analytics.rb b/db/fixtures/development/17_cycle_analytics.rb
index 040505ad52c54..0d7eb1a7c9336 100644
--- a/db/fixtures/development/17_cycle_analytics.rb
+++ b/db/fixtures/development/17_cycle_analytics.rb
@@ -227,7 +227,7 @@ def deploy_to_production(merge_requests)
 
   if ENV[flag]
     Project.all.each do |project|
-      seeder = Gitlab::Seeder::CycleAnalytics.new(pro)
+      seeder = Gitlab::Seeder::CycleAnalytics.new(project)
       seeder.seed!
     end
   elsif ENV['CYCLE_ANALYTICS_PERF_TEST']
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index 3e7bcf2c90fb5..5a13000434ea5 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -93,9 +93,6 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
-msgid "Opened"
-msgstr ""
-
 msgid "OpenedNDaysAgo|Opened"
 msgstr ""
 
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 9f75b5641a4ae..93ecefec762e9 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -93,9 +93,6 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
-msgid "Opened"
-msgstr ""
-
 msgid "OpenedNDaysAgo|Opened"
 msgstr ""
 
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 6806bf216774e..2074912c1477a 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-05-03 21:03-0500\n"
+"PO-Revision-Date: 2017-05-04 14:52-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -94,9 +94,6 @@ msgstr "No disponible"
 msgid "Not enough data"
 msgstr "No hay suficientes datos"
 
-msgid "Opened"
-msgstr "Abiertos"
-
 msgid "OpenedNDaysAgo|Opened"
 msgstr "Abierto"
 
@@ -119,13 +116,13 @@ msgid "Related Merge Requests"
 msgstr "Solicitudes de fusión Relacionadas"
 
 msgid "Relative Builds Trigger by Commits"
-msgstr "Builds Relacionados Originados por Cambios"
+msgstr "Builds Relativos Originados por Cambios"
 
 msgid "Relative Deployed Builds"
-msgstr "Builds Desplegados Relacionados"
+msgstr "Builds Desplegados Relativos"
 
 msgid "Relative Merged Requests"
-msgstr "Solicitudes de fusión Relacionadas"
+msgstr "Solicitudes de fusión Relativas"
 
 msgid "Showing %d event"
 msgid_plural "Showing %d events"
@@ -142,10 +139,10 @@ msgid "The issue stage shows the time it takes from creating an issue to assigni
 msgstr "La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."
 
 msgid "The phase of the development lifecycle."
-msgstr "La etapa del ciclo de vida del desarrollo."
+msgstr "La etapa del ciclo de vida de desarrollo."
 
 msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
-msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe su primer cambio."
+msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe el primer cambio."
 
 msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
 msgstr "La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."
@@ -154,7 +151,7 @@ msgid "The review stage shows the time from creating the merge request to mergin
 msgstr "La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."
 
 msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
-msgstr "La etapa de puesta en escena muestra el tiempo entre la fusión  y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."
+msgstr "La etapa de puesta en escena muestra el tiempo entre la fusión y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."
 
 msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
 msgstr "La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."
@@ -172,23 +169,23 @@ msgid "Time before an issue starts implementation"
 msgstr "Tiempo antes de que empieze la implementación de una incidencia"
 
 msgid "Time between merge request creation and merge/close"
-msgstr "Tiempo entre la creación de la solicitud de fusión y la integración o cierre de esta"
+msgstr "Tiempo entre la creación de la solicitud de fusión y la integración o cierre de ésta"
 
 msgid "Time until first merge request"
 msgstr "Tiempo hasta la primera solicitud de fusión"
 
 msgid "Time|hr"
 msgid_plural "Time|hrs"
-msgstr[0] ""
-msgstr[1] ""
+msgstr[0] "hr"
+msgstr[1] "hrs"
 
 msgid "Time|min"
 msgid_plural "Time|mins"
-msgstr[0] ""
-msgstr[1] ""
+msgstr[0] "min"
+msgstr[1] "mins"
 
 msgid "Time|s"
-msgstr ""
+msgstr "s"
 
 msgid "Total Time"
 msgstr "Tiempo Total"
@@ -197,7 +194,7 @@ msgid "Total test time for all commits/merges"
 msgstr "Tiempo total de pruebas para todos los cambios o integraciones"
 
 msgid "Want to see the data? Please ask an administrator for access."
-msgstr "¿Quieres ver los datos? Por favor pedir acceso al administrador."
+msgstr "¿Quieres ver los datos? Por favor pide acceso al administrador."
 
 msgid "We don't have enough data to show this stage."
 msgstr "No hay suficientes datos para mostrar en esta etapa."
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index e1796d519351f..4473f5715a2f9 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-05-03 20:53-0500\n"
-"PO-Revision-Date: 2017-05-03 20:53-0500\n"
+"POT-Creation-Date: 2017-05-04 14:55-0500\n"
+"PO-Revision-Date: 2017-05-04 14:55-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -94,9 +94,6 @@ msgstr ""
 msgid "Not enough data"
 msgstr ""
 
-msgid "Opened"
-msgstr ""
-
 msgid "OpenedNDaysAgo|Opened"
 msgstr ""
 
-- 
GitLab


From 024427bff6536457e1365709bd15098ddab0e0e8 Mon Sep 17 00:00:00 2001
From: tauriedavis <taurie@gitlab.com>
Date: Thu, 4 May 2017 15:17:16 -0700
Subject: [PATCH 250/363] 31689 Add default margin-top to user request table on
 project members page

---
 app/views/projects/group_links/_index.html.haml        | 4 +---
 app/views/projects/project_members/_index.html.haml    | 2 +-
 app/views/shared/members/_requests.html.haml           | 2 +-
 changelogs/unreleased/31689-request-access-spacing.yml | 4 ++++
 4 files changed, 7 insertions(+), 5 deletions(-)
 create mode 100644 changelogs/unreleased/31689-request-access-spacing.yml

diff --git a/app/views/projects/group_links/_index.html.haml b/app/views/projects/group_links/_index.html.haml
index b6116dbec414a..debb0214d068c 100644
--- a/app/views/projects/group_links/_index.html.haml
+++ b/app/views/projects/group_links/_index.html.haml
@@ -6,11 +6,9 @@
     %p
       Projects can be stored in only one group at once. However you can share a project with other groups here.
   .col-lg-9
-    %h5.prepend-top-0
-      Set a group to share
     = form_tag namespace_project_group_links_path(@project.namespace, @project), class: 'js-requires-input', method: :post do
       .form-group
-        = label_tag :link_group_id, "Group", class: "label-light"
+        = label_tag :link_group_id, "Select a group to share with", class: "label-light"
         = groups_select_tag(:link_group_id, data: { skip_groups: @skip_groups }, required: true)
       .form-group
         = label_tag :link_group_access, "Max access level", class: "label-light"
diff --git a/app/views/projects/project_members/_index.html.haml b/app/views/projects/project_members/_index.html.haml
index f83521052ed58..d080b6c83d472 100644
--- a/app/views/projects/project_members/_index.html.haml
+++ b/app/views/projects/project_members/_index.html.haml
@@ -18,7 +18,7 @@
         = render "projects/project_members/new_project_member"
 
         = render 'shared/members/requests', membership_source: @project, requesters: @requesters
-        .append-bottom-default.clearfix
+        .clearfix
           %h5.member.existing-title
             Existing members and groups
         - if @group_links.any?
diff --git a/app/views/shared/members/_requests.html.haml b/app/views/shared/members/_requests.html.haml
index 10050adfda5a4..92f6e7428ae2e 100644
--- a/app/views/shared/members/_requests.html.haml
+++ b/app/views/shared/members/_requests.html.haml
@@ -1,5 +1,5 @@
 - if requesters.any?
-  .panel.panel-default
+  .panel.panel-default.prepend-top-default
     .panel-heading
       Users requesting access to
       %strong= membership_source.name
diff --git a/changelogs/unreleased/31689-request-access-spacing.yml b/changelogs/unreleased/31689-request-access-spacing.yml
new file mode 100644
index 0000000000000..66076b44f466a
--- /dev/null
+++ b/changelogs/unreleased/31689-request-access-spacing.yml
@@ -0,0 +1,4 @@
+---
+title: Add default margin-top to user request table on project members page
+merge_request:
+author:
-- 
GitLab


From 46d7f0a3a1c5587cca53a930b5d0610db2e59f89 Mon Sep 17 00:00:00 2001
From: tauriedavis <taurie@gitlab.com>
Date: Thu, 4 May 2017 15:36:43 -0700
Subject: [PATCH 251/363] 30007 Add transparent top-border to the hover state
 of done todos

---
 app/assets/stylesheets/pages/todos.scss               | 3 ++-
 changelogs/unreleased/30007-done-todo-hover-state.yml | 4 ++++
 2 files changed, 6 insertions(+), 1 deletion(-)
 create mode 100644 changelogs/unreleased/30007-done-todo-hover-state.yml

diff --git a/app/assets/stylesheets/pages/todos.scss b/app/assets/stylesheets/pages/todos.scss
index a39815319f377..de652a7936988 100644
--- a/app/assets/stylesheets/pages/todos.scss
+++ b/app/assets/stylesheets/pages/todos.scss
@@ -54,8 +54,9 @@
     background-color: $white-light;
 
     &:hover {
-      border-color: $white-dark;
+      border-color: $white-normal;
       background-color: $gray-light;
+      border-top: 1px solid transparent;
 
       .todo-avatar,
       .todo-item {
diff --git a/changelogs/unreleased/30007-done-todo-hover-state.yml b/changelogs/unreleased/30007-done-todo-hover-state.yml
new file mode 100644
index 0000000000000..bfbde7a49c825
--- /dev/null
+++ b/changelogs/unreleased/30007-done-todo-hover-state.yml
@@ -0,0 +1,4 @@
+---
+title: Add transparent top-border to the hover state of done todos
+merge_request:
+author:
-- 
GitLab


From 897f98cc9df483ce00a91a273195da1da52ed3da Mon Sep 17 00:00:00 2001
From: Douwe Maan <douwe@selenight.nl>
Date: Thu, 4 May 2017 18:12:49 -0500
Subject: [PATCH 252/363] Fix wording of Cycle Analytics stage legends

---
 lib/gitlab/cycle_analytics/review_stage.rb  | 2 +-
 lib/gitlab/cycle_analytics/staging_stage.rb | 2 +-
 lib/gitlab/cycle_analytics/test_stage.rb    | 2 +-
 3 files changed, 3 insertions(+), 3 deletions(-)

diff --git a/lib/gitlab/cycle_analytics/review_stage.rb b/lib/gitlab/cycle_analytics/review_stage.rb
index 4744be834de27..80e5b538c6cec 100644
--- a/lib/gitlab/cycle_analytics/review_stage.rb
+++ b/lib/gitlab/cycle_analytics/review_stage.rb
@@ -14,7 +14,7 @@ def name
       end
 
       def legend
-        "Relative Merged Requests"
+        "Related Merged Requests"
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/staging_stage.rb b/lib/gitlab/cycle_analytics/staging_stage.rb
index 3cdbe04fbaf1d..eae0e8ee7b067 100644
--- a/lib/gitlab/cycle_analytics/staging_stage.rb
+++ b/lib/gitlab/cycle_analytics/staging_stage.rb
@@ -15,7 +15,7 @@ def name
       end
 
       def legend
-        "Relative Deployed Builds"
+        "Related Deployed Jobs"
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/test_stage.rb b/lib/gitlab/cycle_analytics/test_stage.rb
index e96943833bc46..de9a36d34f176 100644
--- a/lib/gitlab/cycle_analytics/test_stage.rb
+++ b/lib/gitlab/cycle_analytics/test_stage.rb
@@ -14,7 +14,7 @@ def name
       end
 
       def legend
-        "Relative Builds Trigger by Commits"
+        "Related Jobs"
       end
 
       def description
-- 
GitLab


From 6eb9e981c61969a904ccbae2c5f52f562b02d199 Mon Sep 17 00:00:00 2001
From: Jose Ivan Vargas <jvargas@gitlab.com>
Date: Thu, 4 May 2017 09:16:24 -0500
Subject: [PATCH 253/363] Improved changelog entry, also changed error message
 for HTTParty error

---
 .../unreleased/prometheus-integration-test-setting-fix.yml      | 2 +-
 lib/gitlab/prometheus.rb                                        | 2 +-
 spec/lib/gitlab/prometheus_spec.rb                              | 2 +-
 3 files changed, 3 insertions(+), 3 deletions(-)

diff --git a/changelogs/unreleased/prometheus-integration-test-setting-fix.yml b/changelogs/unreleased/prometheus-integration-test-setting-fix.yml
index c65c682a11442..45b7c2263e694 100644
--- a/changelogs/unreleased/prometheus-integration-test-setting-fix.yml
+++ b/changelogs/unreleased/prometheus-integration-test-setting-fix.yml
@@ -1,4 +1,4 @@
 ---
-title: Added rescue block for the test method
+title: Prevent 500 errors caused by testing the Prometheus service
 merge_request: 10994
 author:
diff --git a/lib/gitlab/prometheus.rb b/lib/gitlab/prometheus.rb
index aa9810c5c4171..8827507955dd7 100644
--- a/lib/gitlab/prometheus.rb
+++ b/lib/gitlab/prometheus.rb
@@ -55,7 +55,7 @@ def get(url)
     rescue OpenSSL::SSL::SSLError
       raise PrometheusError, "#{url} contains invalid SSL data"
     rescue HTTParty::Error
-      raise PrometheusError, "An error has ocurred"
+      raise PrometheusError, "Network connection error"
     end
 
     def handle_response(response)
diff --git a/spec/lib/gitlab/prometheus_spec.rb b/spec/lib/gitlab/prometheus_spec.rb
index 8e187df6ca4fc..fc453a2704b36 100644
--- a/spec/lib/gitlab/prometheus_spec.rb
+++ b/spec/lib/gitlab/prometheus_spec.rb
@@ -73,7 +73,7 @@
         req_stub = stub_prometheus_request_with_exception(prometheus_url, HTTParty::Error)
 
         expect { subject.send(:get, prometheus_url) }
-          .to raise_error(Gitlab::PrometheusError, "An error has ocurred")
+          .to raise_error(Gitlab::PrometheusError, "Network connection error")
         expect(req_stub).to have_been_requested
       end
     end
-- 
GitLab


From b396668ec55fb866ef578e5f0e60df8b13519613 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 4 May 2017 18:44:19 -0500
Subject: [PATCH 254/363] Add CHANGELOG entry, some specs and locale file for
 German.

---
 .../unreleased/implement-i18n-support.yml     |   4 +
 config/locales/de.yml                         | 219 ++++++++++++++++++
 lib/gitlab/i18n.rb                            |   7 +-
 spec/features/cycle_analytics_spec.rb         |  19 ++
 spec/lib/gitlab/i18n_spec.rb                  |  27 +++
 spec/models/user_spec.rb                      |   8 +
 6 files changed, 281 insertions(+), 3 deletions(-)
 create mode 100644 changelogs/unreleased/implement-i18n-support.yml
 create mode 100644 config/locales/de.yml
 create mode 100644 spec/lib/gitlab/i18n_spec.rb

diff --git a/changelogs/unreleased/implement-i18n-support.yml b/changelogs/unreleased/implement-i18n-support.yml
new file mode 100644
index 0000000000000..d304fbecf9088
--- /dev/null
+++ b/changelogs/unreleased/implement-i18n-support.yml
@@ -0,0 +1,4 @@
+---
+title: Add support for i18n on Cycle Analytics page
+merge_request: 10669
+author:
diff --git a/config/locales/de.yml b/config/locales/de.yml
new file mode 100644
index 0000000000000..533663a270495
--- /dev/null
+++ b/config/locales/de.yml
@@ -0,0 +1,219 @@
+---
+de:
+  activerecord:
+    errors:
+      messages:
+        record_invalid: 'Gültigkeitsprüfung ist fehlgeschlagen: %{errors}'
+        restrict_dependent_destroy:
+          has_one: Datensatz kann nicht gelöscht werden, da ein abhängiger %{record}-Datensatz
+            existiert.
+          has_many: Datensatz kann nicht gelöscht werden, da abhängige %{record} existieren.
+  date:
+    abbr_day_names:
+    - So
+    - Mo
+    - Di
+    - Mi
+    - Do
+    - Fr
+    - Sa
+    abbr_month_names:
+    -
+    - Jan
+    - Feb
+    - Mär
+    - Apr
+    - Mai
+    - Jun
+    - Jul
+    - Aug
+    - Sep
+    - Okt
+    - Nov
+    - Dez
+    day_names:
+    - Sonntag
+    - Montag
+    - Dienstag
+    - Mittwoch
+    - Donnerstag
+    - Freitag
+    - Samstag
+    formats:
+      default: "%d.%m.%Y"
+      long: "%e. %B %Y"
+      short: "%e. %b"
+    month_names:
+    -
+    - Januar
+    - Februar
+    - März
+    - April
+    - Mai
+    - Juni
+    - Juli
+    - August
+    - September
+    - Oktober
+    - November
+    - Dezember
+    order:
+    - :day
+    - :month
+    - :year
+  datetime:
+    distance_in_words:
+      about_x_hours:
+        one: etwa eine Stunde
+        other: etwa %{count} Stunden
+      about_x_months:
+        one: etwa ein Monat
+        other: etwa %{count} Monate
+      about_x_years:
+        one: etwa ein Jahr
+        other: etwa %{count} Jahre
+      almost_x_years:
+        one: fast ein Jahr
+        other: fast %{count} Jahre
+      half_a_minute: eine halbe Minute
+      less_than_x_minutes:
+        one: weniger als eine Minute
+        other: weniger als %{count} Minuten
+      less_than_x_seconds:
+        one: weniger als eine Sekunde
+        other: weniger als %{count} Sekunden
+      over_x_years:
+        one: mehr als ein Jahr
+        other: mehr als %{count} Jahre
+      x_days:
+        one: ein Tag
+        other: "%{count} Tage"
+      x_minutes:
+        one: eine Minute
+        other: "%{count} Minuten"
+      x_months:
+        one: ein Monat
+        other: "%{count} Monate"
+      x_seconds:
+        one: eine Sekunde
+        other: "%{count} Sekunden"
+    prompts:
+      day: Tag
+      hour: Stunden
+      minute: Minute
+      month: Monat
+      second: Sekunde
+      year: Jahr
+  errors:
+    format: "%{attribute} %{message}"
+    messages:
+      accepted: muss akzeptiert werden
+      blank: muss ausgefüllt werden
+      present: darf nicht ausgefüllt werden
+      confirmation: stimmt nicht mit %{attribute} überein
+      empty: muss ausgefüllt werden
+      equal_to: muss genau %{count} sein
+      even: muss gerade sein
+      exclusion: ist nicht verfügbar
+      greater_than: muss größer als %{count} sein
+      greater_than_or_equal_to: muss größer oder gleich %{count} sein
+      inclusion: ist kein gültiger Wert
+      invalid: ist nicht gültig
+      less_than: muss kleiner als %{count} sein
+      less_than_or_equal_to: muss kleiner oder gleich %{count} sein
+      model_invalid: 'Gültigkeitsprüfung ist fehlgeschlagen: %{errors}'
+      not_a_number: ist keine Zahl
+      not_an_integer: muss ganzzahlig sein
+      odd: muss ungerade sein
+      required: muss ausgefüllt werden
+      taken: ist bereits vergeben
+      too_long:
+        one: ist zu lang (mehr als 1 Zeichen)
+        other: ist zu lang (mehr als %{count} Zeichen)
+      too_short:
+        one: ist zu kurz (weniger als 1 Zeichen)
+        other: ist zu kurz (weniger als %{count} Zeichen)
+      wrong_length:
+        one: hat die falsche Länge (muss genau 1 Zeichen haben)
+        other: hat die falsche Länge (muss genau %{count} Zeichen haben)
+      other_than: darf nicht gleich %{count} sein
+    template:
+      body: 'Bitte überprüfen Sie die folgenden Felder:'
+      header:
+        one: 'Konnte %{model} nicht speichern: ein Fehler.'
+        other: 'Konnte %{model} nicht speichern: %{count} Fehler.'
+  helpers:
+    select:
+      prompt: Bitte wählen
+    submit:
+      create: "%{model} erstellen"
+      submit: "%{model} speichern"
+      update: "%{model} aktualisieren"
+  number:
+    currency:
+      format:
+        delimiter: "."
+        format: "%n %u"
+        precision: 2
+        separator: ","
+        significant: false
+        strip_insignificant_zeros: false
+        unit: "€"
+    format:
+      delimiter: "."
+      precision: 2
+      separator: ","
+      significant: false
+      strip_insignificant_zeros: false
+    human:
+      decimal_units:
+        format: "%n %u"
+        units:
+          billion:
+            one: Milliarde
+            other: Milliarden
+          million:
+            one: Million
+            other: Millionen
+          quadrillion:
+            one: Billiarde
+            other: Billiarden
+          thousand: Tausend
+          trillion:
+            one: Billion
+            other: Billionen
+          unit: ''
+      format:
+        delimiter: ''
+        precision: 3
+        significant: true
+        strip_insignificant_zeros: true
+      storage_units:
+        format: "%n %u"
+        units:
+          byte:
+            one: Byte
+            other: Bytes
+          gb: GB
+          kb: KB
+          mb: MB
+          tb: TB
+    percentage:
+      format:
+        delimiter: ''
+        format: "%n %"
+    precision:
+      format:
+        delimiter: ''
+  support:
+    array:
+      last_word_connector: " und "
+      two_words_connector: " und "
+      words_connector: ", "
+  time:
+    am: vormittags
+    formats:
+      default: "%A, %d. %B %Y, %H:%M Uhr"
+      long: "%A, %d. %B %Y, %H:%M Uhr"
+      short: "%d. %B, %H:%M Uhr"
+    pm: nachmittags
diff --git a/lib/gitlab/i18n.rb b/lib/gitlab/i18n.rb
index 9081ced0238ee..3411516319f88 100644
--- a/lib/gitlab/i18n.rb
+++ b/lib/gitlab/i18n.rb
@@ -3,9 +3,9 @@ module I18n
     extend self
 
     AVAILABLE_LANGUAGES = {
-      en: 'English',
-      es: 'Español',
-      de: 'Deutsch'
+      'en' => 'English',
+      'es' => 'Español',
+      'de' => 'Deutsch'
     }.freeze
 
     def available_locales
@@ -19,6 +19,7 @@ def set_locale(current_user)
     end
 
     def reset_locale
+      FastGettext.set_locale(::I18n.default_locale)
       ::I18n.locale = ::I18n.default_locale
     end
   end
diff --git a/spec/features/cycle_analytics_spec.rb b/spec/features/cycle_analytics_spec.rb
index b93275c330bbc..7c9d522273bea 100644
--- a/spec/features/cycle_analytics_spec.rb
+++ b/spec/features/cycle_analytics_spec.rb
@@ -62,6 +62,25 @@
         expect_issue_to_be_present
       end
     end
+
+    context "when my preferred language is Spanish" do
+      before do
+        user.update_attribute(:preferred_language, 'es')
+
+        project.team << [user, :master]
+        login_as(user)
+        visit namespace_project_cycle_analytics_path(project.namespace, project)
+        wait_for_ajax
+      end
+
+      it 'shows the content in Spanish' do
+        expect(page).to have_content('Estado del Pipeline')
+      end
+
+      it 'resets the language to English' do
+        expect(I18n.locale).to eq(:en)
+      end
+    end
   end
 
   context "as a guest" do
diff --git a/spec/lib/gitlab/i18n_spec.rb b/spec/lib/gitlab/i18n_spec.rb
new file mode 100644
index 0000000000000..52f2614d5cab6
--- /dev/null
+++ b/spec/lib/gitlab/i18n_spec.rb
@@ -0,0 +1,27 @@
+require 'spec_helper'
+
+module Gitlab
+  describe I18n, lib: true do
+    let(:user) { create(:user, preferred_language: 'es') }
+
+    describe '.set_locale' do
+      it 'sets the locale based on current user preferred language' do
+        Gitlab::I18n.set_locale(user)
+
+        expect(FastGettext.locale).to eq('es')
+        expect(::I18n.locale).to eq(:es)
+      end
+    end
+
+    describe '.reset_locale' do
+      it 'resets the locale to the default language' do
+        Gitlab::I18n.set_locale(user)
+
+        Gitlab::I18n.reset_locale
+
+        expect(FastGettext.locale).to eq('en')
+        expect(::I18n.locale).to eq(:en)
+      end
+    end
+  end
+end
diff --git a/spec/models/user_spec.rb b/spec/models/user_spec.rb
index 1c2df4c9d9741..1317917495649 100644
--- a/spec/models/user_spec.rb
+++ b/spec/models/user_spec.rb
@@ -1663,4 +1663,12 @@ def add_user(access)
       expect(User.active.count).to eq(1)
     end
   end
+
+  describe 'preferred language' do
+    it 'is English by default' do
+      user = create(:user)
+
+      expect(user.preferred_language).to eq('en')
+    end
+  end
 end
-- 
GitLab


From 7f24b87b3beb142cadc7644db3ab4c49fdd04839 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Thu, 4 May 2017 19:27:17 -0500
Subject: [PATCH 255/363] Some small updates for Spanish translations.

---
 app/assets/javascripts/locale/de/app.js     |  2 +-
 app/assets/javascripts/locale/en/app.js     |  2 +-
 app/assets/javascripts/locale/es/app.js     |  2 +-
 lib/gitlab/cycle_analytics/review_stage.rb  |  2 +-
 lib/gitlab/cycle_analytics/staging_stage.rb |  2 +-
 lib/gitlab/cycle_analytics/test_stage.rb    |  2 +-
 locale/de/gitlab.po                         | 10 +++++-----
 locale/en/gitlab.po                         | 10 +++++-----
 locale/es/gitlab.po                         | 18 +++++++++---------
 locale/gitlab.pot                           | 14 +++++++-------
 10 files changed, 32 insertions(+), 32 deletions(-)

diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
index 7dfa38dab8c90..e96090da80e41 100644
--- a/app/assets/javascripts/locale/de/app.js
+++ b/app/assets/javascripts/locale/de/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Deployed Jobs":[""],"Related Issues":[""],"Related Jobs":[""],"Related Merge Requests":[""],"Related Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
index d6dcf739aadc1..ade9b667b3c0f 100644
--- a/app/assets/javascripts/locale/en/app.js
+++ b/app/assets/javascripts/locale/en/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Issues":[""],"Related Merge Requests":[""],"Relative Builds Trigger by Commits":[""],"Relative Deployed Builds":[""],"Relative Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Deployed Jobs":[""],"Related Issues":[""],"Related Jobs":[""],"Related Merge Requests":[""],"Related Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
index 2bb5145e67b65..3dafa21f2359a 100644
--- a/app/assets/javascripts/locale/es/app.js
+++ b/app/assets/javascripts/locale/es/app.js
@@ -1 +1 @@
-var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-04 14:52-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":["por"],"Commit":["Cambio","Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"CycleAnalyticsStage|Code":["Código"],"CycleAnalyticsStage|Issue":["Incidencia"],"CycleAnalyticsStage|Plan":["Planificación"],"CycleAnalyticsStage|Production":["Producción"],"CycleAnalyticsStage|Review":["Revisión"],"CycleAnalyticsStage|Staging":["Puesta en escena"],"CycleAnalyticsStage|Test":["Pruebas"],"Deploy":["Despliegue","Despliegues"],"FirstPushedBy|First":["Primer"],"FirstPushedBy|pushed by":["enviado por"],"From issue creation until deploy to production":["Desde la creación de la incidencia hasta el despliegue a producción"],"From merge request merge until deploy to production":["Desde la integración de la solicitud de fusión hasta el despliegue a producción"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Último %d día","Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d evento","Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issue":["Nueva incidencia","Nuevas incidencias"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OpenedNDaysAgo|Opened":["Abierto"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Issues":["Incidencias Relacionadas"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Relative Builds Trigger by Commits":["Builds Relativos Originados por Cambios"],"Relative Deployed Builds":["Builds Desplegados Relativos"],"Relative Merged Requests":["Solicitudes de fusión Relativas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida de desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe el primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time before an issue gets scheduled":["Tiempo antes de que una incidencia sea programada"],"Time before an issue starts implementation":["Tiempo antes de que empieze la implementación de una incidencia"],"Time between merge request creation and merge/close":["Tiempo entre la creación de la solicitud de fusión y la integración o cierre de ésta"],"Time until first merge request":["Tiempo hasta la primera solicitud de fusión"],"Time|hr":["hr","hrs"],"Time|min":["min","mins"],"Time|s":["s"],"Total Time":["Tiempo Total"],"Total test time for all commits/merges":["Tiempo total de pruebas para todos los cambios o integraciones"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pide acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"]}}};
\ No newline at end of file
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-04 19:24-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":["por"],"Commit":["Cambio","Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"CycleAnalyticsStage|Code":["Código"],"CycleAnalyticsStage|Issue":["Incidencia"],"CycleAnalyticsStage|Plan":["Planificación"],"CycleAnalyticsStage|Production":["Producción"],"CycleAnalyticsStage|Review":["Revisión"],"CycleAnalyticsStage|Staging":["Puesta en escena"],"CycleAnalyticsStage|Test":["Pruebas"],"Deploy":["Despliegue","Despliegues"],"FirstPushedBy|First":["Primer"],"FirstPushedBy|pushed by":["enviado por"],"From issue creation until deploy to production":["Desde la creación de la incidencia hasta el despliegue a producción"],"From merge request merge until deploy to production":["Desde la integración de la solicitud de fusión hasta el despliegue a producción"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Último %d día","Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d evento","Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issue":["Nueva incidencia","Nuevas incidencias"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OpenedNDaysAgo|Opened":["Abierto"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Deployed Jobs":["Trabajos Desplegados Relacionados"],"Related Issues":["Incidencias Relacionadas"],"Related Jobs":["Trabajos Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Related Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida de desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe el primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time before an issue gets scheduled":["Tiempo antes de que una incidencia sea programada"],"Time before an issue starts implementation":["Tiempo antes de que empieze la implementación de una incidencia"],"Time between merge request creation and merge/close":["Tiempo entre la creación de la solicitud de fusión y la integración o cierre de ésta"],"Time until first merge request":["Tiempo hasta la primera solicitud de fusión"],"Time|hr":["hr","hrs"],"Time|min":["min","mins"],"Time|s":["s"],"Total Time":["Tiempo Total"],"Total test time for all commits/merges":["Tiempo total de pruebas para todos los cambios o integraciones"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pide acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"]}}};
\ No newline at end of file
diff --git a/lib/gitlab/cycle_analytics/review_stage.rb b/lib/gitlab/cycle_analytics/review_stage.rb
index 0642e1ce32fa9..cfbbdc43fd93c 100644
--- a/lib/gitlab/cycle_analytics/review_stage.rb
+++ b/lib/gitlab/cycle_analytics/review_stage.rb
@@ -18,7 +18,7 @@ def title
       end
 
       def legend
-        "Related Merged Requests"
+        _("Related Merged Requests")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/staging_stage.rb b/lib/gitlab/cycle_analytics/staging_stage.rb
index d5c7d3f915a5a..d5684bb920178 100644
--- a/lib/gitlab/cycle_analytics/staging_stage.rb
+++ b/lib/gitlab/cycle_analytics/staging_stage.rb
@@ -19,7 +19,7 @@ def title
       end
 
       def legend
-        "Related Deployed Jobs"
+        _("Related Deployed Jobs")
       end
 
       def description
diff --git a/lib/gitlab/cycle_analytics/test_stage.rb b/lib/gitlab/cycle_analytics/test_stage.rb
index 956ee9b8b74a6..2b5f72bef8930 100644
--- a/lib/gitlab/cycle_analytics/test_stage.rb
+++ b/lib/gitlab/cycle_analytics/test_stage.rb
@@ -18,7 +18,7 @@ def title
       end
 
       def legend
-        "Related Jobs"
+        _("Related Jobs")
       end
 
       def description
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
index 5a13000434ea5..b804dc0436f4f 100644
--- a/locale/de/gitlab.po
+++ b/locale/de/gitlab.po
@@ -108,19 +108,19 @@ msgstr ""
 msgid "Related Commits"
 msgstr ""
 
-msgid "Related Issues"
+msgid "Related Deployed Jobs"
 msgstr ""
 
-msgid "Related Merge Requests"
+msgid "Related Issues"
 msgstr ""
 
-msgid "Relative Builds Trigger by Commits"
+msgid "Related Jobs"
 msgstr ""
 
-msgid "Relative Deployed Builds"
+msgid "Related Merge Requests"
 msgstr ""
 
-msgid "Relative Merged Requests"
+msgid "Related Merged Requests"
 msgstr ""
 
 msgid "Showing %d event"
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
index 93ecefec762e9..a43bafbbe2872 100644
--- a/locale/en/gitlab.po
+++ b/locale/en/gitlab.po
@@ -108,19 +108,19 @@ msgstr ""
 msgid "Related Commits"
 msgstr ""
 
-msgid "Related Issues"
+msgid "Related Deployed Jobs"
 msgstr ""
 
-msgid "Related Merge Requests"
+msgid "Related Issues"
 msgstr ""
 
-msgid "Relative Builds Trigger by Commits"
+msgid "Related Jobs"
 msgstr ""
 
-msgid "Relative Deployed Builds"
+msgid "Related Merge Requests"
 msgstr ""
 
-msgid "Relative Merged Requests"
+msgid "Related Merged Requests"
 msgstr ""
 
 msgid "Showing %d event"
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
index 2074912c1477a..c14ddd3b94c6c 100644
--- a/locale/es/gitlab.po
+++ b/locale/es/gitlab.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"PO-Revision-Date: 2017-05-04 14:52-0500\n"
+"PO-Revision-Date: 2017-05-04 19:24-0500\n"
 "Language-Team: Spanish\n"
 "Language: es\n"
 "MIME-Version: 1.0\n"
@@ -109,20 +109,20 @@ msgstr "Leer más"
 msgid "Related Commits"
 msgstr "Cambios Relacionados"
 
+msgid "Related Deployed Jobs"
+msgstr "Trabajos Desplegados Relacionados"
+
 msgid "Related Issues"
 msgstr "Incidencias Relacionadas"
 
+msgid "Related Jobs"
+msgstr "Trabajos Relacionados"
+
 msgid "Related Merge Requests"
 msgstr "Solicitudes de fusión Relacionadas"
 
-msgid "Relative Builds Trigger by Commits"
-msgstr "Builds Relativos Originados por Cambios"
-
-msgid "Relative Deployed Builds"
-msgstr "Builds Desplegados Relativos"
-
-msgid "Relative Merged Requests"
-msgstr "Solicitudes de fusión Relativas"
+msgid "Related Merged Requests"
+msgstr "Solicitudes de fusión Relacionadas"
 
 msgid "Showing %d event"
 msgid_plural "Showing %d events"
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
index 4473f5715a2f9..3967d40ea9e50 100644
--- a/locale/gitlab.pot
+++ b/locale/gitlab.pot
@@ -8,8 +8,8 @@ msgid ""
 msgstr ""
 "Project-Id-Version: gitlab 1.0.0\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-05-04 14:55-0500\n"
-"PO-Revision-Date: 2017-05-04 14:55-0500\n"
+"POT-Creation-Date: 2017-05-04 19:24-0500\n"
+"PO-Revision-Date: 2017-05-04 19:24-0500\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
 "Language: \n"
@@ -109,19 +109,19 @@ msgstr ""
 msgid "Related Commits"
 msgstr ""
 
-msgid "Related Issues"
+msgid "Related Deployed Jobs"
 msgstr ""
 
-msgid "Related Merge Requests"
+msgid "Related Issues"
 msgstr ""
 
-msgid "Relative Builds Trigger by Commits"
+msgid "Related Jobs"
 msgstr ""
 
-msgid "Relative Deployed Builds"
+msgid "Related Merge Requests"
 msgstr ""
 
-msgid "Relative Merged Requests"
+msgid "Related Merged Requests"
 msgstr ""
 
 msgid "Showing %d event"
-- 
GitLab


From c4094b7ec4699811f928699ad67c90e1686da6e2 Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Thu, 4 May 2017 15:55:36 -0500
Subject: [PATCH 256/363] Fix specs

---
 .../blob/file_template_mediator.js            |  6 ++-
 .../blob/file_template_selector.js            | 10 ++++-
 .../template_selectors/ci_yaml_selector.js    |  2 +-
 .../template_selectors/dockerfile_selector.js |  2 +-
 .../template_selectors/gitignore_selector.js  |  2 +-
 .../template_selectors/license_selector.js    | 13 +++++-
 .../blob/template_selectors/type_selector.js  |  2 +-
 .../protected_tag_access_dropdown.js          |  4 +-
 .../protected_tags/protected_tag_dropdown.js  |  4 +-
 app/assets/javascripts/users_select.js        | 41 +++++++++++--------
 .../stylesheets/framework/dropdowns.scss      |  4 ++
 app/helpers/form_helper.rb                    | 11 +++--
 .../_create_protected_branch.html.haml        |  4 +-
 .../_update_protected_branch.html.haml        |  4 +-
 .../protected_tags/_dropdown.html.haml        |  2 +-
 .../protected_tags/_update_protected_tag.haml |  2 +-
 .../shared/issuable/form/_metadata.html.haml  |  5 ++-
 spec/features/issues/form_spec.rb             | 13 +++---
 spec/features/issues_spec.rb                  |  2 +-
 spec/policies/issue_policy_spec.rb            |  8 ++--
 .../authorized_destroy_service_spec.rb        |  6 +--
 spec/services/users/destroy_service_spec.rb   |  4 +-
 22 files changed, 93 insertions(+), 58 deletions(-)

diff --git a/app/assets/javascripts/blob/file_template_mediator.js b/app/assets/javascripts/blob/file_template_mediator.js
index 3062cd51ee370..a20c6ca7a21ab 100644
--- a/app/assets/javascripts/blob/file_template_mediator.js
+++ b/app/assets/javascripts/blob/file_template_mediator.js
@@ -99,7 +99,7 @@ export default class FileTemplateMediator {
     });
   }
 
-  selectTemplateType(item, el, e) {
+  selectTemplateType(item, e) {
     if (e) {
       e.preventDefault();
     }
@@ -117,6 +117,10 @@ export default class FileTemplateMediator {
     this.cacheToggleText();
   }
 
+  selectTemplateTypeOptions(options) {
+    this.selectTemplateType(options.selectedObj, options.e);
+  }
+
   selectTemplateFile(selector, query, data) {
     selector.renderLoading();
     // in case undo menu is already already there
diff --git a/app/assets/javascripts/blob/file_template_selector.js b/app/assets/javascripts/blob/file_template_selector.js
index 31dd45fac89ad..ab5b3751c4e2f 100644
--- a/app/assets/javascripts/blob/file_template_selector.js
+++ b/app/assets/javascripts/blob/file_template_selector.js
@@ -52,9 +52,17 @@ export default class FileTemplateSelector {
       .removeClass('fa-spinner fa-spin');
   }
 
-  reportSelection(query, el, e, data) {
+  reportSelection(options) {
+    const { query, e, data } = options;
     e.preventDefault();
     return this.mediator.selectTemplateFile(this, query, data);
   }
+
+  reportSelectionName(options) {
+    const opts = options;
+    opts.query = options.selectedObj.name;
+
+    this.reportSelection(opts);
+  }
 }
 
diff --git a/app/assets/javascripts/blob/template_selectors/ci_yaml_selector.js b/app/assets/javascripts/blob/template_selectors/ci_yaml_selector.js
index 935df07677cb2..f2f81af137b4b 100644
--- a/app/assets/javascripts/blob/template_selectors/ci_yaml_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/ci_yaml_selector.js
@@ -25,7 +25,7 @@ export default class BlobCiYamlSelector extends FileTemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (query, el, e) => this.reportSelection(query.name, el, e),
+      clicked: options => this.reportSelectionName(options),
       text: item => item.name,
     });
   }
diff --git a/app/assets/javascripts/blob/template_selectors/dockerfile_selector.js b/app/assets/javascripts/blob/template_selectors/dockerfile_selector.js
index b4b4d09c315b3..3cb7b960aaa7c 100644
--- a/app/assets/javascripts/blob/template_selectors/dockerfile_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/dockerfile_selector.js
@@ -25,7 +25,7 @@ export default class DockerfileSelector extends FileTemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (query, el, e) => this.reportSelection(query.name, el, e),
+      clicked: options => this.reportSelectionName(options),
       text: item => item.name,
     });
   }
diff --git a/app/assets/javascripts/blob/template_selectors/gitignore_selector.js b/app/assets/javascripts/blob/template_selectors/gitignore_selector.js
index aefae54ae71a4..7efda8e7f50d8 100644
--- a/app/assets/javascripts/blob/template_selectors/gitignore_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/gitignore_selector.js
@@ -24,7 +24,7 @@ export default class BlobGitignoreSelector extends FileTemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (query, el, e) => this.reportSelection(query.name, el, e),
+      clicked: options => this.reportSelectionName(options),
       text: item => item.name,
     });
   }
diff --git a/app/assets/javascripts/blob/template_selectors/license_selector.js b/app/assets/javascripts/blob/template_selectors/license_selector.js
index c8abd689ab4ef..1d757332f6c9f 100644
--- a/app/assets/javascripts/blob/template_selectors/license_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/license_selector.js
@@ -24,13 +24,22 @@ export default class BlobLicenseSelector extends FileTemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (query, el, e) => {
+      clicked: (options) => {
+        const { e } = options;
+        const el = options.$el;
+        const query = options.selectedObj;
+
         const data = {
           project: this.$dropdown.data('project'),
           fullname: this.$dropdown.data('fullname'),
         };
 
-        this.reportSelection(query.id, el, e, data);
+        this.reportSelection({
+          query: query.id,
+          el,
+          e,
+          data,
+        });
       },
       text: item => item.name,
     });
diff --git a/app/assets/javascripts/blob/template_selectors/type_selector.js b/app/assets/javascripts/blob/template_selectors/type_selector.js
index 56f23ef05687a..a09381014a754 100644
--- a/app/assets/javascripts/blob/template_selectors/type_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/type_selector.js
@@ -17,7 +17,7 @@ export default class FileTemplateTypeSelector extends FileTemplateSelector {
       filterable: false,
       selectable: true,
       toggleLabel: item => item.name,
-      clicked: (item, el, e) => this.mediator.selectTemplateType(item, el, e),
+      clicked: options => this.mediator.selectTemplateTypeOptions(options),
       text: item => item.name,
     });
   }
diff --git a/app/assets/javascripts/protected_tags/protected_tag_access_dropdown.js b/app/assets/javascripts/protected_tags/protected_tag_access_dropdown.js
index fff83f3af3bb3..d4c9a91a74a42 100644
--- a/app/assets/javascripts/protected_tags/protected_tag_access_dropdown.js
+++ b/app/assets/javascripts/protected_tags/protected_tag_access_dropdown.js
@@ -17,8 +17,8 @@ export default class ProtectedTagAccessDropdown {
         }
         return 'Select';
       },
-      clicked(item, $el, e) {
-        e.preventDefault();
+      clicked(options) {
+        options.e.preventDefault();
         onSelect();
       },
     });
diff --git a/app/assets/javascripts/protected_tags/protected_tag_dropdown.js b/app/assets/javascripts/protected_tags/protected_tag_dropdown.js
index 5ff4e4432622b..068e9698e1d02 100644
--- a/app/assets/javascripts/protected_tags/protected_tag_dropdown.js
+++ b/app/assets/javascripts/protected_tags/protected_tag_dropdown.js
@@ -39,8 +39,8 @@ export default class ProtectedTagDropdown {
         return _.escape(protectedTag.id);
       },
       onFilter: this.toggleCreateNewButton.bind(this),
-      clicked: (item, $el, e) => {
-        e.preventDefault();
+      clicked: (options) => {
+        options.e.preventDefault();
         this.onSelect();
       },
     });
diff --git a/app/assets/javascripts/users_select.js b/app/assets/javascripts/users_select.js
index 2dcaa24e93a1f..be29b08c34345 100644
--- a/app/assets/javascripts/users_select.js
+++ b/app/assets/javascripts/users_select.js
@@ -77,7 +77,11 @@ import eventHub from './sidebar/event_hub';
               input.value = _this.currentUser.id;
             }
 
-            $dropdown.before(input);
+            if ($selectbox) {
+              $dropdown.parent().before(input);
+            } else {
+              $dropdown.after(input);
+            }
           };
 
           if ($block[0]) {
@@ -95,6 +99,24 @@ import eventHub from './sidebar/event_hub';
               .get();
           };
 
+          const checkMaxSelect = function() {
+            const maxSelect = $dropdown.data('max-select');
+            if (maxSelect) {
+              const selected = getSelected();
+
+              if (selected.length > maxSelect) {
+                const firstSelectedId = selected[0];
+                const firstSelected = $dropdown.closest('.selectbox')
+                  .find(`input[name='${$dropdown.data('field-name')}'][value=${firstSelectedId}]`);
+
+                firstSelected.remove();
+                eventHub.$emit('sidebar.removeAssignee', {
+                  id: firstSelectedId,
+                });
+              }
+            }
+          };
+
           const getMultiSelectDropdownTitle = function(selectedUser, isSelected) {
             const selectedUsers = getSelected()
               .filter(u => u !== 0);
@@ -125,6 +147,7 @@ import eventHub from './sidebar/event_hub';
 
             if ($dropdown.data('multiSelect')) {
               assignYourself();
+              checkMaxSelect();
 
               const currentUserInfo = $dropdown.data('currentUserInfo');
               $dropdown.find('.dropdown-toggle-text').text(getMultiSelectDropdownTitle(currentUserInfo)).removeClass('is-default');
@@ -333,21 +356,7 @@ import eventHub from './sidebar/event_hub';
 
                 // Enables support for limiting the number of users selected
                 // Automatically removes the first on the list if more users are selected
-                const maxSelect = $dropdown.data('max-select');
-                if (maxSelect) {
-                  const selected = getSelected();
-
-                  if (selected.length > maxSelect) {
-                    const firstSelectedId = selected[0];
-                    const firstSelected = $dropdown.closest('.selectbox')
-                      .find(`input[name='${$dropdown.data('field-name')}'][value=${firstSelectedId}]`);
-
-                    firstSelected.remove();
-                    eventHub.$emit('sidebar.removeAssignee', {
-                      id: firstSelectedId,
-                    });
-                  }
-                }
+                checkMaxSelect();
 
                 if (user.beforeDivider && user.name.toLowerCase() === 'unassigned') {
                   // Unassigned selected
diff --git a/app/assets/stylesheets/framework/dropdowns.scss b/app/assets/stylesheets/framework/dropdowns.scss
index 856989bccf11f..5c9b71a452cc0 100644
--- a/app/assets/stylesheets/framework/dropdowns.scss
+++ b/app/assets/stylesheets/framework/dropdowns.scss
@@ -257,6 +257,10 @@
     padding: 0 16px;
   }
 
+  &.capitalize-header .dropdown-header {
+    text-transform: capitalize;
+  }
+
   .separator + .dropdown-header {
     padding-top: 2px;
   }
diff --git a/app/helpers/form_helper.rb b/app/helpers/form_helper.rb
index 639a720b024f7..53962b846185b 100644
--- a/app/helpers/form_helper.rb
+++ b/app/helpers/form_helper.rb
@@ -18,7 +18,7 @@ def form_errors(model)
 
   def issue_dropdown_options(issuable, has_multiple_assignees = true)
     options = {
-      toggle_class: 'js-user-search js-assignee-search',
+      toggle_class: 'js-user-search js-assignee-search js-multiselect js-save-user-data',
       title: 'Select assignee',
       filter: true,
       dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-assignee',
@@ -32,16 +32,15 @@ def issue_dropdown_options(issuable, has_multiple_assignees = true)
         default_label: 'Assignee',
         'max-select': 1,
         'dropdown-header': 'Assignee',
+        multi_select: true,
+        'input-meta': 'name',
+        'always-show-selectbox': true,
+        current_user_info: current_user.to_json(only: [:id, :name])
       }
     }
 
     if has_multiple_assignees
-      options[:toggle_class] += ' js-multiselect js-save-user-data'
       options[:title] = 'Select assignee(s)'
-      options[:data][:multi_select] = true
-      options[:data][:'input-meta'] = 'name'
-      options[:data][:'always-show-selectbox'] = true
-      options[:data][:current_user_info] = current_user.to_json(only: [:id, :name])
       options[:data][:'dropdown-header'] = 'Assignee(s)'
       options[:data].delete(:'max-select')
     end
diff --git a/app/views/projects/protected_branches/_create_protected_branch.html.haml b/app/views/projects/protected_branches/_create_protected_branch.html.haml
index b8e885b4d9a7a..99bc25163660b 100644
--- a/app/views/projects/protected_branches/_create_protected_branch.html.haml
+++ b/app/views/projects/protected_branches/_create_protected_branch.html.haml
@@ -25,7 +25,7 @@
             .merge_access_levels-container
               = dropdown_tag('Select',
                              options: { toggle_class: 'js-allowed-to-merge wide',
-                             dropdown_class: 'dropdown-menu-selectable',
+                             dropdown_class: 'dropdown-menu-selectable capitalize-header',
                              data: { field_name: 'protected_branch[merge_access_levels_attributes][0][access_level]', input_id: 'merge_access_levels_attributes' }})
         .form-group
           %label.col-md-2.text-right{ for: 'push_access_levels_attributes' }
@@ -34,7 +34,7 @@
             .push_access_levels-container
               = dropdown_tag('Select',
                              options: { toggle_class: 'js-allowed-to-push wide',
-                             dropdown_class: 'dropdown-menu-selectable',
+                             dropdown_class: 'dropdown-menu-selectable capitalize-header',
                              data: { field_name: 'protected_branch[push_access_levels_attributes][0][access_level]', input_id: 'push_access_levels_attributes' }})
 
     .panel-footer
diff --git a/app/views/projects/protected_branches/_update_protected_branch.html.haml b/app/views/projects/protected_branches/_update_protected_branch.html.haml
index d6044aacaec85..c61b2951e1e89 100644
--- a/app/views/projects/protected_branches/_update_protected_branch.html.haml
+++ b/app/views/projects/protected_branches/_update_protected_branch.html.haml
@@ -1,10 +1,10 @@
 %td
   = hidden_field_tag "allowed_to_merge_#{protected_branch.id}", protected_branch.merge_access_levels.first.access_level
   = dropdown_tag( (protected_branch.merge_access_levels.first.humanize || 'Select') ,
-                 options: { toggle_class: 'js-allowed-to-merge', dropdown_class: 'dropdown-menu-selectable js-allowed-to-merge-container',
+                 options: { toggle_class: 'js-allowed-to-merge', dropdown_class: 'dropdown-menu-selectable js-allowed-to-merge-container capitalize-header',
                  data: { field_name: "allowed_to_merge_#{protected_branch.id}", access_level_id: protected_branch.merge_access_levels.first.id }})
 %td
   = hidden_field_tag "allowed_to_push_#{protected_branch.id}", protected_branch.push_access_levels.first.access_level
   = dropdown_tag( (protected_branch.push_access_levels.first.humanize || 'Select') ,
-                 options: { toggle_class: 'js-allowed-to-push', dropdown_class: 'dropdown-menu-selectable js-allowed-to-push-container',
+                 options: { toggle_class: 'js-allowed-to-push', dropdown_class: 'dropdown-menu-selectable js-allowed-to-push-container capitalize-header',
                  data: { field_name: "allowed_to_push_#{protected_branch.id}", access_level_id: protected_branch.push_access_levels.first.id }})
diff --git a/app/views/projects/protected_tags/_dropdown.html.haml b/app/views/projects/protected_tags/_dropdown.html.haml
index 748515190779f..c50515cfe06e9 100644
--- a/app/views/projects/protected_tags/_dropdown.html.haml
+++ b/app/views/projects/protected_tags/_dropdown.html.haml
@@ -2,7 +2,7 @@
 
 = dropdown_tag('Select tag or create wildcard',
                options: { toggle_class: 'js-protected-tag-select js-filter-submit wide',
-                          filter: true, dropdown_class: "dropdown-menu-selectable", placeholder: "Search protected tag",
+                          filter: true, dropdown_class: "dropdown-menu-selectable  capitalize-header", placeholder: "Search protected tag",
                           footer_content: true,
                           data: { show_no: true, show_any: true, show_upcoming: true,
                                   selected: params[:protected_tag_name],
diff --git a/app/views/projects/protected_tags/_update_protected_tag.haml b/app/views/projects/protected_tags/_update_protected_tag.haml
index 62823bee46e1f..cc80bd04dd067 100644
--- a/app/views/projects/protected_tags/_update_protected_tag.haml
+++ b/app/views/projects/protected_tags/_update_protected_tag.haml
@@ -1,5 +1,5 @@
 %td
   = hidden_field_tag "allowed_to_create_#{protected_tag.id}", protected_tag.create_access_levels.first.access_level
   = dropdown_tag( (protected_tag.create_access_levels.first.humanize || 'Select') ,
-                 options: { toggle_class: 'js-allowed-to-create', dropdown_class: 'dropdown-menu-selectable js-allowed-to-create-container',
+                 options: { toggle_class: 'js-allowed-to-create', dropdown_class: 'dropdown-menu-selectable capitalize-header js-allowed-to-create-container',
                  data: { field_name: "allowed_to_create_#{protected_tag.id}", access_level_id: protected_tag.create_access_levels.first.id }})
diff --git a/app/views/shared/issuable/form/_metadata.html.haml b/app/views/shared/issuable/form/_metadata.html.haml
index 411cb717fc766..9281a51574444 100644
--- a/app/views/shared/issuable/form/_metadata.html.haml
+++ b/app/views/shared/issuable/form/_metadata.html.haml
@@ -17,7 +17,10 @@
             - issuable.assignees.each do |assignee|
               = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", assignee.id, id: nil, data: { meta: assignee.name }
 
-            = dropdown_tag(users_dropdown_label(issuable.assignees), options: issue_dropdown_options(issuable, true))
+            - if issuable.assignees.length === 0
+              = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", 0, id: nil, data: { meta: '' }
+
+            = dropdown_tag(users_dropdown_label(issuable.assignees), options: issue_dropdown_options(issuable,false))
           = link_to 'Assign to me', '#', class: "assign-to-me-link #{'hide' if issuable.assignees.include?(current_user)}"
       - else
         = form.label :assignee_id, "Assignee", class: "control-label #{"col-lg-4" if has_due_date}"
diff --git a/spec/features/issues/form_spec.rb b/spec/features/issues/form_spec.rb
index 5798292033b14..87adce3cdddf4 100644
--- a/spec/features/issues/form_spec.rb
+++ b/spec/features/issues/form_spec.rb
@@ -77,11 +77,10 @@
       click_link 'Assign to me'
       assignee_ids = page.all('input[name="issue[assignee_ids][]"]', visible: false)
 
-      expect(assignee_ids[0].value).to match(user2.id.to_s)
-      expect(assignee_ids[1].value).to match(user.id.to_s)
+      expect(assignee_ids[0].value).to match(user.id.to_s)
 
       page.within '.js-assignee-search' do
-        expect(page).to have_content "#{user2.name} + 1 more"
+        expect(page).to have_content user.name
       end
       expect(find('a', text: 'Assign to me', visible: false)).not_to be_visible
 
@@ -109,7 +108,7 @@
 
       page.within '.issuable-sidebar' do
         page.within '.assignee' do
-          expect(page).to have_content "2 Assignees"
+          expect(page).to have_content "Assignee"
         end
 
         page.within '.milestone' do
@@ -148,12 +147,12 @@
     end
 
     it 'correctly updates the selected user when changing assignee' do
-      click_button 'Assignee'
+      click_button 'Unassigned'
       page.within '.dropdown-menu-user' do
         click_link user.name
       end
 
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user.id.to_s)
+      expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match(user.id.to_s)
 
       click_button user.name
 
@@ -167,7 +166,7 @@
         click_link user2.name
       end
 
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user2.id.to_s)
+      expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match(user2.id.to_s)
 
       click_button user2.name
 
diff --git a/spec/features/issues_spec.rb b/spec/features/issues_spec.rb
index cc81303f03292..5285dda361b4e 100644
--- a/spec/features/issues_spec.rb
+++ b/spec/features/issues_spec.rb
@@ -710,7 +710,7 @@
     include WaitForVueResource
 
     it 'updates the title', js: true do
-      issue = create(:issue, author: @user, assignee: @user, project: project, title: 'new title')
+      issue = create(:issue, author: @user, assignees: [@user], project: project, title: 'new title')
 
       visit namespace_project_issue_path(project.namespace, project, issue)
 
diff --git a/spec/policies/issue_policy_spec.rb b/spec/policies/issue_policy_spec.rb
index 9a870b7fda12f..4a07c864428c6 100644
--- a/spec/policies/issue_policy_spec.rb
+++ b/spec/policies/issue_policy_spec.rb
@@ -15,7 +15,7 @@ def permissions(user, issue)
   context 'a private project' do
     let(:non_member) { create(:user) }
     let(:project) { create(:empty_project, :private) }
-    let(:issue) { create(:issue, project: project, assignee: assignee, author: author) }
+    let(:issue) { create(:issue, project: project, assignees: [assignee], author: author) }
     let(:issue_no_assignee) { create(:issue, project: project) }
 
     before do
@@ -69,7 +69,7 @@ def permissions(user, issue)
     end
 
     context 'with confidential issues' do
-      let(:confidential_issue) { create(:issue, :confidential, project: project, assignee: assignee, author: author) }
+      let(:confidential_issue) { create(:issue, :confidential, project: project, assignees: [assignee], author: author) }
       let(:confidential_issue_no_assignee) { create(:issue, :confidential, project: project) }
 
       it 'does not allow non-members to read confidential issues' do
@@ -110,7 +110,7 @@ def permissions(user, issue)
 
   context 'a public project' do
     let(:project) { create(:empty_project, :public) }
-    let(:issue) { create(:issue, project: project, assignee: assignee, author: author) }
+    let(:issue) { create(:issue, project: project, assignees: [assignee], author: author) }
     let(:issue_no_assignee) { create(:issue, project: project) }
 
     before do
@@ -157,7 +157,7 @@ def permissions(user, issue)
     end
 
     context 'with confidential issues' do
-      let(:confidential_issue) { create(:issue, :confidential, project: project, assignee: assignee, author: author) }
+      let(:confidential_issue) { create(:issue, :confidential, project: project, assignees: [assignee], author: author) }
       let(:confidential_issue_no_assignee) { create(:issue, :confidential, project: project) }
 
       it 'does not allow guests to read confidential issues' do
diff --git a/spec/services/members/authorized_destroy_service_spec.rb b/spec/services/members/authorized_destroy_service_spec.rb
index 3b35a3b8e3a3a..ab440d18e9f68 100644
--- a/spec/services/members/authorized_destroy_service_spec.rb
+++ b/spec/services/members/authorized_destroy_service_spec.rb
@@ -14,8 +14,8 @@ def number_of_assigned_issuables(user)
     it "unassigns issues and merge requests" do
       group.add_developer(member_user)
 
-      issue = create :issue, project: group_project, assignee: member_user
-      create :issue, assignee: member_user
+      issue = create :issue, project: group_project, assignees: [member_user]
+      create :issue, assignees: [member_user]
       merge_request = create :merge_request, target_project: group_project, source_project: group_project, assignee: member_user
       create :merge_request, target_project: project, source_project: project, assignee: member_user
 
@@ -33,7 +33,7 @@ def number_of_assigned_issuables(user)
     it "unassigns issues and merge requests" do
       project.team << [member_user, :developer]
 
-      create :issue, project: project, assignee: member_user
+      create :issue, project: project, assignees: [member_user]
       create :merge_request, target_project: project, source_project: project, assignee: member_user
 
       member = project.members.find_by(user_id: member_user.id)
diff --git a/spec/services/users/destroy_service_spec.rb b/spec/services/users/destroy_service_spec.rb
index 4bc30018ebd5d..8aa900d1a75fa 100644
--- a/spec/services/users/destroy_service_spec.rb
+++ b/spec/services/users/destroy_service_spec.rb
@@ -47,7 +47,7 @@
       end
 
       context "for an issue the user was assigned to" do
-        let!(:issue) { create(:issue, project: project, assignee: user) }
+        let!(:issue) { create(:issue, project: project, assignees: [user]) }
 
         before do
           service.execute(user)
@@ -60,7 +60,7 @@
         it 'migrates the issue so that it is "Unassigned"' do
           migrated_issue = Issue.find_by_id(issue.id)
 
-          expect(migrated_issue.assignee).to be_nil
+          expect(migrated_issue.assignees).to be_nil
         end
       end
     end
-- 
GitLab


From cca09bbb1864bf300157d4d9c5500b83acf9f5c7 Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Fri, 5 May 2017 16:39:26 +1100
Subject: [PATCH 257/363] Update Import/Export files

---
 lib/gitlab/import_export/relation_factory.rb            | 2 +-
 spec/lib/gitlab/import_export/all_models.yml            | 3 +++
 spec/lib/gitlab/import_export/safe_model_attributes.yml | 4 ++++
 3 files changed, 8 insertions(+), 1 deletion(-)

diff --git a/lib/gitlab/import_export/relation_factory.rb b/lib/gitlab/import_export/relation_factory.rb
index 4a54e7ef2e736..956763fa39927 100644
--- a/lib/gitlab/import_export/relation_factory.rb
+++ b/lib/gitlab/import_export/relation_factory.rb
@@ -15,7 +15,7 @@ class RelationFactory
                     priorities: :label_priorities,
                     label: :project_label }.freeze
 
-      USER_REFERENCES = %w[author_id assignee_id updated_by_id user_id created_by_id merge_user_id resolved_by_id].freeze
+      USER_REFERENCES = %w[author_id assignee_id updated_by_id user_id created_by_id last_edited_by_id merge_user_id resolved_by_id].freeze
 
       PROJECT_REFERENCES = %w[project_id source_project_id target_project_id].freeze
 
diff --git a/spec/lib/gitlab/import_export/all_models.yml b/spec/lib/gitlab/import_export/all_models.yml
index 0abf89d060cec..9ae294ae8d1fa 100644
--- a/spec/lib/gitlab/import_export/all_models.yml
+++ b/spec/lib/gitlab/import_export/all_models.yml
@@ -9,6 +9,7 @@ issues:
 - notes
 - label_links
 - labels
+- last_edited_by
 - todos
 - user_agent_detail
 - moved_to
@@ -26,6 +27,7 @@ notes:
 - noteable
 - author
 - updated_by
+- last_edited_by
 - resolved_by
 - todos
 - events
@@ -71,6 +73,7 @@ merge_requests:
 - notes
 - label_links
 - labels
+- last_edited_by
 - todos
 - target_project
 - source_project
diff --git a/spec/lib/gitlab/import_export/safe_model_attributes.yml b/spec/lib/gitlab/import_export/safe_model_attributes.yml
index ebfaab4eacd78..5fdc30207c3f1 100644
--- a/spec/lib/gitlab/import_export/safe_model_attributes.yml
+++ b/spec/lib/gitlab/import_export/safe_model_attributes.yml
@@ -23,6 +23,8 @@ Issue:
 - weight
 - time_estimate
 - relative_position
+- last_edited_at
+- last_edited_by_id
 Event:
 - id
 - target_type
@@ -154,6 +156,8 @@ MergeRequest:
 - approvals_before_merge
 - rebase_commit_sha
 - time_estimate
+- last_edited_at
+- last_edited_by_id
 MergeRequestDiff:
 - id
 - state
-- 
GitLab


From a1debf5cf029edc550d7d46eff7ef4ab1b51a2b2 Mon Sep 17 00:00:00 2001
From: Filipa Lacerda <filipa@gitlab.com>
Date: Mon, 10 Apr 2017 11:30:06 +0100
Subject: [PATCH 258/363] Retry and cancel endpoints send 204 json response.
 Request is made with type json

---
 .../pipelines/services/pipelines_service.js   |  2 +-
 .../projects/pipelines_controller.rb          | 20 +++++++++++++++++--
 2 files changed, 19 insertions(+), 3 deletions(-)

diff --git a/app/assets/javascripts/pipelines/services/pipelines_service.js b/app/assets/javascripts/pipelines/services/pipelines_service.js
index 255cd513490e4..b21f84b454529 100644
--- a/app/assets/javascripts/pipelines/services/pipelines_service.js
+++ b/app/assets/javascripts/pipelines/services/pipelines_service.js
@@ -40,6 +40,6 @@ export default class PipelinesService {
    * @return {Promise}
    */
   postAction(endpoint) {
-    return Vue.http.post(endpoint, {}, { emulateJSON: true });
+    return Vue.http.post(`${endpoint}.json`);
   }
 }
diff --git a/app/controllers/projects/pipelines_controller.rb b/app/controllers/projects/pipelines_controller.rb
index 454b8ee17af2b..6d77b9c38fa2d 100644
--- a/app/controllers/projects/pipelines_controller.rb
+++ b/app/controllers/projects/pipelines_controller.rb
@@ -92,13 +92,29 @@ def stage
   def retry
     pipeline.retry_failed(current_user)
 
-    redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
+    respond_to do |format|
+      format.html do
+        redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
+      end
+
+      format.json do
+        render status: 204
+      end
+    end
   end
 
   def cancel
     pipeline.cancel_running
 
-    redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
+    respond_to do |format|
+      format.html do
+        redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
+      end
+
+      format.json do
+        render status: 204
+      end
+    end
   end
 
   def charts
-- 
GitLab


From b8960354fecdaf0a19a370f00f7caad147c57e62 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Wed, 12 Apr 2017 09:44:16 +0200
Subject: [PATCH 259/363] Respond with no content for pipeline JSON actions

---
 app/controllers/projects/pipelines_controller.rb | 8 ++------
 1 file changed, 2 insertions(+), 6 deletions(-)

diff --git a/app/controllers/projects/pipelines_controller.rb b/app/controllers/projects/pipelines_controller.rb
index 6d77b9c38fa2d..95478da019e1c 100644
--- a/app/controllers/projects/pipelines_controller.rb
+++ b/app/controllers/projects/pipelines_controller.rb
@@ -97,9 +97,7 @@ def retry
         redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
       end
 
-      format.json do
-        render status: 204
-      end
+      format.json { head :no_content }
     end
   end
 
@@ -111,9 +109,7 @@ def cancel
         redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
       end
 
-      format.json do
-        render status: 204
-      end
+      format.json { head :no_content }
     end
   end
 
-- 
GitLab


From c68bf4327b3e4811e001adeca84ad0f88c421455 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Mon, 24 Apr 2017 16:34:53 +0200
Subject: [PATCH 260/363] Use wrap_parameters in pipelines controller

This makes it possible to workaround a bug in `safe_constantize` which
caused a `LoadError` exception when doing

```
"Pipeline".safe_constantize
LoadError: Unable to autoload constant Pipeline, expected
/home/grzesiek/gdk/gitlab/app/models/ci/pipeline.rb to define it
```

See https://github.com/rails/rails/issues/28854 for more details.
---
 app/controllers/projects/pipelines_controller.rb | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/app/controllers/projects/pipelines_controller.rb b/app/controllers/projects/pipelines_controller.rb
index 95478da019e1c..2908036607a30 100644
--- a/app/controllers/projects/pipelines_controller.rb
+++ b/app/controllers/projects/pipelines_controller.rb
@@ -6,6 +6,8 @@ class Projects::PipelinesController < Projects::ApplicationController
   before_action :authorize_update_pipeline!, only: [:retry, :cancel]
   before_action :builds_enabled, only: :charts
 
+  wrap_parameters Ci::Pipeline
+
   def index
     @scope = params[:scope]
     @pipelines = PipelinesFinder
-- 
GitLab


From 4f3dc19aafdd71aedb6b086da5707315a9a51ace Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Thu, 4 May 2017 21:01:15 +0200
Subject: [PATCH 261/363] Add specs for new pipeline action for JSON format

---
 .../projects/pipelines_controller_spec.rb     | 36 +++++++++++++++++++
 1 file changed, 36 insertions(+)

diff --git a/spec/controllers/projects/pipelines_controller_spec.rb b/spec/controllers/projects/pipelines_controller_spec.rb
index b9bacc5a64a25..1b47d163c0b3d 100644
--- a/spec/controllers/projects/pipelines_controller_spec.rb
+++ b/spec/controllers/projects/pipelines_controller_spec.rb
@@ -5,6 +5,8 @@
   let(:project) { create(:empty_project, :public) }
 
   before do
+    project.add_developer(user)
+
     sign_in(user)
   end
 
@@ -87,4 +89,38 @@ def get_stage(name)
       expect(json_response['favicon']).to eq "/assets/ci_favicons/#{status.favicon}.ico"
     end
   end
+
+  describe 'POST retry.json' do
+    let!(:pipeline) { create(:ci_pipeline, :failed, project: project) }
+    let!(:build) { create(:ci_build, :failed, pipeline: pipeline) }
+
+    before do
+      post :retry, namespace_id: project.namespace,
+                   project_id: project,
+                   id: pipeline.id,
+                   format: :json
+    end
+
+    it 'retries a pipeline without returning any content' do
+      expect(response).to have_http_status(:no_content)
+      expect(build.reload).to be_retried
+    end
+  end
+
+  describe 'POST cancel.json' do
+    let!(:pipeline) { create(:ci_pipeline, project: project) }
+    let!(:build) { create(:ci_build, :running, pipeline: pipeline) }
+
+    before do
+      post :cancel, namespace_id: project.namespace,
+                    project_id: project,
+                    id: pipeline.id,
+                    format: :json
+    end
+
+    it 'cancels a pipeline without returning any content' do
+      expect(response).to have_http_status(:no_content)
+      expect(pipeline.reload).to be_canceled
+    end
+  end
 end
-- 
GitLab


From f7a5ce923a3c0db9fc34f611b1a655c472503941 Mon Sep 17 00:00:00 2001
From: Filipa Lacerda <filipa@gitlab.com>
Date: Fri, 5 May 2017 07:32:36 +0000
Subject: [PATCH 262/363] Remove # char for commit

---
 app/assets/javascripts/merge_request_widget.js | 2 +-
 changelogs/unreleased/31810-commit-link.yml    | 4 ++++
 2 files changed, 5 insertions(+), 1 deletion(-)
 create mode 100644 changelogs/unreleased/31810-commit-link.yml

diff --git a/app/assets/javascripts/merge_request_widget.js b/app/assets/javascripts/merge_request_widget.js
index 42ecf0d6cb2f5..6f6ae9bde926c 100644
--- a/app/assets/javascripts/merge_request_widget.js
+++ b/app/assets/javascripts/merge_request_widget.js
@@ -291,7 +291,7 @@ import MiniPipelineGraph from './mini_pipeline_graph_dropdown';
 
     MergeRequestWidget.prototype.updateCommitUrls = function(id) {
       const commitsUrl = this.opts.commits_path;
-      $('.js-commit-link').text(`#${id}`).attr('href', [commitsUrl, id].join('/'));
+      $('.js-commit-link').text(id).attr('href', [commitsUrl, id].join('/'));
     };
 
     MergeRequestWidget.prototype.initMiniPipelineGraph = function() {
diff --git a/changelogs/unreleased/31810-commit-link.yml b/changelogs/unreleased/31810-commit-link.yml
new file mode 100644
index 0000000000000..857c9cb95c595
--- /dev/null
+++ b/changelogs/unreleased/31810-commit-link.yml
@@ -0,0 +1,4 @@
+---
+title: Remove `#` being added on commit sha in MR widget
+merge_request:
+author:
-- 
GitLab


From 1fe8b7f646603239f530b1a18427f4f5bc0e2060 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 09:40:44 +0200
Subject: [PATCH 263/363] refactor propagate service to use batch inserts and
 subquery instead of left join

---
 app/services/projects/propagate_service.rb | 32 ++++++++++++++++++----
 1 file changed, 26 insertions(+), 6 deletions(-)

diff --git a/app/services/projects/propagate_service.rb b/app/services/projects/propagate_service.rb
index 6e24a67d8b0b7..b067fc2cd6492 100644
--- a/app/services/projects/propagate_service.rb
+++ b/app/services/projects/propagate_service.rb
@@ -26,7 +26,7 @@ def propagate_projects_with_template
       loop do
         batch = project_ids_batch(offset)
 
-        batch.each { |project_id| create_from_template(project_id) }
+        bulk_create_from_template(batch)
 
         break if batch.size < BATCH_SIZE
 
@@ -34,14 +34,34 @@ def propagate_projects_with_template
       end
     end
 
-    def create_from_template(project_id)
-      Service.build_from_template(project_id, @template).save!
+    def bulk_create_from_template(batch)
+      service_hash_list = batch.map do |project_id|
+        service_hash.merge('project_id' => project_id)
+      end
+
+      Project.transaction do
+        Service.create!(service_hash_list)
+      end
     end
 
     def project_ids_batch(offset)
-      Project.joins('LEFT JOIN services ON services.project_id = projects.id').
-        where('services.type != ? OR services.id IS NULL', @template.type).
-        limit(BATCH_SIZE).offset(offset).pluck(:id)
+      Project.connection.execute(
+        <<-SQL
+          SELECT id
+          FROM projects
+          WHERE NOT EXISTS (
+            SELECT true
+            FROM services
+            WHERE services.project_id = projects.id
+            AND services.type = '#{@template.type}'
+          )
+          LIMIT #{BATCH_SIZE} OFFSET #{offset}
+      SQL
+      ).to_a.flatten
+    end
+
+    def service_hash
+      @service_hash ||= @template.as_json(methods: :type).except('id', 'template')
     end
   end
 end
-- 
GitLab


From 225662a708314947a970c9be6f969ada6c636c23 Mon Sep 17 00:00:00 2001
From: Annabel Dunstone Gray <annabel.dunstone@gmail.com>
Date: Fri, 5 May 2017 08:33:04 +0000
Subject: [PATCH 264/363] Update design of auth error page

---
 .../omniauth_callbacks_controller.rb          |   2 +-
 app/views/errors/omniauth_error.html.haml     |  21 ++-
 app/views/layouts/oauth_error.html.haml       | 127 ++++++++++++++++++
 app/views/shared/errors/_graphic_422.svg      |   1 +
 changelogs/unreleased/29145-oauth-422.yml     |   4 +
 5 files changed, 143 insertions(+), 12 deletions(-)
 create mode 100644 app/views/layouts/oauth_error.html.haml
 create mode 100644 app/views/shared/errors/_graphic_422.svg
 create mode 100644 changelogs/unreleased/29145-oauth-422.yml

diff --git a/app/controllers/omniauth_callbacks_controller.rb b/app/controllers/omniauth_callbacks_controller.rb
index 58d50ad647bf1..2a8c8ca4bad6a 100644
--- a/app/controllers/omniauth_callbacks_controller.rb
+++ b/app/controllers/omniauth_callbacks_controller.rb
@@ -67,7 +67,7 @@ def saml
   def omniauth_error
     @provider = params[:provider]
     @error = params[:error]
-    render 'errors/omniauth_error', layout: "errors", status: 422
+    render 'errors/omniauth_error', layout: "oauth_error", status: 422
   end
 
   def cas3
diff --git a/app/views/errors/omniauth_error.html.haml b/app/views/errors/omniauth_error.html.haml
index 72508b9113411..20b7fa471a0a3 100644
--- a/app/views/errors/omniauth_error.html.haml
+++ b/app/views/errors/omniauth_error.html.haml
@@ -1,16 +1,15 @@
 - content_for(:title, 'Auth Error')
-%img{ :alt => "GitLab Logo", :src => image_path('logo.svg') }
-  %h1
-    422
+
 .container
+  = render "shared/errors/graphic_422.svg"
   %h3 Sign-in using #{@provider} auth failed
-  %hr
-  %p Sign-in failed because #{@error}.
-  %p There are couple of steps you can take:
 
-%ul
-  %li Try logging in using your email
-  %li Try logging in using your username
-  %li If you have forgotten your password, try recovering it using #{ link_to "Password recovery", new_password_path(resource_name) }
+  %p.light.subtitle Sign-in failed because #{@error}.
+
+  %p Try logging in using your username or email. If you have forgotten your password, try recovering it
 
-%p If none of the options work, try contacting the GitLab administrator.
+  = link_to "Sign in", new_session_path(:user), class: 'btn primary'
+  = link_to "Recover password", new_password_path(resource_name), class: 'btn secondary'
+
+  %hr
+  %p.light If none of the options work, try contacting a GitLab administrator.
diff --git a/app/views/layouts/oauth_error.html.haml b/app/views/layouts/oauth_error.html.haml
new file mode 100644
index 0000000000000..34bcd2a8b3a60
--- /dev/null
+++ b/app/views/layouts/oauth_error.html.haml
@@ -0,0 +1,127 @@
+!!! 5
+%html{ lang: "en" }
+  %head
+    %meta{ :content => "width=device-width, initial-scale=1, maximum-scale=1", :name => "viewport" }
+    %title= yield(:title)
+    :css
+      body {
+        color: #666;
+        text-align: center;
+        font-family: "Helvetica Neue", Helvetica, Arial, sans-serif;
+        margin: auto;
+        font-size: 16px;
+      }
+
+      .container {
+        margin: auto 20px;
+      }
+
+      h3 {
+        color: #456;
+        font-size: 22px;
+        font-weight: bold;
+        margin-bottom: 6px;
+      }
+
+      p {
+        max-width: 470px;
+        margin: 16px auto;
+      }
+
+      .subtitle {
+        margin: 0 auto 20px;
+      }
+
+      svg {
+        width: 280px;
+        height: 280px;
+        display: block;
+        margin: 40px auto;
+      }
+
+      .tv-screen path {
+        animation: move-lines 1s linear infinite;
+      }
+
+
+      @keyframes move-lines {
+        0% {transform: translateY(0)}
+        50% {transform: translateY(-10px)}
+        100% {transform: translateY(-20px)}
+      }
+
+      .tv-screen path:nth-child(1) {
+        animation-delay: .2s
+      }
+
+      .tv-screen path:nth-child(2) {
+        animation-delay: .4s
+      }
+
+      .tv-screen path:nth-child(3) {
+        animation-delay: .6s
+      }
+
+      .tv-screen path:nth-child(4) {
+        animation-delay: .8s
+      }
+
+      .tv-screen path:nth-child(5) {
+        animation-delay: 2s
+      }
+
+      .text-422 {
+        animation: flicker 1s infinite;
+      }
+
+      @keyframes flicker {
+        0% {opacity: 0.3;}
+        10% {opacity: 1;}
+        15% {opacity: .3;}
+        20% {opacity: .5;}
+        25% {opacity: 1;}
+      }
+
+      .light {
+        color: #8D8D8D;
+      }
+
+      hr {
+        max-width: 600px;
+        margin: 18px auto;
+        border: 0;
+        border-top: 1px solid #EEE;
+      }
+
+      .btn {
+        padding: 8px 14px;
+        border-radius: 3px;
+        border: 1px solid;
+        display: inline-block;
+        text-decoration: none;
+        margin: 4px 8px;
+        font-size: 14px;
+      }
+
+      .primary {
+        color: #fff;
+        background-color: #1aaa55;
+        border-color: #168f48;
+      }
+
+      .primary:hover {
+        background-color: #168f48;
+      }
+
+      .secondary {
+        color: #1aaa55;
+        background-color: #fff;
+        border-color: #1aaa55;
+      }
+
+      .secondary:hover {
+        background-color: #f3fff8;
+      }
+
+%body
+  = yield
diff --git a/app/views/shared/errors/_graphic_422.svg b/app/views/shared/errors/_graphic_422.svg
new file mode 100644
index 0000000000000..87128ecd69d54
--- /dev/null
+++ b/app/views/shared/errors/_graphic_422.svg
@@ -0,0 +1 @@
+<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 260 246" xmlns:xlink="http://www.w3.org/1999/xlink"><defs><rect id="0" width="178" height="136" rx="10"/><mask id="1" width="178" height="136" x="0" y="0" fill="#fff"><use xlink:href="#0"/></mask></defs><g fill="none" fill-rule="evenodd"><g fill="#e5e5e5" fill-rule="nonzero"><path d="m109.88 37.634c5.587-3.567 12.225-5.634 19.345-5.634 7.445 0 14.363 2.26 20.1 6.132l21.435-37.13c.554-.959 1.771-1.292 2.734-.736.957.552 1.284 1.777.73 2.736l-21.496 37.23c-.065.112-.138.215-.219.309 3.686 3.13 6.733 6.988 8.919 11.353l-3.393.002c-5.775-10.322-16.705-16.901-28.814-16.901-12.12 0-23.06 6.594-28.833 16.935l-3.393.002c2.32-4.646 5.616-8.72 9.618-11.954l-21.349-36.977c-.554-.959-.227-2.184.73-2.736.963-.556 2.181-.223 2.734.736l21.15 36.629"/><path d="m3 70v134c0 9.389 7.611 17 16.997 17h220.01c9.389 0 16.997-7.611 16.997-17v-134c0-9.389-7.611-17-16.997-17h-220.01c-9.389 0-16.997 7.611-16.997 17m-3 0c0-11.05 8.95-20 19.997-20h220.01c11.04 0 19.997 8.958 19.997 20v134c0 11.05-8.95 20-19.997 20h-220.01c-11.04 0-19.997-8.958-19.997-20v-134"/></g><ellipse cx="129" cy="241.5" fill="#f9f9f9" rx="89" ry="4.5"/><g fill-rule="nonzero" transform="translate(210 70)"><path fill="#eaeaea" d="m16 29c7.18 0 13-5.82 13-13 0-7.18-5.82-13-13-13-7.18 0-13 5.82-13 13 0 7.18 5.82 13 13 13m0 3c-8.837 0-16-7.163-16-16 0-8.837 7.163-16 16-16 8.837 0 16 7.163 16 16 0 8.837-7.163 16-16 16" id="2"/><path fill="#6b4fbb" d="m16 21c2.761 0 5-2.239 5-5 0-2.761-2.239-5-5-5-2.761 0-5 2.239-5 5 0 2.761 2.239 5 5 5m0 3c-4.418 0-8-3.582-8-8 0-4.418 3.582-8 8-8 4.418 0 8 3.582 8 8 0 4.418-3.582 8-8 8" id="3"/></g><g fill-rule="nonzero" transform="translate(210 109)"><use xlink:href="#2"/><use xlink:href="#3"/></g><g transform="translate(210 147)"><path fill="#e5e5e5" fill-rule="nonzero" d="m3 5.992v45.02c0 1.647 1.346 2.992 3 2.992h20c1.657 0 3-1.341 3-2.992v-45.02c0-1.647-1.346-2.992-3-2.992h-20c-1.657 0-3 1.341-3 2.992m-3 0c0-3.309 2.687-5.992 6-5.992h20c3.314 0 6 2.692 6 5.992v45.02c0 3.309-2.687 5.992-6 5.992h-20c-3.314 0-6-2.692-6-5.992v-45.02"/><rect width="16" height="4" x="8" y="27" fill="#fdb692" rx="2"/><rect width="16" height="4" x="8" y="19" fill="#fc9867" rx="2"/><rect width="16" height="4" x="8" y="11" fill="#fc6d26" rx="2"/><rect width="16" height="4" x="8" y="35" fill="#fed3bd" rx="2"/><rect width="16" height="4" x="8" y="43" fill="#fef0e9" rx="2"/></g><g transform="translate(16 69)"><use fill="#6b4fbb" fill-opacity=".1" stroke="#e5e5e5" stroke-width="6" mask="url(#1)" xlink:href="#0"/><g class="tv-screen" fill="#fff"><path opacity=".4" mix-blend-mode="overlay" d="m3 17h172v16h-172z"/><path opacity=".6" mix-blend-mode="overlay" d="m3 70h172v24h-172z"/><path opacity=".3" mix-blend-mode="overlay" d="m3 107h172v16h-172z"/><path opacity=".4" mix-blend-mode="overlay" d="m3 40h172v8h-172z"/><path opacity=".3" mix-blend-mode="overlay" d="m3 55h172v8h-172z"/></g></g><path class="text-422" d="m.693 19h5.808c.277 0 .498-.224.498-.5 0-.268-.223-.5-.498-.5h-5.808v-2.094l3.777-5.906h3.916l-4.124 6.454h6.259v-6.454h.978c.273 0 .5-.224.5-.5 0-.268-.224-.5-.5-.5h-.978v-2h4.698v6h-2.721c-.277 0-.498.224-.498.5 0 .268.223.5.498.5h2.721v2.454h2.723v4.2h-2.723v5.346h-4.698v-5.346h-9.828v-1.654m4.417-10l1.279-2h3.914l-1.278 2h-3.916m1.919-3l1.279-2h4.192c.27 0 .5-.224.5-.5 0-.268-.224-.5-.5-.5h-3.552l1.142-1.786h5.13v4.786h-8.191m31.09 19v1h-15.738v-2h5.118c.271 0 .503-.224.503-.5 0-.268-.225-.5-.503-.5h-5.118v-1.184l2.656-2.822c.682-.725 1.306-1.39 1.872-1.994h5.428c-.389.394-.808.815-1.256 1.264-1.428 1.428-2.562 2.568-3.403 3.42h10.442v2.316h-4.614c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h4.614m-6.674-13c.493-.631.87-1.208 1.129-1.73.365-.736.548-1.464.548-2.183 0-1.107-.335-1.962-1-2.565-.67-.603-1.619-.905-2.847-.905-.874 0-1.857.174-2.947.523-1.09.349-2.227.855-3.412 1.519v-2.659h3.589c.27 0 .5-.224.5-.5 0-.268-.224-.5-.5-.5h-3.589v-.906c1.184-.432 2.344-.761 3.478-.988 1.134-.227 2.222-.34 3.262-.34 2.623 0 4.684.611 6.184 1.834.157.128.307.262.448.4h-2.782c-.27 0-.5.224-.5.5 0 .268.224.5.5.5h3.602c.654 1.01.981 2.209.981 3.605 0 .974-.163 1.887-.49 2.739-.326.852-.888 1.798-1.685 2.839-.397.509-1.261 1.448-2.594 2.816h-5.474c1.34-1.436 2.261-2.436 2.763-3h4.396c.271 0 .499-.224.499-.5 0-.268-.223-.5-.499-.5h-3.557m28.14 12v2h-15.738v-4.184l2.651-2.816h5.313c-1.087 1.089-1.976 1.983-2.668 2.684h10.442v1.316h-4.083c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h4.083m-2.069-11c-.045.061-.092.122-.139.184-.567.727-2.089 2.333-4.568 4.816h-5.372c2.601-2.77 4.204-4.503 4.81-5.198.83-.952 1.428-1.796 1.793-2.532.365-.736.548-1.464.548-2.183 0-1.107-.335-1.962-1-2.565-.67-.603-1.619-.905-2.847-.905-.874 0-1.857.174-2.947.523-1.09.349-2.227.855-3.412 1.519v-2.659h3.117c.271 0 .503-.224.503-.5 0-.268-.225-.5-.503-.5h-3.117v-.906c1.184-.432 2.344-.761 3.478-.988 1.134-.227 2.222-.34 3.262-.34 2.623 0 4.684.611 6.184 1.834.157.128.307.262.448.4h-1.248c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h2.069c.654 1.01.981 2.209.981 3.605 0 .844-.123 1.642-.368 2.395h-2.683c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h2.272c-.159.321-.347.655-.566 1h-3.706c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h3.01" transform="translate(75 124)" fill="#5c5c5c"/></g></svg>
diff --git a/changelogs/unreleased/29145-oauth-422.yml b/changelogs/unreleased/29145-oauth-422.yml
new file mode 100644
index 0000000000000..94e4cd84ad195
--- /dev/null
+++ b/changelogs/unreleased/29145-oauth-422.yml
@@ -0,0 +1,4 @@
+---
+title: Redesign auth 422 page
+merge_request:
+author:
-- 
GitLab


From b667fba826e02bec2c94ee4922edf348b84f8075 Mon Sep 17 00:00:00 2001
From: Balasankar C <balasankar@gitlab.com>
Date: Fri, 5 May 2017 08:41:16 +0000
Subject: [PATCH 265/363] Add a manual job to trigger package build in omnibus

---
 .gitlab-ci.yml                        | 23 ++++++++++++++++++
 doc/development/README.md             |  1 +
 doc/development/build_test_package.md | 35 +++++++++++++++++++++++++++
 3 files changed, 59 insertions(+)
 create mode 100644 doc/development/build_test_package.md

diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index 44620d390ad33..588f255eff8bb 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -23,6 +23,7 @@ before_script:
   - source scripts/prepare_build.sh
 
 stages:
+- build
 - prepare
 - test
 - post-test
@@ -137,6 +138,28 @@ stages:
   <<: *only-master-and-ee-or-mysql
   <<: *except-docs
 
+# Trigger a package build on omnibus-gitlab repository
+
+build-package:
+  services: []
+  variables:
+    SETUP_DB: "false"
+    USE_BUNDLE_INSTALL: "false"
+  stage: build
+  when: manual
+  script:
+    # If no branch in omnibus is specified, trigger pipeline against master
+    - if [ -z "$OMNIBUS_BRANCH" ] ; then export OMNIBUS_BRANCH=master ;fi
+    - echo "token=${BUILD_TRIGGER_TOKEN}" > version_details
+    - echo "ref=${OMNIBUS_BRANCH}" >> version_details
+    - echo "variables[ALTERNATIVE_SOURCES]=true" >> version_details
+    - echo "variables[GITLAB_VERSION]=${CI_COMMIT_SHA}" >> version_details
+    # Collect version details of all components
+    - for f in *_VERSION; do echo "variables[$f]=$(cat $f)" >> version_details; done
+    # Trigger the API and pass values collected above as parameters to it
+    - cat version_details | tr '\n' '&' | curl -X POST https://gitlab.com/api/v4/projects/20699/trigger/pipeline --data-binary @-
+    - rm version_details
+
 # Prepare and merge knapsack tests
 knapsack:
   <<: *knapsack-state
diff --git a/doc/development/README.md b/doc/development/README.md
index 77bb0263374f3..d04380e5b3323 100644
--- a/doc/development/README.md
+++ b/doc/development/README.md
@@ -41,6 +41,7 @@
 - [Shell commands](shell_commands.md) in the GitLab codebase
 - [Sidekiq debugging](sidekiq_debugging.md)
 - [Object state models](object_state_models.md)
+- [Building a package for testing purposes](build_test_package.md)
 
 ## Databases
 
diff --git a/doc/development/build_test_package.md b/doc/development/build_test_package.md
new file mode 100644
index 0000000000000..2bc1a7008444d
--- /dev/null
+++ b/doc/development/build_test_package.md
@@ -0,0 +1,35 @@
+# Building a package for testing
+
+While developing a new feature or modifying an existing one, it is helpful if an
+installable package (or a docker image) containing those changes is available
+for testing. For this very purpose, a manual job is provided in the GitLab CI/CD
+pipeline that can be used to trigger a pipeline in the omnibus-gitlab repository
+that will create
+1. A deb package for Ubuntu 16.04, available as a build artifact, and
+2. A docker image, which is pushed to [Omnibus GitLab's container 
+registry](https://gitlab.com/gitlab-org/omnibus-gitlab/container_registry)
+(images titled `gitlab-ce` and `gitlab-ee` respectively and image tag is the
+commit which triggered the pipeline).
+
+When you push a commit to either the gitlab-ce or gitlab-ee project, the
+pipeline for that commit will have a `build-package` manual action you can
+trigger.
+
+## Specifying versions of components
+
+If you want to create a package from a specific branch, commit or tag of any of
+the GitLab components (like GitLab Workhorse, Gitaly, GitLab Pages, etc.), you
+can specify the branch name, commit sha or tag in the component's respective
+`*_VERSION` file. For example, if you want to build a package that uses the
+branch `0-1-stable`, modify the content of `GITALY_SERVER_VERSION` to
+`0-1-stable` and push the commit. This will create a manual job that can be
+used to trigger the build.
+
+## Specifying the branch in omnibus-gitlab repository
+
+In scenarios where a configuration change is to be introduced and omnibus-gitlab
+repository already has the necessary changes in a specific branch, you can build
+a package against that branch through an environment variable named
+`OMNIBUS_BRANCH`. To do this, specify that environment variable with the name of
+the branch as value in `.gitlab-ci.yml` and push a commit. This will create a
+manual job that can be used to trigger the build.
-- 
GitLab


From adcff298f8f3041faa29b75ee3711fb4ce1cbb69 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 10:43:56 +0200
Subject: [PATCH 266/363] fixed all issues - not doing bulk create.

---
 app/services/projects/propagate_service.rb       | 10 +++-------
 spec/services/projects/propagate_service_spec.rb | 12 ++++++++++++
 2 files changed, 15 insertions(+), 7 deletions(-)

diff --git a/app/services/projects/propagate_service.rb b/app/services/projects/propagate_service.rb
index b067fc2cd6492..716a620953719 100644
--- a/app/services/projects/propagate_service.rb
+++ b/app/services/projects/propagate_service.rb
@@ -21,16 +21,12 @@ def propagate
     private
 
     def propagate_projects_with_template
-      offset = 0
-
       loop do
-        batch = project_ids_batch(offset)
+        batch = project_ids_batch
 
         bulk_create_from_template(batch)
 
         break if batch.size < BATCH_SIZE
-
-        offset += BATCH_SIZE
       end
     end
 
@@ -44,7 +40,7 @@ def bulk_create_from_template(batch)
       end
     end
 
-    def project_ids_batch(offset)
+    def project_ids_batch
       Project.connection.execute(
         <<-SQL
           SELECT id
@@ -55,7 +51,7 @@ def project_ids_batch(offset)
             WHERE services.project_id = projects.id
             AND services.type = '#{@template.type}'
           )
-          LIMIT #{BATCH_SIZE} OFFSET #{offset}
+          LIMIT #{BATCH_SIZE}
       SQL
       ).to_a.flatten
     end
diff --git a/spec/services/projects/propagate_service_spec.rb b/spec/services/projects/propagate_service_spec.rb
index d4ec7c0b357bc..ac25c8b3d561b 100644
--- a/spec/services/projects/propagate_service_spec.rb
+++ b/spec/services/projects/propagate_service_spec.rb
@@ -66,5 +66,17 @@
 
       expect(service.properties).to eq(service_template.properties)
     end
+
+    describe 'bulk update' do
+      it 'creates services for all projects' do
+        project_total = 5
+        stub_const 'Projects::PropagateService::BATCH_SIZE', 3
+
+        project_total.times { create(:empty_project) }
+
+        expect { described_class.propagate(service_template) }.
+          to change { Service.count }.by(project_total + 1)
+      end
+    end
   end
 end
-- 
GitLab


From 9ec39568c5284f5a3a17a342d12f87befb6cfb4c Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 10:51:25 +0200
Subject: [PATCH 267/363] use select_values

---
 app/services/projects/propagate_service.rb | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/services/projects/propagate_service.rb b/app/services/projects/propagate_service.rb
index 716a620953719..f952b5108bb1f 100644
--- a/app/services/projects/propagate_service.rb
+++ b/app/services/projects/propagate_service.rb
@@ -41,7 +41,7 @@ def bulk_create_from_template(batch)
     end
 
     def project_ids_batch
-      Project.connection.execute(
+      Project.connection.select_values(
         <<-SQL
           SELECT id
           FROM projects
@@ -53,7 +53,7 @@ def project_ids_batch
           )
           LIMIT #{BATCH_SIZE}
       SQL
-      ).to_a.flatten
+      )
     end
 
     def service_hash
-- 
GitLab


From 6f7933b34947984bf46b6a1f4e9e878199bf48d3 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 5 May 2017 10:22:59 +0100
Subject: [PATCH 268/363] Fixed cycle analytics intro box styling regression

Closes #31844
---
 .../stylesheets/pages/cycle_analytics.scss    | 19 ++++++++++++++++
 .../projects/cycle_analytics/show.html.haml   | 22 +++++++++----------
 2 files changed, 30 insertions(+), 11 deletions(-)

diff --git a/app/assets/stylesheets/pages/cycle_analytics.scss b/app/assets/stylesheets/pages/cycle_analytics.scss
index 52bbb753af337..d29944207c5ba 100644
--- a/app/assets/stylesheets/pages/cycle_analytics.scss
+++ b/app/assets/stylesheets/pages/cycle_analytics.scss
@@ -3,6 +3,25 @@
   margin: 24px auto 0;
   position: relative;
 
+  .landing {
+    margin-top: 10px;
+
+    .inner-content {
+      white-space: normal;
+
+      h4,
+      p {
+        margin: 7px 0 0;
+        max-width: 480px;
+        padding: 0 $gl-padding;
+
+        @media (max-width: $screen-sm-min) {
+          margin: 0 auto;
+        }
+      }
+    }
+  }
+
   .col-headers {
     ul {
       margin: 0;
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index 819f29d3ca594..b158a81471cc4 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -9,17 +9,17 @@
 
 #cycle-analytics{ class: container_class, "v-cloak" => "true", data: { request_path: project_cycle_analytics_path(@project) } }
   - if @cycle_analytics_no_data
-    .bordered-box.landing.content-block{ "v-if" => "!isOverviewDialogDismissed" }
-      = icon("times", class: "dismiss-icon", "@click" => "dismissOverviewDialog()")
-      .row
-        .col-sm-3.col-xs-12.svg-container
-          = custom_icon('icon_cycle_analytics_splash')
-        .col-sm-8.col-xs-12.inner-content
-          %h4
-            {{ __('Introducing Cycle Analytics') }}
-          %p
-            {{ __('Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.') }}
-
+    .landing.content-block{ "v-if" => "!isOverviewDialogDismissed" }
+      %button.dismiss-button{ type: 'button', 'aria-label': 'Dismiss Cycle Analytics introduction box' }
+        = icon("times", "@click" => "dismissOverviewDialog()")
+      .svg-container
+        = custom_icon('icon_cycle_analytics_splash')
+      .inner-content
+        %h4
+          {{ __('Introducing Cycle Analytics') }}
+        %p
+          {{ __('Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.') }}
+        %p
           = link_to _('Read more'), help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn'
   = icon("spinner spin", "v-show" => "isLoading")
   .wrapper{ "v-show" => "!isLoading && !hasError" }
-- 
GitLab


From cba965eba49811e1e97909dbbbb16adf04c4a37d Mon Sep 17 00:00:00 2001
From: Pawel Chojnacki <pawel@chojnacki.ws>
Date: Fri, 5 May 2017 11:34:01 +0200
Subject: [PATCH 269/363] Set minimum latency to be non-negative number.

Sometimes the tests run so fast latency is calculated as 0.
This causes transient failures in our CI.
---
 .../lib/gitlab/health_checks/fs_shards_check_spec.rb | 12 ++++++------
 spec/lib/gitlab/health_checks/simple_check_shared.rb |  6 +++---
 2 files changed, 9 insertions(+), 9 deletions(-)

diff --git a/spec/lib/gitlab/health_checks/fs_shards_check_spec.rb b/spec/lib/gitlab/health_checks/fs_shards_check_spec.rb
index 4cd8cf313a54a..45ccd3d6459d6 100644
--- a/spec/lib/gitlab/health_checks/fs_shards_check_spec.rb
+++ b/spec/lib/gitlab/health_checks/fs_shards_check_spec.rb
@@ -82,9 +82,9 @@
         it { is_expected.to include(metric_class.new(:filesystem_readable, 0, shard: :default)) }
         it { is_expected.to include(metric_class.new(:filesystem_writable, 0, shard: :default)) }
 
-        it { is_expected.to include(have_attributes(name: :filesystem_access_latency, value: be > 0, labels: { shard: :default })) }
-        it { is_expected.to include(have_attributes(name: :filesystem_read_latency, value: be > 0, labels: { shard: :default })) }
-        it { is_expected.to include(have_attributes(name: :filesystem_write_latency, value: be > 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_access_latency, value: be >= 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_read_latency, value: be >= 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_write_latency, value: be >= 0, labels: { shard: :default })) }
       end
 
       context 'storage points to directory that has both read and write rights' do
@@ -96,9 +96,9 @@
         it { is_expected.to include(metric_class.new(:filesystem_readable, 1, shard: :default)) }
         it { is_expected.to include(metric_class.new(:filesystem_writable, 1, shard: :default)) }
 
-        it { is_expected.to include(have_attributes(name: :filesystem_access_latency, value: be > 0, labels: { shard: :default })) }
-        it { is_expected.to include(have_attributes(name: :filesystem_read_latency, value: be > 0, labels: { shard: :default })) }
-        it { is_expected.to include(have_attributes(name: :filesystem_write_latency, value: be > 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_access_latency, value: be >= 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_read_latency, value: be >= 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_write_latency, value: be >= 0, labels: { shard: :default })) }
       end
     end
   end
diff --git a/spec/lib/gitlab/health_checks/simple_check_shared.rb b/spec/lib/gitlab/health_checks/simple_check_shared.rb
index 1fa6d0faef983..3f871d6603442 100644
--- a/spec/lib/gitlab/health_checks/simple_check_shared.rb
+++ b/spec/lib/gitlab/health_checks/simple_check_shared.rb
@@ -8,7 +8,7 @@
 
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_success", value: 1)) }
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_timeout", value: 0)) }
-      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be > 0)) }
+      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be >= 0)) }
     end
 
     context 'Check is misbehaving' do
@@ -18,7 +18,7 @@
 
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_success", value: 0)) }
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_timeout", value: 0)) }
-      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be > 0)) }
+      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be >= 0)) }
     end
 
     context 'Check is timeouting' do
@@ -28,7 +28,7 @@
 
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_success", value: 0)) }
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_timeout", value: 1)) }
-      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be > 0)) }
+      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be >= 0)) }
     end
   end
 
-- 
GitLab


From 7af4c0c4f7dc66f7293a121d1c0e2caa8a5050c6 Mon Sep 17 00:00:00 2001
From: Sean McGivern <sean@gitlab.com>
Date: Fri, 5 May 2017 10:33:41 +0100
Subject: [PATCH 270/363] Fix email receiver metrics events

`Project#inspect` isn't very useful for examining usage of these features.
---
 lib/gitlab/email/handler/create_issue_handler.rb | 2 +-
 lib/gitlab/email/handler/create_note_handler.rb  | 2 +-
 lib/gitlab/email/handler/unsubscribe_handler.rb  | 2 +-
 3 files changed, 3 insertions(+), 3 deletions(-)

diff --git a/lib/gitlab/email/handler/create_issue_handler.rb b/lib/gitlab/email/handler/create_issue_handler.rb
index e7f91607e7e13..a616a80e8f5c3 100644
--- a/lib/gitlab/email/handler/create_issue_handler.rb
+++ b/lib/gitlab/email/handler/create_issue_handler.rb
@@ -37,7 +37,7 @@ def project
         end
 
         def metrics_params
-          super.merge(project: project)
+          super.merge(project: project&.full_path)
         end
 
         private
diff --git a/lib/gitlab/email/handler/create_note_handler.rb b/lib/gitlab/email/handler/create_note_handler.rb
index 31bb775c35789..31579e94a871d 100644
--- a/lib/gitlab/email/handler/create_note_handler.rb
+++ b/lib/gitlab/email/handler/create_note_handler.rb
@@ -29,7 +29,7 @@ def execute
         end
 
         def metrics_params
-          super.merge(project: project)
+          super.merge(project: project&.full_path)
         end
 
         private
diff --git a/lib/gitlab/email/handler/unsubscribe_handler.rb b/lib/gitlab/email/handler/unsubscribe_handler.rb
index df70a0633304a..5894384da5d44 100644
--- a/lib/gitlab/email/handler/unsubscribe_handler.rb
+++ b/lib/gitlab/email/handler/unsubscribe_handler.rb
@@ -20,7 +20,7 @@ def execute
         end
 
         def metrics_params
-          super.merge(project: project)
+          super.merge(project: project&.full_path)
         end
 
         private
-- 
GitLab


From f1f883501aa0035bbc744c7485a9517cb4561893 Mon Sep 17 00:00:00 2001
From: Lin Jen-Shin <godfat@godfat.org>
Date: Fri, 5 May 2017 10:09:54 +0000
Subject: [PATCH 271/363] Update CHANGELOG.md for 9.1.3

[ci skip]
---
 CHANGELOG.md | 11 +++++++++++
 1 file changed, 11 insertions(+)

diff --git a/CHANGELOG.md b/CHANGELOG.md
index 2686d778b098e..188aa73d16a0a 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -2,6 +2,17 @@
 documentation](doc/development/changelog.md) for instructions on adding your own
 entry.
 
+## 9.1.3 (2017-05-05)
+
+- Enforce project features when searching blobs and wikis.
+- Fixed branches dropdown rendering branch names as HTML.
+- Make Asciidoc & other markup go through pipeline to prevent XSS.
+- Validate URLs in markdown using URI to detect the host correctly.
+- Fix for XSS in project import view caused by Hamlit filter usage.
+- Sanitize submodule URLs before linking to them in the file tree view.
+- Refactor snippets finder & dont return internal snippets for external users.
+- Fix snippets visibility for show action - external users can not see internal snippets.
+
 ## 9.1.2 (2017-05-01)
 
 - Add index on ci_runners.contacted_at. !10876 (blackst0ne)
-- 
GitLab


From 444df931e7a0a7101c9f16cc7a20ea11094335d7 Mon Sep 17 00:00:00 2001
From: Lin Jen-Shin <godfat@godfat.org>
Date: Fri, 5 May 2017 10:16:44 +0000
Subject: [PATCH 272/363] Update CHANGELOG.md for 9.0.7

[ci skip]
---
 CHANGELOG.md | 12 ++++++++++++
 1 file changed, 12 insertions(+)

diff --git a/CHANGELOG.md b/CHANGELOG.md
index 188aa73d16a0a..e05b025ce2d6a 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -287,6 +287,18 @@ entry.
 - Only send chat notifications for the default branch.
 - Don't fill in the default kubernetes namespace.
 
+## 9.0.7 (2017-05-05)
+
+- Enforce project features when searching blobs and wikis.
+- Fixed branches dropdown rendering branch names as HTML.
+- Make Asciidoc & other markup go through pipeline to prevent XSS.
+- Validate URLs in markdown using URI to detect the host correctly.
+- Fix for XSS in project import view caused by Hamlit filter usage.
+- Sanitize submodule URLs before linking to them in the file tree view.
+- Refactor snippets finder & dont return internal snippets for external users.
+- Fix snippets visibility for show action - external users can not see internal snippets.
+- Do not show private groups on subgroups page if user doesn't have access to.
+
 ## 9.0.6 (2017-04-21)
 
 - Bugfix: POST /projects/:id/hooks and PUT /projects/:id/hook/:hook_id no longer ignore the the job_events param in the V4 API. !10586
-- 
GitLab


From 936367538043854c7b093b71ca315b8e469c55a4 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Fri, 5 May 2017 12:25:24 +0200
Subject: [PATCH 273/363] Use update build policy instead of new play policy

---
 app/policies/ci/build_policy.rb               | 12 +++++++-----
 app/policies/environment_policy.rb            |  2 +-
 app/serializers/build_action_entity.rb        |  2 +-
 app/serializers/build_entity.rb               |  2 +-
 app/services/ci/play_build_service.rb         |  2 +-
 app/views/projects/ci/builds/_build.html.haml |  2 +-
 lib/gitlab/ci/status/build/play.rb            |  2 +-
 spec/policies/ci/build_policy_spec.rb         | 18 +++++++++---------
 8 files changed, 22 insertions(+), 20 deletions(-)

diff --git a/app/policies/ci/build_policy.rb b/app/policies/ci/build_policy.rb
index 2c39d31488f05..d4af449060862 100644
--- a/app/policies/ci/build_policy.rb
+++ b/app/policies/ci/build_policy.rb
@@ -1,5 +1,7 @@
 module Ci
   class BuildPolicy < CommitStatusPolicy
+    alias_method :build, :subject
+
     def rules
       super
 
@@ -9,17 +11,17 @@ def rules
         cannot! :"#{rule}_commit_status" unless can? :"#{rule}_build"
       end
 
-      can! :play_build if can_play_action?
+      if can?(:update_build) && protected_action?
+        cannot! :update_build
+      end
     end
 
     private
 
-    alias_method :build, :subject
-
-    def can_play_action?
+    def protected_action?
       return false unless build.action?
 
-      ::Gitlab::UserAccess
+      !::Gitlab::UserAccess
         .new(user, project: build.project)
         .can_push_to_branch?(build.ref)
     end
diff --git a/app/policies/environment_policy.rb b/app/policies/environment_policy.rb
index cc94d4a7e058e..2fa15e645629b 100644
--- a/app/policies/environment_policy.rb
+++ b/app/policies/environment_policy.rb
@@ -12,6 +12,6 @@ def rules
   private
 
   def can_play_stop_action?
-    Ability.allowed?(user, :play_build, environment.stop_action)
+    Ability.allowed?(user, :update_build, environment.stop_action)
   end
 end
diff --git a/app/serializers/build_action_entity.rb b/app/serializers/build_action_entity.rb
index 3eff724ae9149..75dda1af709b1 100644
--- a/app/serializers/build_action_entity.rb
+++ b/app/serializers/build_action_entity.rb
@@ -19,6 +19,6 @@ class BuildActionEntity < Grape::Entity
   alias_method :build, :object
 
   def playable?
-    build.playable? && can?(request.user, :play_build, build)
+    build.playable? && can?(request.user, :update_build, build)
   end
 end
diff --git a/app/serializers/build_entity.rb b/app/serializers/build_entity.rb
index 10ba7c90c10c2..1380b347d8e96 100644
--- a/app/serializers/build_entity.rb
+++ b/app/serializers/build_entity.rb
@@ -26,7 +26,7 @@ class BuildEntity < Grape::Entity
   alias_method :build, :object
 
   def playable?
-    build.playable? && can?(request.user, :play_build, build)
+    build.playable? && can?(request.user, :update_build, build)
   end
 
   def detailed_status
diff --git a/app/services/ci/play_build_service.rb b/app/services/ci/play_build_service.rb
index 5b3ff1ced3855..e24f48c2d1654 100644
--- a/app/services/ci/play_build_service.rb
+++ b/app/services/ci/play_build_service.rb
@@ -1,7 +1,7 @@
 module Ci
   class PlayBuildService < ::BaseService
     def execute(build)
-      unless can?(current_user, :play_build, build)
+      unless can?(current_user, :update_build, build)
         raise Gitlab::Access::AccessDeniedError
       end
 
diff --git a/app/views/projects/ci/builds/_build.html.haml b/app/views/projects/ci/builds/_build.html.haml
index 2227d36eed22d..c001999617665 100644
--- a/app/views/projects/ci/builds/_build.html.haml
+++ b/app/views/projects/ci/builds/_build.html.haml
@@ -102,7 +102,7 @@
           = link_to cancel_namespace_project_build_path(job.project.namespace, job.project, job, return_to: request.original_url), method: :post, title: 'Cancel', class: 'btn btn-build' do
             = icon('remove', class: 'cred')
         - elsif allow_retry
-          - if job.playable? && !admin && can?(current_user, :play_build, job)
+          - if job.playable? && !admin && can?(current_user, :update_build, job)
             = link_to play_namespace_project_build_path(job.project.namespace, job.project, job, return_to: request.original_url), method: :post, title: 'Play', class: 'btn btn-build' do
               = custom_icon('icon_play')
           - elsif job.retryable?
diff --git a/lib/gitlab/ci/status/build/play.rb b/lib/gitlab/ci/status/build/play.rb
index 8130c7d1c90ef..fae34a2f9276a 100644
--- a/lib/gitlab/ci/status/build/play.rb
+++ b/lib/gitlab/ci/status/build/play.rb
@@ -14,7 +14,7 @@ def label
           end
 
           def has_action?
-            can?(user, :play_build, subject)
+            can?(user, :update_build, subject)
           end
 
           def action_icon
diff --git a/spec/policies/ci/build_policy_spec.rb b/spec/policies/ci/build_policy_spec.rb
index e4693cdcef0e0..3f4ce222b60fb 100644
--- a/spec/policies/ci/build_policy_spec.rb
+++ b/spec/policies/ci/build_policy_spec.rb
@@ -108,8 +108,8 @@
             create(:ci_build, :manual, ref: 'some-ref', pipeline: pipeline)
           end
 
-          it 'does not include ability to play build' do
-            expect(policies).not_to include :play_build
+          it 'does not include ability to update build' do
+            expect(policies).not_to include :update_build
           end
         end
 
@@ -118,8 +118,8 @@
             create(:ci_build, ref: 'some-ref', pipeline: pipeline)
           end
 
-          it 'does not include ability to play build' do
-            expect(policies).not_to include :play_build
+          it 'includes ability to update build' do
+            expect(policies).to include :update_build
           end
         end
       end
@@ -128,16 +128,16 @@
         context 'when build is a manual action' do
           let(:build) { create(:ci_build, :manual, pipeline: pipeline) }
 
-          it 'includes ability to play build' do
-            expect(policies).to include :play_build
+          it 'includes ability to update build' do
+            expect(policies).to include :update_build
           end
         end
 
         context 'when build is not a manual action' do
-          let(:build) { create(:ci_build,  pipeline: pipeline) }
+          let(:build) { create(:ci_build, pipeline: pipeline) }
 
-          it 'does not include ability to play build' do
-            expect(policies).not_to include :play_build
+          it 'includes ability to update build' do
+            expect(policies).to include :update_build
           end
         end
       end
-- 
GitLab


From 645593e5af1fe9e7fa345788aeaa90d2313d6486 Mon Sep 17 00:00:00 2001
From: Kushal Pandya <kushalspandya@gmail.com>
Date: Fri, 5 May 2017 10:57:29 +0000
Subject: [PATCH 274/363] Add instant comments support

---
 .../javascripts/behaviors/quick_submit.js     |   2 +-
 .../javascripts/lib/utils/common_utils.js     |   8 +
 app/assets/javascripts/notes.js               | 345 +++++++++++++++---
 .../stylesheets/framework/animations.scss     |  28 ++
 app/assets/stylesheets/pages/notes.scss       |  23 ++
 app/views/discussions/_notes.html.haml        |   1 +
 app/views/projects/notes/_edit_form.html.haml |   2 +-
 .../unreleased/27614-instant-comments.yml     |   4 +
 features/steps/project/merge_requests.rb      |   7 +
 features/steps/shared/note.rb                 |   4 +
 .../merge_requests/user_posts_notes_spec.rb   |   1 +
 .../user_uses_slash_commands_spec.rb          |   1 +
 .../lib/utils/common_utils_spec.js            |  11 +
 spec/javascripts/notes_spec.js                | 229 +++++++++++-
 ...issuable_slash_commands_shared_examples.rb |   1 +
 spec/support/time_tracking_shared_examples.rb |   5 +
 16 files changed, 607 insertions(+), 65 deletions(-)
 create mode 100644 changelogs/unreleased/27614-instant-comments.yml

diff --git a/app/assets/javascripts/behaviors/quick_submit.js b/app/assets/javascripts/behaviors/quick_submit.js
index 3d162b244135c..1f9e044808451 100644
--- a/app/assets/javascripts/behaviors/quick_submit.js
+++ b/app/assets/javascripts/behaviors/quick_submit.js
@@ -43,8 +43,8 @@ $(document).on('keydown.quick_submit', '.js-quick-submit', (e) => {
   const $submitButton = $form.find('input[type=submit], button[type=submit]');
 
   if (!$submitButton.attr('disabled')) {
+    $submitButton.trigger('click', [e]);
     $submitButton.disable();
-    $form.submit();
   }
 });
 
diff --git a/app/assets/javascripts/lib/utils/common_utils.js b/app/assets/javascripts/lib/utils/common_utils.js
index 8058672eaa90e..2f682fbd2fbf8 100644
--- a/app/assets/javascripts/lib/utils/common_utils.js
+++ b/app/assets/javascripts/lib/utils/common_utils.js
@@ -35,6 +35,14 @@
       });
     };
 
+    w.gl.utils.ajaxPost = function(url, data) {
+      return $.ajax({
+        type: 'POST',
+        url: url,
+        data: data,
+      });
+    };
+
     w.gl.utils.extractLast = function(term) {
       return this.split(term).pop();
     };
diff --git a/app/assets/javascripts/notes.js b/app/assets/javascripts/notes.js
index 87f03a40eba3f..72709f68070d9 100644
--- a/app/assets/javascripts/notes.js
+++ b/app/assets/javascripts/notes.js
@@ -26,12 +26,13 @@ const normalizeNewlines = function(str) {
 
   this.Notes = (function() {
     const MAX_VISIBLE_COMMIT_LIST_COUNT = 3;
+    const REGEX_SLASH_COMMANDS = /\/\w+/g;
 
     Notes.interval = null;
 
     function Notes(notes_url, note_ids, last_fetched_at, view) {
       this.updateTargetButtons = bind(this.updateTargetButtons, this);
-      this.updateCloseButton = bind(this.updateCloseButton, this);
+      this.updateComment = bind(this.updateComment, this);
       this.visibilityChange = bind(this.visibilityChange, this);
       this.cancelDiscussionForm = bind(this.cancelDiscussionForm, this);
       this.addDiffNote = bind(this.addDiffNote, this);
@@ -47,6 +48,7 @@ const normalizeNewlines = function(str) {
       this.refresh = bind(this.refresh, this);
       this.keydownNoteText = bind(this.keydownNoteText, this);
       this.toggleCommitList = bind(this.toggleCommitList, this);
+      this.postComment = bind(this.postComment, this);
 
       this.notes_url = notes_url;
       this.note_ids = note_ids;
@@ -82,28 +84,19 @@ const normalizeNewlines = function(str) {
     };
 
     Notes.prototype.addBinding = function() {
-      // add note to UI after creation
-      $(document).on("ajax:success", ".js-main-target-form", this.addNote);
-      $(document).on("ajax:success", ".js-discussion-note-form", this.addDiscussionNote);
-      // catch note ajax errors
-      $(document).on("ajax:error", ".js-main-target-form", this.addNoteError);
-      // change note in UI after update
-      $(document).on("ajax:success", "form.edit-note", this.updateNote);
       // Edit note link
       $(document).on("click", ".js-note-edit", this.showEditForm.bind(this));
       $(document).on("click", ".note-edit-cancel", this.cancelEdit);
       // Reopen and close actions for Issue/MR combined with note form submit
-      $(document).on("click", ".js-comment-button", this.updateCloseButton);
+      $(document).on("click", ".js-comment-submit-button", this.postComment);
+      $(document).on("click", ".js-comment-save-button", this.updateComment);
       $(document).on("keyup input", ".js-note-text", this.updateTargetButtons);
       // resolve a discussion
-      $(document).on('click', '.js-comment-resolve-button', this.resolveDiscussion);
+      $(document).on('click', '.js-comment-resolve-button', this.postComment);
       // remove a note (in general)
       $(document).on("click", ".js-note-delete", this.removeNote);
       // delete note attachment
       $(document).on("click", ".js-note-attachment-delete", this.removeAttachment);
-      // reset main target form after submit
-      $(document).on("ajax:complete", ".js-main-target-form", this.reenableTargetFormSubmitButton);
-      $(document).on("ajax:success", ".js-main-target-form", this.resetMainTargetForm);
       // reset main target form when clicking discard
       $(document).on("click", ".js-note-discard", this.resetMainTargetForm);
       // update the file name when an attachment is selected
@@ -120,20 +113,20 @@ const normalizeNewlines = function(str) {
       $(document).on("visibilitychange", this.visibilityChange);
       // when issue status changes, we need to refresh data
       $(document).on("issuable:change", this.refresh);
+      // ajax:events that happen on Form when actions like Reopen, Close are performed on Issues and MRs.
+      $(document).on("ajax:success", ".js-main-target-form", this.addNote);
+      $(document).on("ajax:success", ".js-discussion-note-form", this.addDiscussionNote);
+      $(document).on("ajax:success", ".js-main-target-form", this.resetMainTargetForm);
+      $(document).on("ajax:complete", ".js-main-target-form", this.reenableTargetFormSubmitButton);
       // when a key is clicked on the notes
       return $(document).on("keydown", ".js-note-text", this.keydownNoteText);
     };
 
     Notes.prototype.cleanBinding = function() {
-      $(document).off("ajax:success", ".js-main-target-form");
-      $(document).off("ajax:success", ".js-discussion-note-form");
-      $(document).off("ajax:success", "form.edit-note");
       $(document).off("click", ".js-note-edit");
       $(document).off("click", ".note-edit-cancel");
       $(document).off("click", ".js-note-delete");
       $(document).off("click", ".js-note-attachment-delete");
-      $(document).off("ajax:complete", ".js-main-target-form");
-      $(document).off("ajax:success", ".js-main-target-form");
       $(document).off("click", ".js-discussion-reply-button");
       $(document).off("click", ".js-add-diff-note-button");
       $(document).off("visibilitychange");
@@ -144,6 +137,9 @@ const normalizeNewlines = function(str) {
       $(document).off("keydown", ".js-note-text");
       $(document).off('click', '.js-comment-resolve-button');
       $(document).off("click", '.system-note-commit-list-toggler');
+      $(document).off("ajax:success", ".js-main-target-form");
+      $(document).off("ajax:success", ".js-discussion-note-form");
+      $(document).off("ajax:complete", ".js-main-target-form");
     };
 
     Notes.initCommentTypeToggle = function (form) {
@@ -276,12 +272,8 @@ const normalizeNewlines = function(str) {
       return this.initRefresh();
     };
 
-    Notes.prototype.handleCreateChanges = function(noteEntity) {
+    Notes.prototype.handleSlashCommands = function(noteEntity) {
       var votesBlock;
-      if (typeof noteEntity === 'undefined') {
-        return;
-      }
-
       if (noteEntity.commands_changes) {
         if ('merge' in noteEntity.commands_changes) {
           $.get(mrRefreshWidgetUrl);
@@ -556,24 +548,29 @@ const normalizeNewlines = function(str) {
     Adds new note to list.
      */
 
-    Notes.prototype.addNote = function(xhr, note, status) {
-      this.handleCreateChanges(note);
+    Notes.prototype.addNote = function($form, note) {
       return this.renderNote(note);
     };
 
-    Notes.prototype.addNoteError = function(xhr, note, status) {
-      return new Flash('Your comment could not be submitted! Please check your network connection and try again.', 'alert', this.parentTimeline);
+    Notes.prototype.addNoteError = ($form) => {
+      let formParentTimeline;
+      if ($form.hasClass('js-main-target-form')) {
+        formParentTimeline = $form.parents('.timeline');
+      } else if ($form.hasClass('js-discussion-note-form')) {
+        formParentTimeline = $form.closest('.discussion-notes').find('.notes');
+      }
+      return new Flash('Your comment could not be submitted! Please check your network connection and try again.', 'alert', formParentTimeline);
     };
 
+    Notes.prototype.updateNoteError = $parentTimeline => new Flash('Your comment could not be updated! Please check your network connection and try again.');
+
     /*
     Called in response to the new note form being submitted
 
     Adds new note to list.
      */
 
-    Notes.prototype.addDiscussionNote = function(xhr, note, status) {
-      var $form = $(xhr.target);
-
+    Notes.prototype.addDiscussionNote = function($form, note, isNewDiffComment) {
       if ($form.attr('data-resolve-all') != null) {
         var projectPath = $form.data('project-path');
         var discussionId = $form.data('discussion-id');
@@ -586,7 +583,9 @@ const normalizeNewlines = function(str) {
 
       this.renderNote(note, $form);
       // cleanup after successfully creating a diff/discussion note
-      this.removeDiscussionNoteForm($form);
+      if (isNewDiffComment) {
+        this.removeDiscussionNoteForm($form);
+      }
     };
 
     /*
@@ -596,17 +595,18 @@ const normalizeNewlines = function(str) {
      */
 
     Notes.prototype.updateNote = function(_xhr, noteEntity, _status) {
-      var $html, $note_li;
+      var $noteEntityEl, $note_li;
       // Convert returned HTML to a jQuery object so we can modify it further
-      $html = $(noteEntity.html);
+      $noteEntityEl = $(noteEntity.html);
+      $noteEntityEl.addClass('fade-in-full');
       this.revertNoteEditForm();
-      gl.utils.localTimeAgo($('.js-timeago', $html));
-      $html.renderGFM();
-      $html.find('.js-task-list-container').taskList('enable');
+      gl.utils.localTimeAgo($('.js-timeago', $noteEntityEl));
+      $noteEntityEl.renderGFM();
+      $noteEntityEl.find('.js-task-list-container').taskList('enable');
       // Find the note's `li` element by ID and replace it with the updated HTML
       $note_li = $('.note-row-' + noteEntity.id);
 
-      $note_li.replaceWith($html);
+      $note_li.replaceWith($noteEntityEl);
 
       if (typeof gl.diffNotesCompileComponents !== 'undefined') {
         gl.diffNotesCompileComponents();
@@ -698,7 +698,7 @@ const normalizeNewlines = function(str) {
       var $editForm = $(selector);
 
       $editForm.insertBefore('.notes-form');
-      $editForm.find('.js-comment-button').enable();
+      $editForm.find('.js-comment-save-button').enable();
       $editForm.find('.js-finish-edit-warning').hide();
     };
 
@@ -982,14 +982,6 @@ const normalizeNewlines = function(str) {
       return this.refresh();
     };
 
-    Notes.prototype.updateCloseButton = function(e) {
-      var closebtn, form, textarea;
-      textarea = $(e.target);
-      form = textarea.parents('form');
-      closebtn = form.find('.js-note-target-close');
-      return closebtn.text(closebtn.data('original-text'));
-    };
-
     Notes.prototype.updateTargetButtons = function(e) {
       var closebtn, closetext, discardbtn, form, reopenbtn, reopentext, textarea;
       textarea = $(e.target);
@@ -1078,17 +1070,6 @@ const normalizeNewlines = function(str) {
       return this.notesCountBadge.text(parseInt(this.notesCountBadge.text(), 10) + updateCount);
     };
 
-    Notes.prototype.resolveDiscussion = function() {
-      var $this = $(this);
-      var discussionId = $this.attr('data-discussion-id');
-
-      $this
-        .closest('form')
-        .attr('data-discussion-id', discussionId)
-        .attr('data-resolve-all', 'true')
-        .attr('data-project-path', $this.attr('data-project-path'));
-    };
-
     Notes.prototype.toggleCommitList = function(e) {
       const $element = $(e.currentTarget);
       const $closestSystemCommitList = $element.siblings('.system-note-commit-list');
@@ -1137,7 +1118,7 @@ const normalizeNewlines = function(str) {
     Notes.animateAppendNote = function(noteHtml, $notesList) {
       const $note = $(noteHtml);
 
-      $note.addClass('fade-in').renderGFM();
+      $note.addClass('fade-in-full').renderGFM();
       $notesList.append($note);
       return $note;
     };
@@ -1150,6 +1131,254 @@ const normalizeNewlines = function(str) {
       return $updatedNote;
     };
 
+    /**
+     * Get data from Form attributes to use for saving/submitting comment.
+     */
+    Notes.prototype.getFormData = function($form) {
+      return {
+        formData: $form.serialize(),
+        formContent: $form.find('.js-note-text').val(),
+        formAction: $form.attr('action'),
+      };
+    };
+
+    /**
+     * Identify if comment has any slash commands
+     */
+    Notes.prototype.hasSlashCommands = function(formContent) {
+      return REGEX_SLASH_COMMANDS.test(formContent);
+    };
+
+    /**
+     * Remove slash commands and leave comment with pure message
+     */
+    Notes.prototype.stripSlashCommands = function(formContent) {
+      return formContent.replace(REGEX_SLASH_COMMANDS, '').trim();
+    };
+
+    /**
+     * Create placeholder note DOM element populated with comment body
+     * that we will show while comment is being posted.
+     * Once comment is _actually_ posted on server, we will have final element
+     * in response that we will show in place of this temporary element.
+     */
+    Notes.prototype.createPlaceholderNote = function({ formContent, uniqueId, isDiscussionNote, currentUsername, currentUserFullname }) {
+      const discussionClass = isDiscussionNote ? 'discussion' : '';
+      const $tempNote = $(
+        `<li id="${uniqueId}" class="note being-posted fade-in-half timeline-entry">
+           <div class="timeline-entry-inner">
+              <div class="timeline-icon">
+                 <a href="/${currentUsername}"><span class="dummy-avatar"></span></a>
+              </div>
+              <div class="timeline-content ${discussionClass}">
+                 <div class="note-header">
+                    <div class="note-header-info">
+                       <a href="/${currentUsername}">
+                         <span class="hidden-xs">${currentUserFullname}</span>
+                         <span class="note-headline-light">@${currentUsername}</span>
+                       </a>
+                       <span class="note-headline-light">
+                          <i class="fa fa-spinner fa-spin" aria-label="Comment is being posted" aria-hidden="true"></i>
+                       </span>
+                    </div>
+                 </div>
+                 <div class="note-body">
+                   <div class="note-text">
+                     <p>${formContent}</p>
+                   </div>
+                 </div>
+              </div>
+           </div>
+        </li>`
+      );
+
+      return $tempNote;
+    };
+
+    /**
+     * This method does following tasks step-by-step whenever a new comment
+     * is submitted by user (both main thread comments as well as discussion comments).
+     *
+     * 1) Get Form metadata
+     * 2) Identify comment type; a) Main thread b) Discussion thread c) Discussion resolve
+     * 3) Build temporary placeholder element (using `createPlaceholderNote`)
+     * 4) Show placeholder note on UI
+     * 5) Perform network request to submit the note using `gl.utils.ajaxPost`
+     *    a) If request is successfully completed
+     *        1. Remove placeholder element
+     *        2. Show submitted Note element
+     *        3. Perform post-submit errands
+     *           a. Mark discussion as resolved if comment submission was for resolve.
+     *           b. Reset comment form to original state.
+     *    b) If request failed
+     *        1. Remove placeholder element
+     *        2. Show error Flash message about failure
+     */
+    Notes.prototype.postComment = function(e) {
+      e.preventDefault();
+
+      // Get Form metadata
+      const $submitBtn = $(e.target);
+      let $form = $submitBtn.parents('form');
+      const $closeBtn = $form.find('.js-note-target-close');
+      const isDiscussionNote = $submitBtn.parent().find('li.droplab-item-selected').attr('id') === 'discussion';
+      const isMainForm = $form.hasClass('js-main-target-form');
+      const isDiscussionForm = $form.hasClass('js-discussion-note-form');
+      const isDiscussionResolve = $submitBtn.hasClass('js-comment-resolve-button');
+      const { formData, formContent, formAction } = this.getFormData($form);
+      const uniqueId = _.uniqueId('tempNote_');
+      let $notesContainer;
+      let tempFormContent;
+
+      // Get reference to notes container based on type of comment
+      if (isDiscussionForm) {
+        $notesContainer = $form.parent('.discussion-notes').find('.notes');
+      } else if (isMainForm) {
+        $notesContainer = $('ul.main-notes-list');
+      }
+
+      // If comment is to resolve discussion, disable submit buttons while
+      // comment posting is finished.
+      if (isDiscussionResolve) {
+        $submitBtn.disable();
+        $form.find('.js-comment-submit-button').disable();
+      }
+
+      tempFormContent = formContent;
+      if (this.hasSlashCommands(formContent)) {
+        tempFormContent = this.stripSlashCommands(formContent);
+      }
+
+      if (tempFormContent) {
+        // Show placeholder note
+        $notesContainer.append(this.createPlaceholderNote({
+          formContent: tempFormContent,
+          uniqueId,
+          isDiscussionNote,
+          currentUsername: gon.current_username,
+          currentUserFullname: gon.current_user_fullname,
+        }));
+      }
+
+      // Clear the form textarea
+      if ($notesContainer.length) {
+        if (isMainForm) {
+          this.resetMainTargetForm(e);
+        } else if (isDiscussionForm) {
+          this.removeDiscussionNoteForm($form);
+        }
+      }
+
+      /* eslint-disable promise/catch-or-return */
+      // Make request to submit comment on server
+      gl.utils.ajaxPost(formAction, formData)
+        .then((note) => {
+          // Submission successful! remove placeholder
+          $notesContainer.find(`#${uniqueId}`).remove();
+
+          // Check if this was discussion comment
+          if (isDiscussionForm) {
+            // Remove flash-container
+            $notesContainer.find('.flash-container').remove();
+
+            // If comment intends to resolve discussion, do the same.
+            if (isDiscussionResolve) {
+              $form
+                .attr('data-discussion-id', $submitBtn.data('discussion-id'))
+                .attr('data-resolve-all', 'true')
+                .attr('data-project-path', $submitBtn.data('project-path'));
+            }
+
+            // Show final note element on UI
+            this.addDiscussionNote($form, note, $notesContainer.length === 0);
+
+            // append flash-container to the Notes list
+            if ($notesContainer.length) {
+              $notesContainer.append('<div class="flash-container" style="display: none;"></div>');
+            }
+          } else if (isMainForm) { // Check if this was main thread comment
+            // Show final note element on UI and perform form and action buttons cleanup
+            this.addNote($form, note);
+            this.reenableTargetFormSubmitButton(e);
+          }
+
+          if (note.commands_changes) {
+            this.handleSlashCommands(note);
+          }
+
+          $form.trigger('ajax:success', [note]);
+        }).fail(() => {
+          // Submission failed, remove placeholder note and show Flash error message
+          $notesContainer.find(`#${uniqueId}`).remove();
+
+          // Show form again on UI on failure
+          if (isDiscussionForm && $notesContainer.length) {
+            const replyButton = $notesContainer.parent().find('.js-discussion-reply-button');
+            $.proxy(this.replyToDiscussionNote, replyButton[0], { target: replyButton[0] }).call();
+            $form = $notesContainer.parent().find('form');
+          }
+
+          $form.find('.js-note-text').val(formContent);
+          this.reenableTargetFormSubmitButton(e);
+          this.addNoteError($form);
+        });
+
+      return $closeBtn.text($closeBtn.data('original-text'));
+    };
+
+    /**
+     * This method does following tasks step-by-step whenever an existing comment
+     * is updated by user (both main thread comments as well as discussion comments).
+     *
+     * 1) Get Form metadata
+     * 2) Update note element with new content
+     * 3) Perform network request to submit the updated note using `gl.utils.ajaxPost`
+     *    a) If request is successfully completed
+     *        1. Show submitted Note element
+     *    b) If request failed
+     *        1. Revert Note element to original content
+     *        2. Show error Flash message about failure
+     */
+    Notes.prototype.updateComment = function(e) {
+      e.preventDefault();
+
+      // Get Form metadata
+      const $submitBtn = $(e.target);
+      const $form = $submitBtn.parents('form');
+      const $closeBtn = $form.find('.js-note-target-close');
+      const $editingNote = $form.parents('.note.is-editing');
+      const $noteBody = $editingNote.find('.js-task-list-container');
+      const $noteBodyText = $noteBody.find('.note-text');
+      const { formData, formContent, formAction } = this.getFormData($form);
+
+      // Cache original comment content
+      const cachedNoteBodyText = $noteBodyText.html();
+
+      // Show updated comment content temporarily
+      $noteBodyText.html(formContent);
+      $editingNote.removeClass('is-editing').addClass('being-posted fade-in-half');
+      $editingNote.find('.note-headline-meta a').html('<i class="fa fa-spinner fa-spin" aria-label="Comment is being updated" aria-hidden="true"></i>');
+
+      /* eslint-disable promise/catch-or-return */
+      // Make request to update comment on server
+      gl.utils.ajaxPost(formAction, formData)
+        .then((note) => {
+          // Submission successful! render final note element
+          this.updateNote(null, note, null);
+        })
+        .fail(() => {
+          // Submission failed, revert back to original note
+          $noteBodyText.html(cachedNoteBodyText);
+          $editingNote.removeClass('being-posted fade-in');
+          $editingNote.find('.fa.fa-spinner').remove();
+
+          // Show Flash message about failure
+          this.updateNoteError();
+        });
+
+      return $closeBtn.text($closeBtn.data('original-text'));
+    };
+
     return Notes;
   })();
 }).call(window);
diff --git a/app/assets/stylesheets/framework/animations.scss b/app/assets/stylesheets/framework/animations.scss
index 7c50b80fd2bb5..3cd7f81da4719 100644
--- a/app/assets/stylesheets/framework/animations.scss
+++ b/app/assets/stylesheets/framework/animations.scss
@@ -159,3 +159,31 @@ a {
 .fade-in {
   animation: fadeIn $fade-in-duration 1;
 }
+
+@keyframes fadeInHalf {
+  0% {
+    opacity: 0;
+  }
+
+  100% {
+    opacity: 0.5;
+  }
+}
+
+.fade-in-half {
+  animation: fadeInHalf $fade-in-duration 1;
+}
+
+@keyframes fadeInFull {
+  0% {
+    opacity: 0.5;
+  }
+
+  100% {
+    opacity: 1;
+  }
+}
+
+.fade-in-full {
+  animation: fadeInFull $fade-in-duration 1;
+}
diff --git a/app/assets/stylesheets/pages/notes.scss b/app/assets/stylesheets/pages/notes.scss
index f89150ebead7e..cfea52c6e5742 100644
--- a/app/assets/stylesheets/pages/notes.scss
+++ b/app/assets/stylesheets/pages/notes.scss
@@ -57,6 +57,25 @@ ul.notes {
     position: relative;
     border-bottom: 1px solid $white-normal;
 
+    &.being-posted {
+      pointer-events: none;
+      opacity: 0.5;
+
+      .dummy-avatar {
+        display: inline-block;
+        height: 40px;
+        width: 40px;
+        border-radius: 50%;
+        background-color: $kdb-border;
+        border: 1px solid darken($kdb-border, 25%);
+      }
+
+      .note-headline-light,
+      .fa-spinner {
+        margin-left: 3px;
+      }
+    }
+
     &.note-discussion {
       &.timeline-entry {
         padding: 14px 10px;
@@ -687,6 +706,10 @@ ul.notes {
   }
 }
 
+.discussion-notes .flash-container {
+  margin-bottom: 0;
+}
+
 // Merge request notes in diffs
 .diff-file {
   // Diff is side by side
diff --git a/app/views/discussions/_notes.html.haml b/app/views/discussions/_notes.html.haml
index 964473ee3e08c..7ba3f3f6c42a5 100644
--- a/app/views/discussions/_notes.html.haml
+++ b/app/views/discussions/_notes.html.haml
@@ -1,6 +1,7 @@
 .discussion-notes
   %ul.notes{ data: { discussion_id: discussion.id } }
     = render partial: "shared/notes/note", collection: discussion.notes, as: :note
+    .flash-container
 
   - if current_user
     .discussion-reply-holder
diff --git a/app/views/projects/notes/_edit_form.html.haml b/app/views/projects/notes/_edit_form.html.haml
index 00230b0bdf822..3867072225f88 100644
--- a/app/views/projects/notes/_edit_form.html.haml
+++ b/app/views/projects/notes/_edit_form.html.haml
@@ -9,6 +9,6 @@
     .note-form-actions.clearfix
       .settings-message.note-edit-warning.js-finish-edit-warning
         Finish editing this message first!
-      = submit_tag 'Save comment', class: 'btn btn-nr btn-save js-comment-button'
+      = submit_tag 'Save comment', class: 'btn btn-nr btn-save js-comment-save-button'
       %button.btn.btn-nr.btn-cancel.note-edit-cancel{ type: 'button' }
         Cancel
diff --git a/changelogs/unreleased/27614-instant-comments.yml b/changelogs/unreleased/27614-instant-comments.yml
new file mode 100644
index 0000000000000..7b2592f46ede2
--- /dev/null
+++ b/changelogs/unreleased/27614-instant-comments.yml
@@ -0,0 +1,4 @@
+---
+title: Add support for instantly updating comments
+merge_request: 10760
+author:
diff --git a/features/steps/project/merge_requests.rb b/features/steps/project/merge_requests.rb
index a06b2f2911f11..4b7d6cd840b14 100644
--- a/features/steps/project/merge_requests.rb
+++ b/features/steps/project/merge_requests.rb
@@ -458,6 +458,8 @@ class Spinach::Features::ProjectMergeRequests < Spinach::FeatureSteps
       click_button "Comment"
     end
 
+    wait_for_ajax
+
     page.within ".files>div:nth-child(2) .note-body > .note-text" do
       expect(page).to have_content "Line is correct"
     end
@@ -470,6 +472,8 @@ class Spinach::Features::ProjectMergeRequests < Spinach::FeatureSteps
       fill_in "note_note", with: "Line is wrong on here"
       click_button "Comment"
     end
+
+    wait_for_ajax
   end
 
   step 'I should still see a comment like "Line is correct" in the second file' do
@@ -574,6 +578,9 @@ def leave_comment(message)
       fill_in "note_note", with: message
       click_button "Comment"
     end
+
+    wait_for_ajax
+
     page.within(".notes_holder", visible: true) do
       expect(page).to have_content message
     end
diff --git a/features/steps/shared/note.rb b/features/steps/shared/note.rb
index 7885cc7ab7729..7d260025052cf 100644
--- a/features/steps/shared/note.rb
+++ b/features/steps/shared/note.rb
@@ -24,6 +24,8 @@ module SharedNote
       fill_in "note[note]", with: "XML attached"
       click_button "Comment"
     end
+
+    wait_for_ajax
   end
 
   step 'I preview a comment text like "Bug fixed :smile:"' do
@@ -37,6 +39,8 @@ module SharedNote
     page.within(".js-main-target-form") do
       click_button "Comment"
     end
+
+    wait_for_ajax
   end
 
   step 'I write a comment like ":+1: Nice"' do
diff --git a/spec/features/merge_requests/user_posts_notes_spec.rb b/spec/features/merge_requests/user_posts_notes_spec.rb
index c7cc4d6bc724e..7fc0e2ce6eca4 100644
--- a/spec/features/merge_requests/user_posts_notes_spec.rb
+++ b/spec/features/merge_requests/user_posts_notes_spec.rb
@@ -98,6 +98,7 @@
           find('.btn-save').click
         end
 
+        wait_for_ajax
         find('.note').hover
         find('.js-note-edit').click
 
diff --git a/spec/features/merge_requests/user_uses_slash_commands_spec.rb b/spec/features/merge_requests/user_uses_slash_commands_spec.rb
index 1c0f21e5616b5..f0ad57eb92fd8 100644
--- a/spec/features/merge_requests/user_uses_slash_commands_spec.rb
+++ b/spec/features/merge_requests/user_uses_slash_commands_spec.rb
@@ -160,6 +160,7 @@
         it 'changes target branch from a note' do
           write_note("message start \n/target_branch merge-test\n message end.")
 
+          wait_for_ajax
           expect(page).not_to have_content('/target_branch')
           expect(page).to have_content('message start')
           expect(page).to have_content('message end.')
diff --git a/spec/javascripts/lib/utils/common_utils_spec.js b/spec/javascripts/lib/utils/common_utils_spec.js
index a00efa10119ce..5eb147ed88806 100644
--- a/spec/javascripts/lib/utils/common_utils_spec.js
+++ b/spec/javascripts/lib/utils/common_utils_spec.js
@@ -362,5 +362,16 @@ require('~/lib/utils/common_utils');
         gl.utils.setCiStatusFavicon(BUILD_URL);
       });
     });
+
+    describe('gl.utils.ajaxPost', () => {
+      it('should perform `$.ajax` call and do `POST` request', () => {
+        const requestURL = '/some/random/api';
+        const data = { keyname: 'value' };
+        const ajaxSpy = spyOn($, 'ajax').and.callFake(() => {});
+
+        gl.utils.ajaxPost(requestURL, data);
+        expect(ajaxSpy.calls.allArgs()[0][0].type).toEqual('POST');
+      });
+    });
   });
 })();
diff --git a/spec/javascripts/notes_spec.js b/spec/javascripts/notes_spec.js
index cdc5c4510ffcd..7bffa90ab145e 100644
--- a/spec/javascripts/notes_spec.js
+++ b/spec/javascripts/notes_spec.js
@@ -26,7 +26,7 @@ import '~/notes';
 
     describe('task lists', function() {
       beforeEach(function() {
-        $('form').on('submit', function(e) {
+        $('.js-comment-button').on('click', function(e) {
           e.preventDefault();
         });
         this.notes = new Notes();
@@ -60,9 +60,12 @@ import '~/notes';
           reset: function() {}
         });
 
-        $('form').on('submit', function(e) {
+        $('.js-comment-button').on('click', (e) => {
+          const $form = $(this);
           e.preventDefault();
-          $('.js-main-target-form').trigger('ajax:success');
+          this.notes.addNote($form);
+          this.notes.reenableTargetFormSubmitButton(e);
+          this.notes.resetMainTargetForm(e);
         });
       });
 
@@ -238,8 +241,8 @@ import '~/notes';
         $resultantNote = Notes.animateAppendNote(noteHTML, $notesList);
       });
 
-      it('should have `fade-in` class', () => {
-        expect($resultantNote.hasClass('fade-in')).toEqual(true);
+      it('should have `fade-in-full` class', () => {
+        expect($resultantNote.hasClass('fade-in-full')).toEqual(true);
       });
 
       it('should append note to the notes list', () => {
@@ -269,5 +272,221 @@ import '~/notes';
         expect($note.replaceWith).toHaveBeenCalledWith($updatedNote);
       });
     });
+
+    describe('getFormData', () => {
+      it('should return form metadata object from form reference', () => {
+        this.notes = new Notes();
+
+        const $form = $('form');
+        const sampleComment = 'foobar';
+        $form.find('textarea.js-note-text').val(sampleComment);
+        const { formData, formContent, formAction } = this.notes.getFormData($form);
+
+        expect(formData.indexOf(sampleComment) > -1).toBe(true);
+        expect(formContent).toEqual(sampleComment);
+        expect(formAction).toEqual($form.attr('action'));
+      });
+    });
+
+    describe('hasSlashCommands', () => {
+      beforeEach(() => {
+        this.notes = new Notes();
+      });
+
+      it('should return true when comment has slash commands', () => {
+        const sampleComment = '/wip /milestone %1.0 /merge /unassign Merging this';
+        const hasSlashCommands = this.notes.hasSlashCommands(sampleComment);
+
+        expect(hasSlashCommands).toBeTruthy();
+      });
+
+      it('should return false when comment does NOT have any slash commands', () => {
+        const sampleComment = 'Looking good, Awesome!';
+        const hasSlashCommands = this.notes.hasSlashCommands(sampleComment);
+
+        expect(hasSlashCommands).toBeFalsy();
+      });
+    });
+
+    describe('stripSlashCommands', () => {
+      const REGEX_SLASH_COMMANDS = /\/\w+/g;
+
+      it('should strip slash commands from the comment', () => {
+        this.notes = new Notes();
+        const sampleComment = '/wip /milestone %1.0 /merge /unassign Merging this';
+        const stripedComment = this.notes.stripSlashCommands(sampleComment);
+
+        expect(REGEX_SLASH_COMMANDS.test(stripedComment)).toBeFalsy();
+      });
+    });
+
+    describe('createPlaceholderNote', () => {
+      const sampleComment = 'foobar';
+      const uniqueId = 'b1234-a4567';
+      const currentUsername = 'root';
+      const currentUserFullname = 'Administrator';
+
+      beforeEach(() => {
+        this.notes = new Notes();
+      });
+
+      it('should return constructed placeholder element for regular note based on form contents', () => {
+        const $tempNote = this.notes.createPlaceholderNote({
+          formContent: sampleComment,
+          uniqueId,
+          isDiscussionNote: false,
+          currentUsername,
+          currentUserFullname
+        });
+        const $tempNoteHeader = $tempNote.find('.note-header');
+
+        expect($tempNote.prop('nodeName')).toEqual('LI');
+        expect($tempNote.attr('id')).toEqual(uniqueId);
+        $tempNote.find('.timeline-icon > a, .note-header-info > a').each(function() {
+          expect($(this).attr('href')).toEqual(`/${currentUsername}`);
+        });
+        expect($tempNote.find('.timeline-content').hasClass('discussion')).toBeFalsy();
+        expect($tempNoteHeader.find('.hidden-xs').text().trim()).toEqual(currentUserFullname);
+        expect($tempNoteHeader.find('.note-headline-light').text().trim()).toEqual(`@${currentUsername}`);
+        expect($tempNote.find('.note-body .note-text').text().trim()).toEqual(sampleComment);
+      });
+
+      it('should return constructed placeholder element for discussion note based on form contents', () => {
+        const $tempNote = this.notes.createPlaceholderNote({
+          formContent: sampleComment,
+          uniqueId,
+          isDiscussionNote: true,
+          currentUsername,
+          currentUserFullname
+        });
+
+        expect($tempNote.prop('nodeName')).toEqual('LI');
+        expect($tempNote.find('.timeline-content').hasClass('discussion')).toBeTruthy();
+      });
+    });
+
+    describe('postComment & updateComment', () => {
+      const sampleComment = 'foo';
+      const updatedComment = 'bar';
+      const note = {
+        id: 1234,
+        html: `<li class="note note-row-1234 timeline-entry" id="note_1234">
+                <div class="note-text">${sampleComment}</div>
+               </li>`,
+        note: sampleComment,
+        valid: true
+      };
+      let $form;
+      let $notesContainer;
+
+      beforeEach(() => {
+        this.notes = new Notes();
+        window.gon.current_username = 'root';
+        window.gon.current_user_fullname = 'Administrator';
+        $form = $('form');
+        $notesContainer = $('ul.main-notes-list');
+        $form.find('textarea.js-note-text').val(sampleComment);
+        $('.js-comment-button').click();
+      });
+
+      it('should show placeholder note while new comment is being posted', () => {
+        expect($notesContainer.find('.note.being-posted').length > 0).toEqual(true);
+      });
+
+      it('should remove placeholder note when new comment is done posting', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.success(note);
+          expect($notesContainer.find('.note.being-posted').length).toEqual(0);
+        });
+      });
+
+      it('should show actual note element when new comment is done posting', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.success(note);
+          expect($notesContainer.find(`#${note.id}`).length > 0).toEqual(true);
+        });
+      });
+
+      it('should reset Form when new comment is done posting', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.success(note);
+          expect($form.find('textarea.js-note-text')).toEqual('');
+        });
+      });
+
+      it('should trigger ajax:success event on Form when new comment is done posting', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.success(note);
+          spyOn($form, 'trigger');
+          expect($form.trigger).toHaveBeenCalledWith('ajax:success', [note]);
+        });
+      });
+
+      it('should show flash error message when new comment failed to be posted', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.error();
+          expect($notesContainer.parent().find('.flash-container .flash-text').is(':visible')).toEqual(true);
+        });
+      });
+
+      it('should refill form textarea with original comment content when new comment failed to be posted', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.error();
+          expect($form.find('textarea.js-note-text')).toEqual(sampleComment);
+        });
+      });
+
+      it('should show updated comment as _actively being posted_ while comment being updated', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.success(note);
+          const $noteEl = $notesContainer.find(`#note_${note.id}`);
+          $noteEl.find('.js-note-edit').click();
+          $noteEl.find('textarea.js-note-text').val(updatedComment);
+          $noteEl.find('.js-comment-save-button').click();
+          expect($noteEl.hasClass('.being-posted')).toEqual(true);
+          expect($noteEl.find('.note-text').text()).toEqual(updatedComment);
+        });
+      });
+
+      it('should show updated comment when comment update is done posting', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.success(note);
+          const $noteEl = $notesContainer.find(`#note_${note.id}`);
+          $noteEl.find('.js-note-edit').click();
+          $noteEl.find('textarea.js-note-text').val(updatedComment);
+          $noteEl.find('.js-comment-save-button').click();
+
+          spyOn($, 'ajax').and.callFake((updateOptions) => {
+            const updatedNote = Object.assign({}, note);
+            updatedNote.note = updatedComment;
+            updatedNote.html = `<li class="note note-row-1234 timeline-entry" id="note_1234">
+                                  <div class="note-text">${updatedComment}</div>
+                                </li>`;
+            updateOptions.success(updatedNote);
+            const $updatedNoteEl = $notesContainer.find(`#note_${updatedNote.id}`);
+            expect($updatedNoteEl.hasClass('.being-posted')).toEqual(false); // Remove being-posted visuals
+            expect($updatedNoteEl.find('note-text').text().trim()).toEqual(updatedComment); // Verify if comment text updated
+          });
+        });
+      });
+
+      it('should show flash error message when comment failed to be updated', () => {
+        spyOn($, 'ajax').and.callFake((options) => {
+          options.success(note);
+          const $noteEl = $notesContainer.find(`#note_${note.id}`);
+          $noteEl.find('.js-note-edit').click();
+          $noteEl.find('textarea.js-note-text').val(updatedComment);
+          $noteEl.find('.js-comment-save-button').click();
+
+          spyOn($, 'ajax').and.callFake((updateOptions) => {
+            updateOptions.error();
+            const $updatedNoteEl = $notesContainer.find(`#note_${note.id}`);
+            expect($updatedNoteEl.hasClass('.being-posted')).toEqual(false); // Remove being-posted visuals
+            expect($updatedNoteEl.find('note-text').text().trim()).toEqual(sampleComment); // See if comment reverted back to original
+            expect($notesContainer.parent().find('.flash-container .flash-text').is(':visible')).toEqual(true); // Flash error message shown
+          });
+        });
+      });
+    });
   });
 }).call(window);
diff --git a/spec/support/features/issuable_slash_commands_shared_examples.rb b/spec/support/features/issuable_slash_commands_shared_examples.rb
index 6efcdd7a1ded9..610decdcddbaf 100644
--- a/spec/support/features/issuable_slash_commands_shared_examples.rb
+++ b/spec/support/features/issuable_slash_commands_shared_examples.rb
@@ -58,6 +58,7 @@
         expect(page).not_to have_content '/label ~bug'
         expect(page).not_to have_content '/milestone %"ASAP"'
 
+        wait_for_ajax
         issuable.reload
         note = issuable.notes.user.first
 
diff --git a/spec/support/time_tracking_shared_examples.rb b/spec/support/time_tracking_shared_examples.rb
index 01bc80f957e24..e94e17da7e588 100644
--- a/spec/support/time_tracking_shared_examples.rb
+++ b/spec/support/time_tracking_shared_examples.rb
@@ -8,6 +8,7 @@
   it 'updates the sidebar component when estimate is added' do
     submit_time('/estimate 3w 1d 1h')
 
+    wait_for_ajax
     page.within '.time-tracking-estimate-only-pane' do
       expect(page).to have_content '3w 1d 1h'
     end
@@ -16,6 +17,7 @@
   it 'updates the sidebar component when spent is added' do
     submit_time('/spend 3w 1d 1h')
 
+    wait_for_ajax
     page.within '.time-tracking-spend-only-pane' do
       expect(page).to have_content '3w 1d 1h'
     end
@@ -25,6 +27,7 @@
     submit_time('/estimate 3w 1d 1h')
     submit_time('/spend 3w 1d 1h')
 
+    wait_for_ajax
     page.within '.time-tracking-comparison-pane' do
       expect(page).to have_content '3w 1d 1h'
     end
@@ -34,6 +37,7 @@
     submit_time('/estimate 3w 1d 1h')
     submit_time('/remove_estimate')
 
+    wait_for_ajax
     page.within '#issuable-time-tracker' do
       expect(page).to have_content 'No estimate or time spent'
     end
@@ -43,6 +47,7 @@
     submit_time('/spend 3w 1d 1h')
     submit_time('/remove_time_spent')
 
+    wait_for_ajax
     page.within '#issuable-time-tracker' do
       expect(page).to have_content 'No estimate or time spent'
     end
-- 
GitLab


From 8c4c40d09b6947f4ac652dd76cc422fea2a6443d Mon Sep 17 00:00:00 2001
From: blackst0ne <blackst0ne.ru@gmail.com>
Date: Fri, 5 May 2017 23:01:50 +1100
Subject: [PATCH 275/363] Updated specs

---
 spec/features/issuables/system_notes_spec.rb  | 30 -------------------
 spec/features/issues_spec.rb                  | 22 --------------
 spec/features/merge_requests/edit_mr_spec.rb  | 18 -----------
 spec/services/issues/update_service_spec.rb   | 11 +++++++
 .../merge_requests/update_service_spec.rb     |  7 +++++
 spec/services/system_note_service_spec.rb     |  2 +-
 6 files changed, 19 insertions(+), 71 deletions(-)
 delete mode 100644 spec/features/issuables/system_notes_spec.rb

diff --git a/spec/features/issuables/system_notes_spec.rb b/spec/features/issuables/system_notes_spec.rb
deleted file mode 100644
index e12d81e76cbaf..0000000000000
--- a/spec/features/issuables/system_notes_spec.rb
+++ /dev/null
@@ -1,30 +0,0 @@
-require 'spec_helper'
-
-describe 'issuable system notes', feature: true do
-  let(:issue)         { create(:issue, project: project, author: user) }
-  let(:merge_request) { create(:merge_request, :simple, source_project: project) }
-  let(:project)       { create(:project, :public) }
-  let(:user)          { create(:user) }
-
-  before do
-    project.add_user(user, :master)
-    login_as(user)
-  end
-
-  [:issue, :merge_request].each do |issuable_type|
-    context "when #{issuable_type}" do
-      before do
-        issuable = issuable_type == :issue ? issue : merge_request
-
-        visit(edit_polymorphic_path([project.namespace.becomes(Namespace), project, issuable]))
-      end
-
-      it 'adds system note "description changed"' do
-        fill_in("#{issuable_type}_description", with: 'hello world')
-        click_button('Save changes')
-
-        expect(page).to have_content("#{user.name} #{user.to_reference} changed the description")
-      end
-    end
-  end
-end
diff --git a/spec/features/issues_spec.rb b/spec/features/issues_spec.rb
index 93b7880b96f97..81cc8513454ef 100644
--- a/spec/features/issues_spec.rb
+++ b/spec/features/issues_spec.rb
@@ -714,26 +714,4 @@
       expect(page).to have_text("updated title")
     end
   end
-
-  describe '"edited by" message', js: true do
-    let(:issue) { create(:issue, project: project, author: @user) }
-
-    context 'when issue is updated' do
-      before { visit edit_namespace_project_issue_path(project.namespace, project, issue) }
-
-      it 'shows "edited by" mesage on title update' do
-        fill_in 'issue_title', with: 'hello world'
-        click_button 'Save changes'
-
-        expect(page).to have_content("Edited less than a minute ago by #{@user.name}")
-      end
-
-      it 'shows "edited by" mesage on description update' do
-        fill_in 'issue_description', with: 'hello world'
-        click_button 'Save changes'
-
-        expect(page).to have_content("Edited less than a minute ago by #{@user.name}")
-      end
-    end
-  end
 end
diff --git a/spec/features/merge_requests/edit_mr_spec.rb b/spec/features/merge_requests/edit_mr_spec.rb
index b0855af95247f..cb3bc3929031a 100644
--- a/spec/features/merge_requests/edit_mr_spec.rb
+++ b/spec/features/merge_requests/edit_mr_spec.rb
@@ -67,23 +67,5 @@
     def get_textarea_height
       page.evaluate_script('document.getElementById("merge_request_description").offsetHeight')
     end
-
-    describe '"edited by" message', js: true do
-      context 'when merge request is updated' do
-        it 'shows "edited by" mesage on title update' do
-          fill_in 'merge_request_title', with: 'hello world'
-          click_button 'Save changes'
-
-          expect(page).to have_content("Edited less than a minute ago by #{user.name}")
-        end
-
-        it 'shows "edited by" mesage on description update' do
-          fill_in 'merge_request_description', with: 'hello world'
-          click_button 'Save changes'
-
-          expect(page).to have_content("Edited less than a minute ago by #{user.name}")
-        end
-      end
-    end
   end
 end
diff --git a/spec/services/issues/update_service_spec.rb b/spec/services/issues/update_service_spec.rb
index 5b324f3c706d6..ee01b3871f3ac 100644
--- a/spec/services/issues/update_service_spec.rb
+++ b/spec/services/issues/update_service_spec.rb
@@ -132,6 +132,17 @@ def update_issue(opts)
       end
     end
 
+    context 'when description changed' do
+      it 'creates system note about description change' do
+        update_issue(description: 'Changed description')
+
+        note = find_note('changed the description')
+
+        expect(note).not_to be_nil
+        expect(note.note).to eq('changed the description')
+      end
+    end
+
     context 'when issue turns confidential' do
       let(:opts) do
         {
diff --git a/spec/services/merge_requests/update_service_spec.rb b/spec/services/merge_requests/update_service_spec.rb
index f2ca1e6fcbd80..98fc41f2ac07d 100644
--- a/spec/services/merge_requests/update_service_spec.rb
+++ b/spec/services/merge_requests/update_service_spec.rb
@@ -102,6 +102,13 @@ def update_merge_request(opts)
         expect(note.note).to eq 'changed title from **{-Old-} title** to **{+New+} title**'
       end
 
+      it 'creates system note about description change' do
+        note = find_note('changed the description')
+
+        expect(note).not_to be_nil
+        expect(note.note).to eq('changed the description')
+      end
+
       it 'creates system note about branch change' do
         note = find_note('changed target')
 
diff --git a/spec/services/system_note_service_spec.rb b/spec/services/system_note_service_spec.rb
index 5e85c3c162186..5775874ebe0d9 100644
--- a/spec/services/system_note_service_spec.rb
+++ b/spec/services/system_note_service_spec.rb
@@ -301,7 +301,7 @@
       end
 
       it 'sets the note text' do
-        expect(subject.note).to eq 'changed the description'
+        expect(subject.note).to eq('changed the description')
       end
     end
   end
-- 
GitLab


From 61dd92aaff822759941bb224de9f45bfc5f7cc9b Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Fri, 5 May 2017 13:24:07 +0200
Subject: [PATCH 276/363] Authorize build update on per object basis

---
 .../projects/application_controller.rb        |  8 ++++---
 app/controllers/projects/builds_controller.rb | 23 +++++++++++++++----
 2 files changed, 24 insertions(+), 7 deletions(-)

diff --git a/app/controllers/projects/application_controller.rb b/app/controllers/projects/application_controller.rb
index 89f1128ec36ad..afed0ac05a088 100644
--- a/app/controllers/projects/application_controller.rb
+++ b/app/controllers/projects/application_controller.rb
@@ -55,13 +55,15 @@ def can_collaborate_with_project?(project = nil)
       (current_user && current_user.already_forked?(project))
   end
 
-  def authorize_project!(action)
-    return access_denied! unless can?(current_user, action, project)
+  def authorize_action!(action)
+    unless can?(current_user, action, project)
+      return access_denied!
+    end
   end
 
   def method_missing(method_sym, *arguments, &block)
     if method_sym.to_s =~ /\Aauthorize_(.*)!\z/
-      authorize_project!($1.to_sym)
+      authorize_action!($1.to_sym)
     else
       super
     end
diff --git a/app/controllers/projects/builds_controller.rb b/app/controllers/projects/builds_controller.rb
index e24fc45d16660..d97bc93f8dcd9 100644
--- a/app/controllers/projects/builds_controller.rb
+++ b/app/controllers/projects/builds_controller.rb
@@ -1,7 +1,11 @@
 class Projects::BuildsController < Projects::ApplicationController
   before_action :build, except: [:index, :cancel_all]
-  before_action :authorize_read_build!, only: [:index, :show, :status, :raw, :trace]
-  before_action :authorize_update_build!, except: [:index, :show, :status, :raw, :trace]
+
+  before_action :authorize_read_build!,
+    only: [:index, :show, :status, :raw, :trace]
+  before_action :authorize_update_build!,
+    except: [:index, :show, :status, :raw, :trace, :cancel_all]
+
   layout 'project'
 
   def index
@@ -28,7 +32,12 @@ def index
   end
 
   def cancel_all
-    @project.builds.running_or_pending.each(&:cancel)
+    return access_denied! unless can?(current_user, :update_build, project)
+
+    @project.builds.running_or_pending.each do |build|
+      build.cancel if can?(current_user, :update_build, build)
+    end
+
     redirect_to namespace_project_builds_path(project.namespace, project)
   end
 
@@ -107,8 +116,14 @@ def raw
 
   private
 
+  def authorize_update_build!
+    return access_denied! unless can?(current_user, :update_build, build)
+  end
+
   def build
-    @build ||= project.builds.find_by!(id: params[:id]).present(current_user: current_user)
+    @build ||= project.builds
+      .find_by!(id: params[:id])
+      .present(current_user: current_user)
   end
 
   def build_path(build)
-- 
GitLab


From 3264e09c6fbe07831db74b83d6a6620d9f8f47d9 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Fri, 5 May 2017 13:25:48 +0200
Subject: [PATCH 277/363] Require build to be present in the controller

---
 app/controllers/projects/builds_controller.rb | 3 +--
 1 file changed, 1 insertion(+), 2 deletions(-)

diff --git a/app/controllers/projects/builds_controller.rb b/app/controllers/projects/builds_controller.rb
index d97bc93f8dcd9..0fd35bcb790c9 100644
--- a/app/controllers/projects/builds_controller.rb
+++ b/app/controllers/projects/builds_controller.rb
@@ -121,8 +121,7 @@ def authorize_update_build!
   end
 
   def build
-    @build ||= project.builds
-      .find_by!(id: params[:id])
+    @build ||= project.builds.find(params[:id])
       .present(current_user: current_user)
   end
 
-- 
GitLab


From 53219857dd9f97516c6f24f6efb4f405998d9ff2 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Fri, 5 May 2017 13:56:07 +0200
Subject: [PATCH 278/363] Check ability to update build on the API resource

---
 lib/api/jobs.rb      |  9 +++++++--
 lib/api/v3/builds.rb | 10 +++++++---
 2 files changed, 14 insertions(+), 5 deletions(-)

diff --git a/lib/api/jobs.rb b/lib/api/jobs.rb
index 288b03d940ce5..0223957fde11d 100644
--- a/lib/api/jobs.rb
+++ b/lib/api/jobs.rb
@@ -132,6 +132,7 @@ class Jobs < Grape::API
         authorize_update_builds!
 
         build = get_build!(params[:job_id])
+        authorize!(:update_build, build)
 
         build.cancel
 
@@ -148,6 +149,7 @@ class Jobs < Grape::API
         authorize_update_builds!
 
         build = get_build!(params[:job_id])
+        authorize!(:update_build, build)
         return forbidden!('Job is not retryable') unless build.retryable?
 
         build = Ci::Build.retry(build, current_user)
@@ -165,6 +167,7 @@ class Jobs < Grape::API
         authorize_update_builds!
 
         build = get_build!(params[:job_id])
+        authorize!(:update_build, build)
         return forbidden!('Job is not erasable!') unless build.erasable?
 
         build.erase(erased_by: current_user)
@@ -181,6 +184,7 @@ class Jobs < Grape::API
         authorize_update_builds!
 
         build = get_build!(params[:job_id])
+        authorize!(:update_build, build)
         return not_found!(build) unless build.artifacts?
 
         build.keep_artifacts!
@@ -201,6 +205,7 @@ class Jobs < Grape::API
 
         build = get_build!(params[:job_id])
 
+        authorize!(:update_build, build)
         bad_request!("Unplayable Job") unless build.playable?
 
         build.play(current_user)
@@ -211,12 +216,12 @@ class Jobs < Grape::API
     end
 
     helpers do
-      def get_build(id)
+      def find_build(id)
         user_project.builds.find_by(id: id.to_i)
       end
 
       def get_build!(id)
-        get_build(id) || not_found!
+        find_build(id) || not_found!
       end
 
       def present_artifacts!(artifacts_file)
diff --git a/lib/api/v3/builds.rb b/lib/api/v3/builds.rb
index 4dd03cdf24bc9..219359224148e 100644
--- a/lib/api/v3/builds.rb
+++ b/lib/api/v3/builds.rb
@@ -134,6 +134,7 @@ class Builds < Grape::API
           authorize_update_builds!
 
           build = get_build!(params[:build_id])
+          authorize!(:update_build, build)
 
           build.cancel
 
@@ -150,6 +151,7 @@ class Builds < Grape::API
           authorize_update_builds!
 
           build = get_build!(params[:build_id])
+          authorize!(:update_build, build)
           return forbidden!('Build is not retryable') unless build.retryable?
 
           build = Ci::Build.retry(build, current_user)
@@ -167,6 +169,7 @@ class Builds < Grape::API
           authorize_update_builds!
 
           build = get_build!(params[:build_id])
+          authorize!(:update_build, build)
           return forbidden!('Build is not erasable!') unless build.erasable?
 
           build.erase(erased_by: current_user)
@@ -183,6 +186,7 @@ class Builds < Grape::API
           authorize_update_builds!
 
           build = get_build!(params[:build_id])
+          authorize!(:update_build, build)
           return not_found!(build) unless build.artifacts?
 
           build.keep_artifacts!
@@ -202,7 +206,7 @@ class Builds < Grape::API
           authorize_read_builds!
 
           build = get_build!(params[:build_id])
-
+          authorize!(:update_build, build)
           bad_request!("Unplayable Job") unless build.playable?
 
           build.play(current_user)
@@ -213,12 +217,12 @@ class Builds < Grape::API
       end
 
       helpers do
-        def get_build(id)
+        def find_build(id)
           user_project.builds.find_by(id: id.to_i)
         end
 
         def get_build!(id)
-          get_build(id) || not_found!
+          find_build(id) || not_found!
         end
 
         def present_artifacts!(artifacts_file)
-- 
GitLab


From 34be1835af2913c86bc468131e6bcbd530daf48d Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Fri, 5 May 2017 13:41:35 +0300
Subject: [PATCH 279/363] [Multiple issue assignee] Fix a number of specs

---
 app/models/concerns/issuable.rb               |  1 -
 .../members/authorized_destroy_service.rb     | 15 ++++++---
 app/services/system_note_service.rb           |  2 +-
 ...0320171632_create_issue_assignees_table.rb |  2 +-
 db/schema.rb                                  |  2 +-
 lib/github/import.rb                          |  2 +-
 .../dashboard/issuables_counter_spec.rb       |  2 ++
 spec/finders/issues_finder_spec.rb            | 22 ++++++-------
 spec/requests/api/v3/issues_spec.rb           | 32 -------------------
 spec/services/system_note_service_spec.rb     |  4 ++-
 spec/services/users/destroy_service_spec.rb   |  2 +-
 11 files changed, 32 insertions(+), 54 deletions(-)

diff --git a/app/models/concerns/issuable.rb b/app/models/concerns/issuable.rb
index edf4e9e5d780f..16f04305a4345 100644
--- a/app/models/concerns/issuable.rb
+++ b/app/models/concerns/issuable.rb
@@ -94,7 +94,6 @@ def award_emojis_loaded?
 
     acts_as_paranoid
 
-    after_save :update_assignee_cache_counts, if: :assignee_id_changed?
     after_save :record_metrics, unless: :imported?
 
     # We want to use optimistic lock for cases when only title or description are involved
diff --git a/app/services/members/authorized_destroy_service.rb b/app/services/members/authorized_destroy_service.rb
index 1711be7211c6f..a85b9465c8483 100644
--- a/app/services/members/authorized_destroy_service.rb
+++ b/app/services/members/authorized_destroy_service.rb
@@ -26,15 +26,22 @@ def execute
 
     def unassign_issues_and_merge_requests(member)
       if member.is_a?(GroupMember)
-        IssuesFinder.new(user, group_id: member.source_id, assignee_id: member.user_id).
-          execute.
-          update_all(assignee_id: nil)
+        issue_ids = IssuesFinder.new(user, group_id: member.source_id, assignee_id: member.user_id).
+          execute.pluck(:id)
+
+        IssueAssignee.destroy_all(issue_id: issue_ids, user_id: member.user_id)
+
         MergeRequestsFinder.new(user, group_id: member.source_id, assignee_id: member.user_id).
           execute.
           update_all(assignee_id: nil)
       else
         project = member.source
-        project.issues.opened.assigned_to(member.user).update_all(assignee_id: nil)
+
+        IssueAssignee.destroy_all(
+          user_id: member.user_id,
+          issue_id: project.issues.opened.assigned_to(member.user).select(:id)
+        )
+
         project.merge_requests.opened.assigned_to(member.user).update_all(assignee_id: nil)
         member.user.update_cache_counts
       end
diff --git a/app/services/system_note_service.rb b/app/services/system_note_service.rb
index 1dee791cfd615..fb1f56c9cc6a9 100644
--- a/app/services/system_note_service.rb
+++ b/app/services/system_note_service.rb
@@ -84,7 +84,7 @@ def change_issue_assignees(issue, project, author, old_assignees)
         "assigned to #{issue.assignees.map(&:to_reference).to_sentence}"
       end
 
-    NoteSummary.new(issue, project, author, body, action: 'assignee')
+    create_note(NoteSummary.new(issue, project, author, body, action: 'assignee'))
   end
 
   # Called when one or more labels on a Noteable are added and/or removed
diff --git a/db/migrate/20170320171632_create_issue_assignees_table.rb b/db/migrate/20170320171632_create_issue_assignees_table.rb
index 72b70baa8d916..23b8da37b6d75 100644
--- a/db/migrate/20170320171632_create_issue_assignees_table.rb
+++ b/db/migrate/20170320171632_create_issue_assignees_table.rb
@@ -26,7 +26,7 @@ class CreateIssueAssigneesTable < ActiveRecord::Migration
   # disable_ddl_transaction!
 
   def up
-    create_table :issue_assignees, id: false do |t|
+    create_table :issue_assignees do |t|
       t.references :user, foreign_key: { on_delete: :cascade }, index: true, null: false
       t.references :issue, foreign_key: { on_delete: :cascade }, null: false
     end
diff --git a/db/schema.rb b/db/schema.rb
index 340d4064c0245..7b960f79c1fb0 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -452,7 +452,7 @@
 
   add_index "identities", ["user_id"], name: "index_identities_on_user_id", using: :btree
 
-  create_table "issue_assignees", id: false, force: :cascade do |t|
+  create_table "issue_assignees", force: :cascade do |t|
     t.integer "user_id", null: false
     t.integer "issue_id", null: false
   end
diff --git a/lib/github/import.rb b/lib/github/import.rb
index d49761fd6c608..06beb607a3eca 100644
--- a/lib/github/import.rb
+++ b/lib/github/import.rb
@@ -245,7 +245,7 @@ def fetch_issues
               issue.label_ids    = label_ids(representation.labels)
               issue.milestone_id = milestone_id(representation.milestone)
               issue.author_id    = author_id
-              issue.assignee_id  = user_id(representation.assignee)
+              issue.assignee_ids = [user_id(representation.assignee)]
               issue.created_at   = representation.created_at
               issue.updated_at   = representation.updated_at
               issue.save!(validate: false)
diff --git a/spec/features/dashboard/issuables_counter_spec.rb b/spec/features/dashboard/issuables_counter_spec.rb
index 3d536c5ba4025..6f7bf0eba6ebf 100644
--- a/spec/features/dashboard/issuables_counter_spec.rb
+++ b/spec/features/dashboard/issuables_counter_spec.rb
@@ -19,6 +19,8 @@
 
     issue.assignees = []
 
+    user.update_cache_counts
+
     Timecop.travel(3.minutes.from_now) do
       visit issues_path
 
diff --git a/spec/finders/issues_finder_spec.rb b/spec/finders/issues_finder_spec.rb
index e6bf77aee6a61..9615168935989 100644
--- a/spec/finders/issues_finder_spec.rb
+++ b/spec/finders/issues_finder_spec.rb
@@ -1,19 +1,19 @@
 require 'spec_helper'
 
 describe IssuesFinder do
-  let(:user) { create(:user) }
-  let(:user2) { create(:user) }
-  let(:project1) { create(:empty_project) }
-  let(:project2) { create(:empty_project) }
-  let(:milestone) { create(:milestone, project: project1) }
-  let(:label) { create(:label, project: project2) }
-  let(:issue1) { create(:issue, author: user, assignees: [user], project: project1, milestone: milestone, title: 'gitlab') }
-  let(:issue2) { create(:issue, author: user, assignees: [user], project: project2, description: 'gitlab') }
-  let(:issue3) { create(:issue, author: user2, assignees: [user2], project: project2, title: 'tanuki', description: 'tanuki') }
+  set(:user) { create(:user) }
+  set(:user2) { create(:user) }
+  set(:project1) { create(:empty_project) }
+  set(:project2) { create(:empty_project) }
+  set(:milestone) { create(:milestone, project: project1) }
+  set(:label) { create(:label, project: project2) }
+  set(:issue1) { create(:issue, author: user, assignees: [user], project: project1, milestone: milestone, title: 'gitlab') }
+  set(:issue2) { create(:issue, author: user, assignees: [user], project: project2, description: 'gitlab') }
+  set(:issue3) { create(:issue, author: user2, assignees: [user2], project: project2, title: 'tanuki', description: 'tanuki') }
 
   describe '#execute' do
-    let(:closed_issue) { create(:issue, author: user2, assignees: [user2], project: project2, state: 'closed') }
-    let!(:label_link) { create(:label_link, label: label, target: issue2) }
+    set(:closed_issue) { create(:issue, author: user2, assignees: [user2], project: project2, state: 'closed') }
+    set(:label_link) { create(:label_link, label: label, target: issue2) }
     let(:search_user) { user }
     let(:params) { {} }
     let(:issues) { described_class.new(search_user, params.reverse_merge(scope: scope, state: 'opened')).execute }
diff --git a/spec/requests/api/v3/issues_spec.rb b/spec/requests/api/v3/issues_spec.rb
index 49191b2eb4e72..cc81922697ae6 100644
--- a/spec/requests/api/v3/issues_spec.rb
+++ b/spec/requests/api/v3/issues_spec.rb
@@ -1157,38 +1157,6 @@
     end
   end
 
-  describe 'PUT /projects/:id/issues/:issue_id to update weight' do
-    it 'updates an issue with no weight' do
-      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), weight: 5
-
-      expect(response).to have_http_status(200)
-      expect(json_response['weight']).to eq(5)
-    end
-
-    it 'removes a weight from an issue' do
-      weighted_issue = create(:issue, project: project, weight: 2)
-
-      put v3_api("/projects/#{project.id}/issues/#{weighted_issue.id}", user), weight: nil
-
-      expect(response).to have_http_status(200)
-      expect(json_response['weight']).to be_nil
-    end
-
-    it 'returns 400 if weight is less than minimum weight' do
-      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), weight: -1
-
-      expect(response).to have_http_status(400)
-      expect(json_response['error']).to eq('weight does not have a valid value')
-    end
-
-    it 'returns 400 if weight is more than maximum weight' do
-      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), weight: 10
-
-      expect(response).to have_http_status(400)
-      expect(json_response['error']).to eq('weight does not have a valid value')
-    end
-  end
-
   describe "DELETE /projects/:id/issues/:issue_id" do
     it "rejects a non member from deleting an issue" do
       delete v3_api("/projects/#{project.id}/issues/#{issue.id}", non_member)
diff --git a/spec/services/system_note_service_spec.rb b/spec/services/system_note_service_spec.rb
index 5f1b82e835568..68816bf36b833 100644
--- a/spec/services/system_note_service_spec.rb
+++ b/spec/services/system_note_service_spec.rb
@@ -164,7 +164,9 @@
     let(:assignee2) { create(:user) }
     let(:assignee3) { create(:user) }
 
-    it_behaves_like 'a system note'
+    it_behaves_like 'a system note' do
+      let(:action) { 'assignee' }
+    end
 
     def build_note(old_assignees, new_assignees)
       issue.assignees = new_assignees
diff --git a/spec/services/users/destroy_service_spec.rb b/spec/services/users/destroy_service_spec.rb
index 8aa900d1a75fa..de37a61e38880 100644
--- a/spec/services/users/destroy_service_spec.rb
+++ b/spec/services/users/destroy_service_spec.rb
@@ -60,7 +60,7 @@
         it 'migrates the issue so that it is "Unassigned"' do
           migrated_issue = Issue.find_by_id(issue.id)
 
-          expect(migrated_issue.assignees).to be_nil
+          expect(migrated_issue.assignees).to be_empty
         end
       end
     end
-- 
GitLab


From 606584c115d5f7a22f3b5c7e0ac6803b96fe999e Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 14:43:22 +0200
Subject: [PATCH 280/363] bulk insert FTW - This would introduce more
 complexity, but should be faster

---
 app/services/projects/propagate_service.rb | 27 ++++++++++++++++------
 lib/gitlab/sql/bulk_insert.rb              | 21 +++++++++++++++++
 2 files changed, 41 insertions(+), 7 deletions(-)
 create mode 100644 lib/gitlab/sql/bulk_insert.rb

diff --git a/app/services/projects/propagate_service.rb b/app/services/projects/propagate_service.rb
index f952b5108bb1f..c420f24fe028b 100644
--- a/app/services/projects/propagate_service.rb
+++ b/app/services/projects/propagate_service.rb
@@ -24,20 +24,23 @@ def propagate_projects_with_template
       loop do
         batch = project_ids_batch
 
-        bulk_create_from_template(batch)
+        bulk_create_from_template(batch) unless batch.empty?
 
         break if batch.size < BATCH_SIZE
       end
     end
 
     def bulk_create_from_template(batch)
-      service_hash_list = batch.map do |project_id|
-        service_hash.merge('project_id' => project_id)
+      service_list = batch.map do |project_id|
+        service_hash.merge('project_id' => project_id).values
       end
 
-      Project.transaction do
-        Service.create!(service_hash_list)
-      end
+      # Project.transaction do
+      #   Service.create!(service_hash_list)
+      # end
+      Gitlab::SQL::BulkInsert.new(service_hash.keys + ['project_id'],
+                                  service_list,
+                                  'services').execute
     end
 
     def project_ids_batch
@@ -57,7 +60,17 @@ def project_ids_batch
     end
 
     def service_hash
-      @service_hash ||= @template.as_json(methods: :type).except('id', 'template')
+      @service_hash ||=
+        begin
+          template_hash = @template.as_json(methods: :type).except('id', 'template', 'project_id')
+
+          template_hash.each_with_object({}) do |(key, value), service_hash|
+            value = value.is_a?(Hash) ? value.to_json : value
+            key = Gitlab::Database.postgresql? ? "\"#{key}\"" : "`#{key}`"
+
+            service_hash[key] = ActiveRecord::Base.sanitize(value)
+          end
+        end
     end
   end
 end
diff --git a/lib/gitlab/sql/bulk_insert.rb b/lib/gitlab/sql/bulk_insert.rb
new file mode 100644
index 0000000000000..097f9ff237b1e
--- /dev/null
+++ b/lib/gitlab/sql/bulk_insert.rb
@@ -0,0 +1,21 @@
+module Gitlab
+  module SQL
+    # Class for building SQL bulk inserts
+    class BulkInsert
+      def initialize(columns, values_array, table)
+        @columns = columns
+        @values_array = values_array
+        @table = table
+      end
+
+      def execute
+        ActiveRecord::Base.connection.execute(
+          <<-SQL.strip_heredoc
+          INSERT INTO #{@table} (#{@columns.join(', ')})
+          VALUES #{@values_array.map { |tuple| "(#{tuple.join(', ')})" }.join(', ')}
+        SQL
+        )
+      end
+    end
+  end
+end
-- 
GitLab


From 715cdc1afdf59cde4bd4a6183b81e2e19ef3ab78 Mon Sep 17 00:00:00 2001
From: Lin Jen-Shin <godfat@godfat.org>
Date: Fri, 5 May 2017 12:53:44 +0000
Subject: [PATCH 281/363] Update CHANGELOG.md for 8.17.6

[ci skip]
---
 CHANGELOG.md | 11 +++++++++++
 1 file changed, 11 insertions(+)

diff --git a/CHANGELOG.md b/CHANGELOG.md
index e05b025ce2d6a..c9de0113e2452 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -643,6 +643,17 @@ entry.
 - Change development tanuki favicon colors to match logo color order.
 - API issues - support filtering by iids.
 
+## 8.17.6 (2017-05-05)
+
+- Enforce project features when searching blobs and wikis.
+- Fixed branches dropdown rendering branch names as HTML.
+- Make Asciidoc & other markup go through pipeline to prevent XSS.
+- Validate URLs in markdown using URI to detect the host correctly.
+- Fix for XSS in project import view caused by Hamlit filter usage.
+- Sanitize submodule URLs before linking to them in the file tree view.
+- Refactor snippets finder & dont return internal snippets for external users.
+- Fix snippets visibility for show action - external users can not see internal snippets.
+
 ## 8.17.5 (2017-04-05)
 
 - Don’t show source project name when user does not have access.
-- 
GitLab


From 2cc8f43e54d2b653a4f2e80c57339acb11dcba86 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Fri, 5 May 2017 15:13:58 +0200
Subject: [PATCH 282/363] Introduce generic manual action extended status class

---
 lib/gitlab/ci/status/build/action.rb          | 23 ++++++++++++
 lib/gitlab/ci/status/build/factory.rb         |  3 +-
 lib/gitlab/ci/status/build/play.rb            |  6 +---
 .../gitlab/ci/status/build/factory_spec.rb    | 32 +++++++++--------
 spec/lib/gitlab/ci/status/build/play_spec.rb  | 36 +++++++------------
 5 files changed, 57 insertions(+), 43 deletions(-)
 create mode 100644 lib/gitlab/ci/status/build/action.rb

diff --git a/lib/gitlab/ci/status/build/action.rb b/lib/gitlab/ci/status/build/action.rb
new file mode 100644
index 0000000000000..1397c35145a72
--- /dev/null
+++ b/lib/gitlab/ci/status/build/action.rb
@@ -0,0 +1,23 @@
+module Gitlab
+  module Ci
+    module Status
+      module Build
+        class Action < SimpleDelegator
+          include Status::Extended
+
+          def label
+            if has_action?
+              __getobj__.label
+            else
+              "#{__getobj__.label} (not allowed)"
+            end
+          end
+
+          def self.matches?(build, user)
+            build.action?
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/lib/gitlab/ci/status/build/factory.rb b/lib/gitlab/ci/status/build/factory.rb
index 38ac6edc9f188..c852d6073736d 100644
--- a/lib/gitlab/ci/status/build/factory.rb
+++ b/lib/gitlab/ci/status/build/factory.rb
@@ -8,7 +8,8 @@ def self.extended_statuses
               Status::Build::Retryable],
              [Status::Build::FailedAllowed,
               Status::Build::Play,
-              Status::Build::Stop]]
+              Status::Build::Stop],
+             [Status::Build::Action]]
           end
 
           def self.common_helpers
diff --git a/lib/gitlab/ci/status/build/play.rb b/lib/gitlab/ci/status/build/play.rb
index fae34a2f9276a..3495b8d0448b9 100644
--- a/lib/gitlab/ci/status/build/play.rb
+++ b/lib/gitlab/ci/status/build/play.rb
@@ -6,11 +6,7 @@ class Play < SimpleDelegator
           include Status::Extended
 
           def label
-            if has_action?
-              'manual play action'
-            else
-              'manual play action (not allowed)'
-            end
+            'manual play action'
           end
 
           def has_action?
diff --git a/spec/lib/gitlab/ci/status/build/factory_spec.rb b/spec/lib/gitlab/ci/status/build/factory_spec.rb
index 2de00c289450c..39c5e3658d910 100644
--- a/spec/lib/gitlab/ci/status/build/factory_spec.rb
+++ b/spec/lib/gitlab/ci/status/build/factory_spec.rb
@@ -204,11 +204,12 @@
 
       it 'matches correct extended statuses' do
         expect(factory.extended_statuses)
-          .to eq [Gitlab::Ci::Status::Build::Play]
+          .to eq [Gitlab::Ci::Status::Build::Play,
+                 Gitlab::Ci::Status::Build::Action]
       end
 
-      it 'fabricates a play detailed status' do
-        expect(status).to be_a Gitlab::Ci::Status::Build::Play
+      it 'fabricates action detailed status' do
+        expect(status).to be_a Gitlab::Ci::Status::Build::Action
       end
 
       it 'fabricates status with correct details' do
@@ -247,21 +248,24 @@
 
       it 'matches correct extended statuses' do
         expect(factory.extended_statuses)
-          .to eq [Gitlab::Ci::Status::Build::Stop]
+          .to eq [Gitlab::Ci::Status::Build::Stop,
+                  Gitlab::Ci::Status::Build::Action]
       end
 
-      it 'fabricates a stop detailed status' do
-        expect(status).to be_a Gitlab::Ci::Status::Build::Stop
+      it 'fabricates action detailed status' do
+        expect(status).to be_a Gitlab::Ci::Status::Build::Action
       end
 
-      it 'fabricates status with correct details' do
-        expect(status.text).to eq 'manual'
-        expect(status.group).to eq 'manual'
-        expect(status.icon).to eq 'icon_status_manual'
-        expect(status.favicon).to eq 'favicon_status_manual'
-        expect(status.label).to eq 'manual stop action'
-        expect(status).to have_details
-        expect(status).to have_action
+      context 'when user is not allowed to execute manual action' do
+        it 'fabricates status with correct details' do
+          expect(status.text).to eq 'manual'
+          expect(status.group).to eq 'manual'
+          expect(status.icon).to eq 'icon_status_manual'
+          expect(status.favicon).to eq 'favicon_status_manual'
+          expect(status.label).to eq 'manual stop action (not allowed)'
+          expect(status).to have_details
+          expect(status).not_to have_action
+        end
       end
     end
   end
diff --git a/spec/lib/gitlab/ci/status/build/play_spec.rb b/spec/lib/gitlab/ci/status/build/play_spec.rb
index abefdbe7c66ca..f5d0f977768fc 100644
--- a/spec/lib/gitlab/ci/status/build/play_spec.rb
+++ b/spec/lib/gitlab/ci/status/build/play_spec.rb
@@ -7,38 +7,28 @@
 
   subject { described_class.new(status) }
 
-  context 'when user is allowed to update build' do
-    context 'when user can push to branch' do
-      before { build.project.add_master(user) }
+  describe '#label' do
+    it 'has a label that says it is a manual action' do
+      expect(subject.label).to eq 'manual play action'
+    end
+  end
+
+  describe '#has_action?' do
+    context 'when user is allowed to update build' do
+      context 'when user can push to branch' do
+        before { build.project.add_master(user) }
 
-      describe '#has_action?' do
         it { is_expected.to have_action }
       end
 
-      describe '#label' do
-        it 'has a label that says it is a manual action' do
-          expect(subject.label).to eq 'manual play action'
-        end
-      end
-    end
+      context 'when user can not push to the branch' do
+        before { build.project.add_developer(user) }
 
-    context 'when user can not push to the branch' do
-      before { build.project.add_developer(user) }
-
-      describe 'has_action?' do
         it { is_expected.not_to have_action }
       end
-
-      describe '#label' do
-        it 'has a label that says user is not allowed to play it' do
-          expect(subject.label).to eq 'manual play action (not allowed)'
-        end
-      end
     end
-  end
 
-  context 'when user is not allowed to update build' do
-    describe '#has_action?' do
+    context 'when user is not allowed to update build' do
       it { is_expected.not_to have_action }
     end
   end
-- 
GitLab


From 55cec2177cd97a5c0939ab34b01aa408de887d1a Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Fri, 5 May 2017 15:21:06 +0200
Subject: [PATCH 283/363] Refine inheritance model of extended CI/CD statuses

---
 lib/gitlab/ci/status/build/action.rb            |  8 +++-----
 lib/gitlab/ci/status/build/cancelable.rb        |  4 +---
 lib/gitlab/ci/status/build/failed_allowed.rb    |  4 +---
 lib/gitlab/ci/status/build/play.rb              |  4 +---
 lib/gitlab/ci/status/build/retryable.rb         |  4 +---
 lib/gitlab/ci/status/build/stop.rb              |  4 +---
 lib/gitlab/ci/status/extended.rb                | 12 ++++++------
 lib/gitlab/ci/status/pipeline/blocked.rb        |  4 +---
 lib/gitlab/ci/status/success_warning.rb         |  4 +---
 spec/lib/gitlab/ci/status/build/factory_spec.rb |  2 +-
 spec/lib/gitlab/ci/status/extended_spec.rb      |  6 +-----
 11 files changed, 18 insertions(+), 38 deletions(-)

diff --git a/lib/gitlab/ci/status/build/action.rb b/lib/gitlab/ci/status/build/action.rb
index 1397c35145a72..45fd0d4aa0739 100644
--- a/lib/gitlab/ci/status/build/action.rb
+++ b/lib/gitlab/ci/status/build/action.rb
@@ -2,14 +2,12 @@ module Gitlab
   module Ci
     module Status
       module Build
-        class Action < SimpleDelegator
-          include Status::Extended
-
+        class Action < Status::Extended
           def label
             if has_action?
-              __getobj__.label
+              @status.label
             else
-              "#{__getobj__.label} (not allowed)"
+              "#{@status.label} (not allowed)"
             end
           end
 
diff --git a/lib/gitlab/ci/status/build/cancelable.rb b/lib/gitlab/ci/status/build/cancelable.rb
index 67bbc3c484932..57b533bad99c6 100644
--- a/lib/gitlab/ci/status/build/cancelable.rb
+++ b/lib/gitlab/ci/status/build/cancelable.rb
@@ -2,9 +2,7 @@ module Gitlab
   module Ci
     module Status
       module Build
-        class Cancelable < SimpleDelegator
-          include Status::Extended
-
+        class Cancelable < Status::Extended
           def has_action?
             can?(user, :update_build, subject)
           end
diff --git a/lib/gitlab/ci/status/build/failed_allowed.rb b/lib/gitlab/ci/status/build/failed_allowed.rb
index 807afe24bd576..e42d3574357cb 100644
--- a/lib/gitlab/ci/status/build/failed_allowed.rb
+++ b/lib/gitlab/ci/status/build/failed_allowed.rb
@@ -2,9 +2,7 @@ module Gitlab
   module Ci
     module Status
       module Build
-        class FailedAllowed < SimpleDelegator
-          include Status::Extended
-
+        class FailedAllowed < Status::Extended
           def label
             'failed (allowed to fail)'
           end
diff --git a/lib/gitlab/ci/status/build/play.rb b/lib/gitlab/ci/status/build/play.rb
index 3495b8d0448b9..c6139f1b71647 100644
--- a/lib/gitlab/ci/status/build/play.rb
+++ b/lib/gitlab/ci/status/build/play.rb
@@ -2,9 +2,7 @@ module Gitlab
   module Ci
     module Status
       module Build
-        class Play < SimpleDelegator
-          include Status::Extended
-
+        class Play < Status::Extended
           def label
             'manual play action'
           end
diff --git a/lib/gitlab/ci/status/build/retryable.rb b/lib/gitlab/ci/status/build/retryable.rb
index 6b362af76343e..505f80848b250 100644
--- a/lib/gitlab/ci/status/build/retryable.rb
+++ b/lib/gitlab/ci/status/build/retryable.rb
@@ -2,9 +2,7 @@ module Gitlab
   module Ci
     module Status
       module Build
-        class Retryable < SimpleDelegator
-          include Status::Extended
-
+        class Retryable < Status::Extended
           def has_action?
             can?(user, :update_build, subject)
           end
diff --git a/lib/gitlab/ci/status/build/stop.rb b/lib/gitlab/ci/status/build/stop.rb
index e8530f2aaaed4..0b5199e5483a7 100644
--- a/lib/gitlab/ci/status/build/stop.rb
+++ b/lib/gitlab/ci/status/build/stop.rb
@@ -2,9 +2,7 @@ module Gitlab
   module Ci
     module Status
       module Build
-        class Stop < SimpleDelegator
-          include Status::Extended
-
+        class Stop < Status::Extended
           def label
             'manual stop action'
           end
diff --git a/lib/gitlab/ci/status/extended.rb b/lib/gitlab/ci/status/extended.rb
index d367c9bda69ff..1e8101f894950 100644
--- a/lib/gitlab/ci/status/extended.rb
+++ b/lib/gitlab/ci/status/extended.rb
@@ -1,13 +1,13 @@
 module Gitlab
   module Ci
     module Status
-      module Extended
-        extend ActiveSupport::Concern
+      class Extended < SimpleDelegator
+        def initialize(status)
+          super(@status = status)
+        end
 
-        class_methods do
-          def matches?(_subject, _user)
-            raise NotImplementedError
-          end
+        def self.matches?(_subject, _user)
+          raise NotImplementedError
         end
       end
     end
diff --git a/lib/gitlab/ci/status/pipeline/blocked.rb b/lib/gitlab/ci/status/pipeline/blocked.rb
index a250c3fcb419d..37dfe43fb621a 100644
--- a/lib/gitlab/ci/status/pipeline/blocked.rb
+++ b/lib/gitlab/ci/status/pipeline/blocked.rb
@@ -2,9 +2,7 @@ module Gitlab
   module Ci
     module Status
       module Pipeline
-        class Blocked < SimpleDelegator
-          include Status::Extended
-
+        class Blocked < Status::Extended
           def text
             'blocked'
           end
diff --git a/lib/gitlab/ci/status/success_warning.rb b/lib/gitlab/ci/status/success_warning.rb
index d4cdab6957a00..df6e76b015183 100644
--- a/lib/gitlab/ci/status/success_warning.rb
+++ b/lib/gitlab/ci/status/success_warning.rb
@@ -5,9 +5,7 @@ module Status
       # Extended status used when pipeline or stage passed conditionally.
       # This means that failed jobs that are allowed to fail were present.
       #
-      class SuccessWarning < SimpleDelegator
-        include Status::Extended
-
+      class SuccessWarning < Status::Extended
         def text
           'passed'
         end
diff --git a/spec/lib/gitlab/ci/status/build/factory_spec.rb b/spec/lib/gitlab/ci/status/build/factory_spec.rb
index 39c5e3658d910..185bb9098dac0 100644
--- a/spec/lib/gitlab/ci/status/build/factory_spec.rb
+++ b/spec/lib/gitlab/ci/status/build/factory_spec.rb
@@ -205,7 +205,7 @@
       it 'matches correct extended statuses' do
         expect(factory.extended_statuses)
           .to eq [Gitlab::Ci::Status::Build::Play,
-                 Gitlab::Ci::Status::Build::Action]
+                  Gitlab::Ci::Status::Build::Action]
       end
 
       it 'fabricates action detailed status' do
diff --git a/spec/lib/gitlab/ci/status/extended_spec.rb b/spec/lib/gitlab/ci/status/extended_spec.rb
index c2d74ca5cde0c..6eacb07078bf9 100644
--- a/spec/lib/gitlab/ci/status/extended_spec.rb
+++ b/spec/lib/gitlab/ci/status/extended_spec.rb
@@ -1,12 +1,8 @@
 require 'spec_helper'
 
 describe Gitlab::Ci::Status::Extended do
-  subject do
-    Class.new.include(described_class)
-  end
-
   it 'requires subclass to implement matcher' do
-    expect { subject.matches?(double, double) }
+    expect { described_class.matches?(double, double) }
       .to raise_error(NotImplementedError)
   end
 end
-- 
GitLab


From e5f24c5490294a4e990103da24e9861f21d7bfd9 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Fri, 5 May 2017 15:31:00 +0200
Subject: [PATCH 284/363] Add specs for extended status for manual actions

---
 .../lib/gitlab/ci/status/build/action_spec.rb | 56 +++++++++++++++++++
 1 file changed, 56 insertions(+)
 create mode 100644 spec/lib/gitlab/ci/status/build/action_spec.rb

diff --git a/spec/lib/gitlab/ci/status/build/action_spec.rb b/spec/lib/gitlab/ci/status/build/action_spec.rb
new file mode 100644
index 0000000000000..8c25f72804b68
--- /dev/null
+++ b/spec/lib/gitlab/ci/status/build/action_spec.rb
@@ -0,0 +1,56 @@
+require 'spec_helper'
+
+describe Gitlab::Ci::Status::Build::Action do
+  let(:status) { double('core status') }
+  let(:user) { double('user') }
+
+  subject do
+    described_class.new(status)
+  end
+
+  describe '#label' do
+    before do
+      allow(status).to receive(:label).and_return('label')
+    end
+
+    context 'when status has action' do
+      before do
+        allow(status).to receive(:has_action?).and_return(true)
+      end
+
+      it 'does not append text' do
+        expect(subject.label).to eq 'label'
+      end
+    end
+
+    context 'when status does not have action' do
+      before do
+        allow(status).to receive(:has_action?).and_return(false)
+      end
+
+      it 'appends text about action not allowed' do
+        expect(subject.label).to eq 'label (not allowed)'
+      end
+    end
+  end
+
+  describe '.matches?' do
+    subject { described_class.matches?(build, user) }
+
+    context 'when build is an action' do
+      let(:build) { create(:ci_build, :manual) }
+
+      it 'is a correct match' do
+        expect(subject).to be true
+      end
+    end
+
+    context 'when build is not manual' do
+      let(:build) { create(:ci_build) }
+
+      it 'does not match' do
+        expect(subject).to be false
+      end
+    end
+  end
+end
-- 
GitLab


From e4f7b87ddb4ba83456871eb83b841192b1b56799 Mon Sep 17 00:00:00 2001
From: Jarka Kadlecova <jarka@gitlab.com>
Date: Wed, 3 May 2017 10:48:01 +0200
Subject: [PATCH 285/363] Support comments for personal snippets

---
 app/controllers/concerns/notes_actions.rb     | 44 +++++++++++
 app/controllers/projects/notes_controller.rb  | 44 -----------
 app/controllers/snippets/notes_controller.rb  |  9 ---
 app/controllers/snippets_controller.rb        |  1 +
 app/helpers/gitlab_routing_helper.rb          |  6 +-
 app/helpers/notes_helper.rb                   | 43 +++++++++++
 app/services/notes/build_service.rb           | 18 ++++-
 app/views/layouts/snippets.html.haml          |  6 ++
 app/views/projects/commit/show.html.haml      |  2 +-
 .../projects/issues/_discussion.html.haml     |  2 +-
 .../merge_requests/_discussion.html.haml      |  2 +-
 app/views/projects/milestones/_form.html.haml |  2 +-
 app/views/projects/notes/_edit.html.haml      |  3 -
 app/views/projects/releases/edit.html.haml    |  2 +-
 app/views/projects/snippets/show.html.haml    |  2 +-
 app/views/projects/tags/new.html.haml         |  2 +-
 app/views/projects/wikis/_form.html.haml      |  2 +-
 .../issuable/form/_description.html.haml      |  2 +-
 .../notes/_comment_button.html.haml           |  0
 app/views/shared/notes/_edit.html.haml        |  3 +
 .../notes/_edit_form.html.haml                |  2 +-
 .../notes/_form.html.haml                     |  6 +-
 .../notes/_hints.html.haml                    |  0
 app/views/shared/notes/_note.html.haml        |  5 +-
 .../notes/_notes_with_form.html.haml          |  8 +-
 app/views/snippets/notes/_edit.html.haml      |  0
 app/views/snippets/notes/_notes.html.haml     |  2 -
 app/views/snippets/show.html.haml             | 12 +--
 .../12910-personal-snippets-notes.yml         |  4 +
 spec/factories/notes.rb                       |  2 +
 .../notes_on_personal_snippets_spec.rb        | 64 +++++++++++++++-
 spec/helpers/notes_helper_spec.rb             | 75 +++++++++++++++++++
 spec/services/notes/build_service_spec.rb     | 74 +++++++++++++++++-
 .../notes/_form.html.haml_spec.rb             |  2 +-
 34 files changed, 359 insertions(+), 92 deletions(-)
 delete mode 100644 app/views/projects/notes/_edit.html.haml
 rename app/views/{projects => shared}/notes/_comment_button.html.haml (100%)
 create mode 100644 app/views/shared/notes/_edit.html.haml
 rename app/views/{projects => shared}/notes/_edit_form.html.haml (95%)
 rename app/views/{projects => shared}/notes/_form.html.haml (74%)
 rename app/views/{projects => shared}/notes/_hints.html.haml (100%)
 rename app/views/{projects => shared}/notes/_notes_with_form.html.haml (64%)
 delete mode 100644 app/views/snippets/notes/_edit.html.haml
 delete mode 100644 app/views/snippets/notes/_notes.html.haml
 create mode 100644 changelogs/unreleased/12910-personal-snippets-notes.yml
 rename spec/views/{projects => shared}/notes/_form.html.haml_spec.rb (96%)

diff --git a/app/controllers/concerns/notes_actions.rb b/app/controllers/concerns/notes_actions.rb
index c32038d07bfaa..a57d9e6e6c0ec 100644
--- a/app/controllers/concerns/notes_actions.rb
+++ b/app/controllers/concerns/notes_actions.rb
@@ -65,6 +65,15 @@ def destroy
 
   private
 
+  def note_html(note)
+    render_to_string(
+      "shared/notes/_note",
+      layout: false,
+      formats: [:html],
+      locals: { note: note }
+    )
+  end
+
   def note_json(note)
     attrs = {
       commands_changes: note.commands_changes
@@ -98,6 +107,41 @@ def note_json(note)
     attrs
   end
 
+  def diff_discussion_html(discussion)
+    return unless discussion.diff_discussion?
+
+    if params[:view] == 'parallel'
+      template = "discussions/_parallel_diff_discussion"
+      locals =
+        if params[:line_type] == 'old'
+          { discussions_left: [discussion], discussions_right: nil }
+        else
+          { discussions_left: nil, discussions_right: [discussion] }
+        end
+    else
+      template = "discussions/_diff_discussion"
+      locals = { discussions: [discussion] }
+    end
+
+    render_to_string(
+      template,
+      layout: false,
+      formats: [:html],
+      locals: locals
+    )
+  end
+
+  def discussion_html(discussion)
+    return if discussion.individual_note?
+
+    render_to_string(
+      "discussions/_discussion",
+      layout: false,
+      formats: [:html],
+      locals: { discussion: discussion }
+    )
+  end
+
   def authorize_admin_note!
     return access_denied! unless can?(current_user, :admin_note, note)
   end
diff --git a/app/controllers/projects/notes_controller.rb b/app/controllers/projects/notes_controller.rb
index 37f51b2ebe3ba..41a13f6f5778e 100644
--- a/app/controllers/projects/notes_controller.rb
+++ b/app/controllers/projects/notes_controller.rb
@@ -62,50 +62,6 @@ def note
   end
   alias_method :awardable, :note
 
-  def note_html(note)
-    render_to_string(
-      "shared/notes/_note",
-      layout: false,
-      formats: [:html],
-      locals: { note: note }
-    )
-  end
-
-  def discussion_html(discussion)
-    return if discussion.individual_note?
-
-    render_to_string(
-      "discussions/_discussion",
-      layout: false,
-      formats: [:html],
-      locals: { discussion: discussion }
-    )
-  end
-
-  def diff_discussion_html(discussion)
-    return unless discussion.diff_discussion?
-
-    if params[:view] == 'parallel'
-      template = "discussions/_parallel_diff_discussion"
-      locals =
-        if params[:line_type] == 'old'
-          { discussions_left: [discussion], discussions_right: nil }
-        else
-          { discussions_left: nil, discussions_right: [discussion] }
-        end
-    else
-      template = "discussions/_diff_discussion"
-      locals = { discussions: [discussion] }
-    end
-
-    render_to_string(
-      template,
-      layout: false,
-      formats: [:html],
-      locals: locals
-    )
-  end
-
   def finder_params
     params.merge(last_fetched_at: last_fetched_at)
   end
diff --git a/app/controllers/snippets/notes_controller.rb b/app/controllers/snippets/notes_controller.rb
index 3c4ddc1680de0..f9496787b15f5 100644
--- a/app/controllers/snippets/notes_controller.rb
+++ b/app/controllers/snippets/notes_controller.rb
@@ -13,15 +13,6 @@ def note
   end
   alias_method :awardable, :note
 
-  def note_html(note)
-    render_to_string(
-      "shared/notes/_note",
-      layout: false,
-      formats: [:html],
-      locals: { note: note }
-    )
-  end
-
   def project
     nil
   end
diff --git a/app/controllers/snippets_controller.rb b/app/controllers/snippets_controller.rb
index afed27a41d1fb..19e07e3ab86f6 100644
--- a/app/controllers/snippets_controller.rb
+++ b/app/controllers/snippets_controller.rb
@@ -64,6 +64,7 @@ def show
     blob = @snippet.blob
     override_max_blob_size(blob)
 
+    @note = Note.new(noteable: @snippet)
     @noteable = @snippet
 
     @discussions = @snippet.discussions
diff --git a/app/helpers/gitlab_routing_helper.rb b/app/helpers/gitlab_routing_helper.rb
index 88f9a691a17a3..3769830de2a1f 100644
--- a/app/helpers/gitlab_routing_helper.rb
+++ b/app/helpers/gitlab_routing_helper.rb
@@ -123,7 +123,11 @@ def project_snippet_url(entity, *args)
   end
 
   def preview_markdown_path(project, *args)
-    preview_markdown_namespace_project_path(project.namespace, project, *args)
+    if @snippet.is_a?(PersonalSnippet)
+      preview_markdown_snippet_path(@snippet)
+    else
+      preview_markdown_namespace_project_path(project.namespace, project, *args)
+    end
   end
 
   def toggle_subscription_path(entity, *args)
diff --git a/app/helpers/notes_helper.rb b/app/helpers/notes_helper.rb
index 08180883eb9e0..52403640c053e 100644
--- a/app/helpers/notes_helper.rb
+++ b/app/helpers/notes_helper.rb
@@ -76,4 +76,47 @@ def discussion_path(discussion)
       namespace_project_commit_path(discussion.project.namespace, discussion.project, discussion.noteable, anchor: anchor)
     end
   end
+
+  def notes_url
+    if @snippet.is_a?(PersonalSnippet)
+      snippet_notes_path(@snippet)
+    else
+      namespace_project_noteable_notes_path(
+        namespace_id: @project.namespace,
+        project_id: @project,
+        target_id: @noteable.id,
+        target_type: @noteable.class.name.underscore
+      )
+    end
+  end
+
+  def note_url(note)
+    if note.noteable.is_a?(PersonalSnippet)
+      snippet_note_path(note.noteable, note)
+    else
+      namespace_project_note_path(@project.namespace, @project, note)
+    end
+  end
+
+  def form_resources
+    if @snippet.is_a?(PersonalSnippet)
+      [@note]
+    else
+      [@project.namespace.becomes(Namespace), @project, @note]
+    end
+  end
+
+  def new_form_url
+    return nil unless @snippet.is_a?(PersonalSnippet)
+
+    snippet_notes_path(@snippet)
+  end
+
+  def can_create_note?
+    if @snippet.is_a?(PersonalSnippet)
+      can?(current_user, :comment_personal_snippet, @snippet)
+    else
+      can?(current_user, :create_note, @project)
+    end
+  end
 end
diff --git a/app/services/notes/build_service.rb b/app/services/notes/build_service.rb
index ea7cacc956c1c..abf25bb778bba 100644
--- a/app/services/notes/build_service.rb
+++ b/app/services/notes/build_service.rb
@@ -3,8 +3,8 @@ class BuildService < ::BaseService
     def execute
       in_reply_to_discussion_id = params.delete(:in_reply_to_discussion_id)
 
-      if project && in_reply_to_discussion_id.present?
-        discussion = project.notes.find_discussion(in_reply_to_discussion_id)
+      if in_reply_to_discussion_id.present?
+        discussion = find_discussion(in_reply_to_discussion_id)
 
         unless discussion
           note = Note.new
@@ -21,5 +21,19 @@ def execute
 
       note
     end
+
+    def find_discussion(discussion_id)
+      if project
+        project.notes.find_discussion(discussion_id)
+      else
+        # only PersonalSnippets can have discussions without project association
+        discussion = Note.find_discussion(discussion_id)
+        noteable = discussion.noteable
+
+        return nil unless noteable.is_a?(PersonalSnippet) && can?(current_user, :comment_personal_snippet, noteable)
+
+        discussion
+      end
+    end
   end
 end
diff --git a/app/views/layouts/snippets.html.haml b/app/views/layouts/snippets.html.haml
index 02ca3ee7a2875..98b75cea03fd7 100644
--- a/app/views/layouts/snippets.html.haml
+++ b/app/views/layouts/snippets.html.haml
@@ -1,3 +1,9 @@
 - header_title  "Snippets", snippets_path
 
+- content_for :page_specific_javascripts do
+  - if @snippet&.persisted? && current_user
+    :javascript
+      window.uploads_path = "#{upload_path('personal_snippet', @snippet)}";
+      window.preview_markdown_path = "#{preview_markdown_snippet_path(@snippet)}";
+
 = render template: "layouts/application"
diff --git a/app/views/projects/commit/show.html.haml b/app/views/projects/commit/show.html.haml
index 16d2646cb4e08..6051ea2f1ce0d 100644
--- a/app/views/projects/commit/show.html.haml
+++ b/app/views/projects/commit/show.html.haml
@@ -13,7 +13,7 @@
     .block-connector
   = render "projects/diffs/diffs", diffs: @diffs, environment: @environment
 
-  = render "projects/notes/notes_with_form"
+  = render "shared/notes/notes_with_form"
   - if can_collaborate_with_project?
     - %w(revert cherry-pick).each do |type|
       = render "projects/commit/change", type: type, commit: @commit, title: @commit.title
diff --git a/app/views/projects/issues/_discussion.html.haml b/app/views/projects/issues/_discussion.html.haml
index 5d4e593e4ef21..4dfda54feb554 100644
--- a/app/views/projects/issues/_discussion.html.haml
+++ b/app/views/projects/issues/_discussion.html.haml
@@ -4,4 +4,4 @@
     = link_to 'Close issue', issue_path(@issue, issue: {state_event: :close}, format: 'json'), data: {no_turbolink: true, original_text: "Close issue", alternative_text: "Comment & close issue"}, class: "btn btn-nr btn-close btn-comment js-note-target-close #{issue_button_visibility(@issue, true)}", title: 'Close issue'
 
 #notes
-  = render 'projects/notes/notes_with_form'
+  = render 'shared/notes/notes_with_form'
diff --git a/app/views/projects/merge_requests/_discussion.html.haml b/app/views/projects/merge_requests/_discussion.html.haml
index 15b5a51c1d0c4..2e6420db21223 100644
--- a/app/views/projects/merge_requests/_discussion.html.haml
+++ b/app/views/projects/merge_requests/_discussion.html.haml
@@ -8,4 +8,4 @@
     %button.btn.btn-nr.btn-default.append-right-10.js-comment-resolve-button{ "v-if" => "showButton", type: "submit", data: { project_path: "#{project_path(@merge_request.project)}" } }
       {{ buttonText }}
 
-#notes= render "projects/notes/notes_with_form"
+#notes= render "shared/notes/notes_with_form"
diff --git a/app/views/projects/milestones/_form.html.haml b/app/views/projects/milestones/_form.html.haml
index 2e978fda62422..9a95b2a82ffd7 100644
--- a/app/views/projects/milestones/_form.html.haml
+++ b/app/views/projects/milestones/_form.html.haml
@@ -11,7 +11,7 @@
         .col-sm-10
           = render layout: 'projects/md_preview', locals: { url: preview_markdown_path(@project) } do
             = render 'projects/zen', f: f, attr: :description, classes: 'note-textarea', placeholder: 'Write milestone description...'
-            = render 'projects/notes/hints'
+            = render 'shared/notes/hints'
           .clearfix
           .error-alert
     = render "shared/milestones/form_dates", f: f
diff --git a/app/views/projects/notes/_edit.html.haml b/app/views/projects/notes/_edit.html.haml
deleted file mode 100644
index f1e251d65b75b..0000000000000
--- a/app/views/projects/notes/_edit.html.haml
+++ /dev/null
@@ -1,3 +0,0 @@
-.original-note-content.hidden{ data: { post_url: namespace_project_note_path(@project.namespace, @project, note), target_id: note.noteable.id, target_type: note.noteable.class.name.underscore } }
-  #{note.note}
-%textarea.hidden.js-task-list-field.original-task-list{ data: {update_url: namespace_project_note_path(@project.namespace, @project, note) } }= note.note
diff --git a/app/views/projects/releases/edit.html.haml b/app/views/projects/releases/edit.html.haml
index faa24a3c88e21..93ee9382a6e09 100644
--- a/app/views/projects/releases/edit.html.haml
+++ b/app/views/projects/releases/edit.html.haml
@@ -13,7 +13,7 @@
   = form_for(@release, method: :put, url: namespace_project_tag_release_path(@project.namespace, @project, @tag.name), html: { class: 'form-horizontal common-note-form release-form js-quick-submit' }) do |f|
     = render layout: 'projects/md_preview', locals: { url: preview_markdown_path(@project), referenced_users: true } do
       = render 'projects/zen', f: f, attr: :description, classes: 'note-textarea', placeholder: "Write your release notes or drag files here..."
-      = render 'projects/notes/hints'
+      = render 'shared/notes/hints'
     .error-alert
     .prepend-top-default
       = f.submit 'Save changes', class: 'btn btn-save'
diff --git a/app/views/projects/snippets/show.html.haml b/app/views/projects/snippets/show.html.haml
index 7a175f63eeb06..aab1c043e66ea 100644
--- a/app/views/projects/snippets/show.html.haml
+++ b/app/views/projects/snippets/show.html.haml
@@ -9,4 +9,4 @@
   .row-content-block.top-block.content-component-block
     = render 'award_emoji/awards_block', awardable: @snippet, inline: true
 
-  #notes= render "projects/notes/notes_with_form"
+  #notes= render "shared/notes/notes_with_form"
diff --git a/app/views/projects/tags/new.html.haml b/app/views/projects/tags/new.html.haml
index a6894b9adc001..7c607d2956bf3 100644
--- a/app/views/projects/tags/new.html.haml
+++ b/app/views/projects/tags/new.html.haml
@@ -30,7 +30,7 @@
     .col-sm-10
       = render layout: 'projects/md_preview', locals: { url: preview_markdown_path(@project), referenced_users: true } do
         = render 'projects/zen', attr: :release_description, classes: 'note-textarea', placeholder: "Write your release notes or drag files here..."
-        = render 'projects/notes/hints'
+        = render 'shared/notes/hints'
       .help-block Optionally, add release notes to the tag. They will be stored in the GitLab database and displayed on the tags page.
   .form-actions
     = button_tag 'Create tag', class: 'btn btn-create', tabindex: 3
diff --git a/app/views/projects/wikis/_form.html.haml b/app/views/projects/wikis/_form.html.haml
index 00869aff27bf4..6cb7c1e9c4d0f 100644
--- a/app/views/projects/wikis/_form.html.haml
+++ b/app/views/projects/wikis/_form.html.haml
@@ -14,7 +14,7 @@
     .col-sm-10
       = render layout: 'projects/md_preview', locals: { url: namespace_project_wiki_preview_markdown_path(@project.namespace, @project, @page.slug) } do
         = render 'projects/zen', f: f, attr: :content, classes: 'note-textarea', placeholder: 'Write your content or drag files here...'
-        = render 'projects/notes/hints'
+        = render 'shared/notes/hints'
 
       .clearfix
       .error-alert
diff --git a/app/views/shared/issuable/form/_description.html.haml b/app/views/shared/issuable/form/_description.html.haml
index cbc7125c0d56a..7ef0ae96be287 100644
--- a/app/views/shared/issuable/form/_description.html.haml
+++ b/app/views/shared/issuable/form/_description.html.haml
@@ -17,6 +17,6 @@
                                classes: 'note-textarea',
                                placeholder: "Write a comment or drag your files here...",
                                supports_slash_commands: supports_slash_commands
-      = render 'projects/notes/hints', supports_slash_commands: supports_slash_commands
+      = render 'shared/notes/hints', supports_slash_commands: supports_slash_commands
       .clearfix
       .error-alert
diff --git a/app/views/projects/notes/_comment_button.html.haml b/app/views/shared/notes/_comment_button.html.haml
similarity index 100%
rename from app/views/projects/notes/_comment_button.html.haml
rename to app/views/shared/notes/_comment_button.html.haml
diff --git a/app/views/shared/notes/_edit.html.haml b/app/views/shared/notes/_edit.html.haml
new file mode 100644
index 0000000000000..4a020865828d2
--- /dev/null
+++ b/app/views/shared/notes/_edit.html.haml
@@ -0,0 +1,3 @@
+.original-note-content.hidden{ data: { post_url: note_url(note), target_id: note.noteable.id, target_type: note.noteable.class.name.underscore } }
+  #{note.note}
+%textarea.hidden.js-task-list-field.original-task-list{ data: {update_url: note_url(note) } }= note.note
diff --git a/app/views/projects/notes/_edit_form.html.haml b/app/views/shared/notes/_edit_form.html.haml
similarity index 95%
rename from app/views/projects/notes/_edit_form.html.haml
rename to app/views/shared/notes/_edit_form.html.haml
index 3867072225f88..8923e5602a41f 100644
--- a/app/views/projects/notes/_edit_form.html.haml
+++ b/app/views/shared/notes/_edit_form.html.haml
@@ -4,7 +4,7 @@
     = hidden_field_tag :target_type, '', class: 'js-form-target-type'
     = render layout: 'projects/md_preview', locals: { url: preview_markdown_path(project), referenced_users: true } do
       = render 'projects/zen', attr: 'note[note]', classes: 'note-textarea js-note-text js-task-list-field', placeholder: "Write a comment or drag your files here..."
-      = render 'projects/notes/hints'
+      = render 'shared/notes/hints'
 
     .note-form-actions.clearfix
       .settings-message.note-edit-warning.js-finish-edit-warning
diff --git a/app/views/projects/notes/_form.html.haml b/app/views/shared/notes/_form.html.haml
similarity index 74%
rename from app/views/projects/notes/_form.html.haml
rename to app/views/shared/notes/_form.html.haml
index 46f785fefcafe..eaf50bc2115aa 100644
--- a/app/views/projects/notes/_form.html.haml
+++ b/app/views/shared/notes/_form.html.haml
@@ -4,7 +4,7 @@
 - else
   - preview_url = preview_markdown_path(@project)
 
-= form_for [@project.namespace.becomes(Namespace), @project, @note], remote: true, html: { :'data-type' => 'json', multipart: true, id: nil, class: "new-note js-new-note-form js-quick-submit common-note-form", "data-noteable-iid" => @note.noteable.try(:iid), }, authenticity_token: true do |f|
+= form_for form_resources, url: new_form_url, remote: true, html: { :'data-type' => 'json', multipart: true, id: nil, class: "new-note js-new-note-form js-quick-submit common-note-form", "data-noteable-iid" => @note.noteable.try(:iid), }, authenticity_token: true do |f|
   = hidden_field_tag :view, diff_view
   = hidden_field_tag :line_type
   = hidden_field_tag :merge_request_diff_head_sha, @note.noteable.try(:diff_head_sha)
@@ -28,11 +28,11 @@
       classes: 'note-textarea js-note-text',
       placeholder: "Write a comment or drag your files here...",
       supports_slash_commands: supports_slash_commands
-    = render 'projects/notes/hints', supports_slash_commands: supports_slash_commands
+    = render 'shared/notes/hints', supports_slash_commands: supports_slash_commands
     .error-alert
 
   .note-form-actions.clearfix
-    = render partial: 'projects/notes/comment_button'
+    = render partial: 'shared/notes/comment_button'
 
     = yield(:note_actions)
 
diff --git a/app/views/projects/notes/_hints.html.haml b/app/views/shared/notes/_hints.html.haml
similarity index 100%
rename from app/views/projects/notes/_hints.html.haml
rename to app/views/shared/notes/_hints.html.haml
diff --git a/app/views/shared/notes/_note.html.haml b/app/views/shared/notes/_note.html.haml
index 9657b4eea82f0..071c48fa2e494 100644
--- a/app/views/shared/notes/_note.html.haml
+++ b/app/views/shared/notes/_note.html.haml
@@ -42,10 +42,7 @@
           = note.redacted_note_html
         = edited_time_ago_with_tooltip(note, placement: 'bottom', html_class: 'note_edited_ago', include_author: true)
         - if note_editable
-          - if note.for_personal_snippet?
-            = render 'snippets/notes/edit', note: note
-          - else
-            = render 'projects/notes/edit', note: note
+          = render 'shared/notes/edit', note: note
         .note-awards
           = render 'award_emoji/awards_block', awardable: note, inline: false
         - if note.system
diff --git a/app/views/projects/notes/_notes_with_form.html.haml b/app/views/shared/notes/_notes_with_form.html.haml
similarity index 64%
rename from app/views/projects/notes/_notes_with_form.html.haml
rename to app/views/shared/notes/_notes_with_form.html.haml
index 2a66addb08a1d..9930cbd96d7fd 100644
--- a/app/views/projects/notes/_notes_with_form.html.haml
+++ b/app/views/shared/notes/_notes_with_form.html.haml
@@ -1,18 +1,18 @@
 %ul#notes-list.notes.main-notes-list.timeline
   = render "shared/notes/notes"
 
-= render 'projects/notes/edit_form', project: @project
+= render 'shared/notes/edit_form', project: @project
 
 %ul.notes.notes-form.timeline
   %li.timeline-entry
     .flash-container.timeline-content
 
-    - if can? current_user, :create_note, @project
+    - if  can_create_note?
       .timeline-icon.hidden-xs.hidden-sm
         %a.author_link{ href: user_path(current_user) }
           = image_tag avatar_icon(current_user), alt: current_user.to_reference, class: 'avatar s40'
       .timeline-content.timeline-content-form
-        = render "projects/notes/form", view: diff_view
+        = render "shared/notes/form", view: diff_view
     - elsif !current_user
       .disabled-comment.text-center
         .disabled-comment-text.inline
@@ -23,4 +23,4 @@
           to post a comment
 
 :javascript
-  var notes = new Notes("#{namespace_project_noteable_notes_path(namespace_id: @project.namespace, project_id: @project, target_id: @noteable.id, target_type: @noteable.class.name.underscore)}", #{@notes.map(&:id).to_json}, #{Time.now.to_i}, "#{diff_view}")
+  var notes = new Notes("#{notes_url}", #{@notes.map(&:id).to_json}, #{Time.now.to_i}, "#{diff_view}")
diff --git a/app/views/snippets/notes/_edit.html.haml b/app/views/snippets/notes/_edit.html.haml
deleted file mode 100644
index e69de29bb2d1d..0000000000000
diff --git a/app/views/snippets/notes/_notes.html.haml b/app/views/snippets/notes/_notes.html.haml
deleted file mode 100644
index f07d6b8c1266b..0000000000000
--- a/app/views/snippets/notes/_notes.html.haml
+++ /dev/null
@@ -1,2 +0,0 @@
-%ul#notes-list.notes.main-notes-list.timeline
-  = render "projects/notes/notes"
diff --git a/app/views/snippets/show.html.haml b/app/views/snippets/show.html.haml
index 98287cba5b47c..51dbbc32cc97a 100644
--- a/app/views/snippets/show.html.haml
+++ b/app/views/snippets/show.html.haml
@@ -2,11 +2,11 @@
 
 = render 'shared/snippets/header'
 
-%article.file-holder.snippet-file-content
-  = render 'shared/snippets/blob'
+.personal-snippets
+  %article.file-holder.snippet-file-content
+    = render 'shared/snippets/blob'
 
-.row-content-block.top-block.content-component-block
-  = render 'award_emoji/awards_block', awardable: @snippet, inline: true
+  .row-content-block.top-block.content-component-block
+    = render 'award_emoji/awards_block', awardable: @snippet, inline: true
 
-%ul#notes-list.notes.main-notes-list.timeline
-  #notes= render 'shared/notes/notes'
+  #notes= render "shared/notes/notes_with_form"
diff --git a/changelogs/unreleased/12910-personal-snippets-notes.yml b/changelogs/unreleased/12910-personal-snippets-notes.yml
new file mode 100644
index 0000000000000..7f1576c351342
--- /dev/null
+++ b/changelogs/unreleased/12910-personal-snippets-notes.yml
@@ -0,0 +1,4 @@
+---
+title: Support comments for personal snippets
+merge_request:
+author:
diff --git a/spec/factories/notes.rb b/spec/factories/notes.rb
index 44c3186d8138d..046974dcd6e8a 100644
--- a/spec/factories/notes.rb
+++ b/spec/factories/notes.rb
@@ -29,6 +29,8 @@
 
     factory :discussion_note_on_commit, traits: [:on_commit], class: DiscussionNote
 
+    factory :discussion_note_on_personal_snippet, traits: [:on_personal_snippet], class: DiscussionNote
+
     factory :legacy_diff_note_on_commit, traits: [:on_commit, :legacy_diff_note], class: LegacyDiffNote
 
     factory :legacy_diff_note_on_merge_request, traits: [:on_merge_request, :legacy_diff_note], class: LegacyDiffNote do
diff --git a/spec/features/snippets/notes_on_personal_snippets_spec.rb b/spec/features/snippets/notes_on_personal_snippets_spec.rb
index c646039e0b145..957baac02eb00 100644
--- a/spec/features/snippets/notes_on_personal_snippets_spec.rb
+++ b/spec/features/snippets/notes_on_personal_snippets_spec.rb
@@ -1,6 +1,6 @@
 require 'spec_helper'
 
-describe 'Comments on personal snippets', feature: true do
+describe 'Comments on personal snippets', :js, feature: true do
   let!(:user)    { create(:user) }
   let!(:snippet) { create(:personal_snippet, :public) }
   let!(:snippet_notes) do
@@ -18,7 +18,7 @@
 
   subject { page }
 
-  context 'viewing the snippet detail page' do
+  context 'when viewing the snippet detail page' do
     it 'contains notes for a snippet with correct action icons' do
       expect(page).to have_selector('#notes-list li', count: 2)
 
@@ -36,4 +36,64 @@
       end
     end
   end
+
+  context 'when submitting a note' do
+    it 'shows a valid form' do
+      is_expected.to have_css('.js-main-target-form', visible: true, count: 1)
+      expect(find('.js-main-target-form .js-comment-button').value).
+        to eq('Comment')
+
+      page.within('.js-main-target-form') do
+        expect(page).not_to have_link('Cancel')
+      end
+    end
+
+    it 'previews a note' do
+      fill_in 'note[note]', with: 'This is **awesome**!'
+      find('.js-md-preview-button').click
+
+      page.within('.new-note .md-preview') do
+        expect(page).to have_content('This is awesome!')
+        expect(page).to have_selector('strong')
+      end
+    end
+
+    it 'creates a note' do
+      fill_in 'note[note]', with: 'This is **awesome**!'
+      click_button 'Comment'
+
+      expect(find('div#notes')).to have_content('This is awesome!')
+    end
+  end
+
+  context 'when editing a note' do
+    it 'changes the text' do
+      page.within("#notes-list li#note_#{snippet_notes[0].id}") do
+        click_on 'Edit comment'
+      end
+
+      page.within('.current-note-edit-form') do
+        fill_in 'note[note]', with: 'new content'
+        find('.btn-save').click
+      end
+
+      page.within("#notes-list li#note_#{snippet_notes[0].id}") do
+        expect(page).to have_css('.note_edited_ago')
+        expect(page).to have_content('new content')
+        expect(find('.note_edited_ago').text).to match(/less than a minute ago/)
+      end
+    end
+  end
+
+  context 'when deleting a note' do
+    it 'removes the note from the snippet detail page' do
+      page.within("#notes-list li#note_#{snippet_notes[0].id}") do
+        click_on 'Remove comment'
+      end
+
+      wait_for_ajax
+
+      expect(page).not_to have_selector("#notes-list li#note_#{snippet_notes[0].id}")
+    end
+  end
 end
diff --git a/spec/helpers/notes_helper_spec.rb b/spec/helpers/notes_helper_spec.rb
index 6c990f9417508..099146678ae2d 100644
--- a/spec/helpers/notes_helper_spec.rb
+++ b/spec/helpers/notes_helper_spec.rb
@@ -175,4 +175,79 @@
       end
     end
   end
+
+  describe '#notes_url' do
+    it 'return snippet notes path for personal snippet' do
+      @snippet = create(:personal_snippet)
+
+      expect(helper.notes_url).to eq("/snippets/#{@snippet.id}/notes")
+    end
+
+    it 'return project notes path for project snippet' do
+      namespace = create(:namespace, path: 'nm')
+      @project = create(:empty_project, path: 'test', namespace: namespace)
+      @snippet = create(:project_snippet, project: @project)
+      @noteable = @snippet
+
+      expect(helper.notes_url).to eq("/nm/test/noteable/project_snippet/#{@noteable.id}/notes")
+    end
+
+    it 'return project notes path for other noteables' do
+      namespace = create(:namespace, path: 'nm')
+      @project = create(:empty_project, path: 'test', namespace: namespace)
+      @noteable = create(:issue, project: @project)
+
+      expect(helper.notes_url).to eq("/nm/test/noteable/issue/#{@noteable.id}/notes")
+    end
+  end
+
+  describe '#note_url' do
+    it 'return snippet notes path for personal snippet' do
+      note = create(:note_on_personal_snippet)
+
+      expect(helper.note_url(note)).to eq("/snippets/#{note.noteable.id}/notes/#{note.id}")
+    end
+
+    it 'return project notes path for project snippet' do
+      namespace = create(:namespace, path: 'nm')
+      @project = create(:empty_project, path: 'test', namespace: namespace)
+      note = create(:note_on_project_snippet, project: @project)
+
+      expect(helper.note_url(note)).to eq("/nm/test/notes/#{note.id}")
+    end
+
+    it 'return project notes path for other noteables' do
+      namespace = create(:namespace, path: 'nm')
+      @project = create(:empty_project, path: 'test', namespace: namespace)
+      note = create(:note_on_issue, project: @project)
+
+      expect(helper.note_url(note)).to eq("/nm/test/notes/#{note.id}")
+    end
+  end
+
+  describe '#form_resurces' do
+    it 'returns note for personal snippet' do
+      @snippet = create(:personal_snippet)
+      @note = create(:note_on_personal_snippet)
+
+      expect(helper.form_resources).to eq([@note])
+    end
+
+    it 'returns namespace, project and note for project snippet' do
+      namespace = create(:namespace, path: 'nm')
+      @project = create(:empty_project, path: 'test', namespace: namespace)
+      @snippet = create(:project_snippet, project: @project)
+      @note = create(:note_on_personal_snippet)
+
+      expect(helper.form_resources).to eq([@project.namespace, @project, @note])
+    end
+
+    it 'returns namespace, project and note path for other noteables' do
+      namespace = create(:namespace, path: 'nm')
+      @project = create(:empty_project, path: 'test', namespace: namespace)
+      @note = create(:note_on_issue, project: @project)
+
+      expect(helper.form_resources).to eq([@project.namespace, @project, @note])
+    end
+  end
 end
diff --git a/spec/services/notes/build_service_spec.rb b/spec/services/notes/build_service_spec.rb
index f9dd5541b10b2..133175769ca49 100644
--- a/spec/services/notes/build_service_spec.rb
+++ b/spec/services/notes/build_service_spec.rb
@@ -29,10 +29,82 @@
           expect(new_note.errors[:base]).to include('Discussion to reply to cannot be found')
         end
       end
+
+      context 'personal snippet note' do
+        def reply(note, user = nil)
+          user ||= create(:user)
+
+          described_class.new(nil,
+                              user,
+                              note: 'Test',
+                              in_reply_to_discussion_id: note.discussion_id).execute
+        end
+
+        let(:snippet_author) { create(:user) }
+
+        context 'when a snippet is public' do
+          it 'creates a reply note' do
+            snippet = create(:personal_snippet, :public)
+            note = create(:discussion_note_on_personal_snippet, noteable: snippet)
+
+            new_note = reply(note)
+
+            expect(new_note).to be_valid
+            expect(new_note.in_reply_to?(note)).to be_truthy
+          end
+        end
+
+        context 'when a snippet is private' do
+          let(:snippet) { create(:personal_snippet, :private, author: snippet_author) }
+          let(:note) { create(:discussion_note_on_personal_snippet, noteable: snippet) }
+
+          it 'creates a reply note when the author replies' do
+            new_note = reply(note, snippet_author)
+
+            expect(new_note).to be_valid
+            expect(new_note.in_reply_to?(note)).to be_truthy
+          end
+
+          it 'sets an error when another user replies' do
+            new_note = reply(note)
+
+            expect(new_note.errors[:base]).to include('Discussion to reply to cannot be found')
+          end
+        end
+
+        context 'when a snippet is internal' do
+          let(:snippet) { create(:personal_snippet, :internal, author: snippet_author) }
+          let(:note) { create(:discussion_note_on_personal_snippet, noteable: snippet) }
+
+          it 'creates a reply note when the author replies' do
+            new_note = reply(note, snippet_author)
+
+            expect(new_note).to be_valid
+            expect(new_note.in_reply_to?(note)).to be_truthy
+          end
+
+          it 'creates a reply note when a regular user replies' do
+            new_note = reply(note)
+
+            expect(new_note).to be_valid
+            expect(new_note.in_reply_to?(note)).to be_truthy
+          end
+
+          it 'sets an error when an external user replies' do
+            new_note = reply(note, create(:user, :external))
+
+            expect(new_note.errors[:base]).to include('Discussion to reply to cannot be found')
+          end
+        end
+      end
     end
 
     it 'builds a note without saving it' do
-      new_note = described_class.new(project, author, noteable_type: note.noteable_type, noteable_id: note.noteable_id, note: 'Test').execute
+      new_note = described_class.new(project,
+                                    author,
+                                    noteable_type: note.noteable_type,
+                                    noteable_id: note.noteable_id,
+                                    note: 'Test').execute
       expect(new_note).to be_valid
       expect(new_note).not_to be_persisted
     end
diff --git a/spec/views/projects/notes/_form.html.haml_spec.rb b/spec/views/shared/notes/_form.html.haml_spec.rb
similarity index 96%
rename from spec/views/projects/notes/_form.html.haml_spec.rb
rename to spec/views/shared/notes/_form.html.haml_spec.rb
index a364f9bce92b0..d7d0a5bf56a9b 100644
--- a/spec/views/projects/notes/_form.html.haml_spec.rb
+++ b/spec/views/shared/notes/_form.html.haml_spec.rb
@@ -1,6 +1,6 @@
 require 'spec_helper'
 
-describe 'projects/notes/_form' do
+describe 'shared/notes/_form' do
   include Devise::Test::ControllerHelpers
 
   let(:user) { create(:user) }
-- 
GitLab


From 5249157552bbf4cbf279b1decbd4a0e90e056077 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Alejandro=20Rodr=C3=ADguez?= <alejorro70@gmail.com>
Date: Tue, 2 May 2017 18:15:12 -0300
Subject: [PATCH 286/363] Allow gl-repository strings as project identifiers in
 PostReceive worker

---
 app/workers/post_receive.rb       | 31 ++++++++++++----------
 lib/gitlab/git_post_receive.rb    | 29 +++------------------
 spec/workers/post_receive_spec.rb | 43 +++++++++++++++++++++++--------
 3 files changed, 53 insertions(+), 50 deletions(-)

diff --git a/app/workers/post_receive.rb b/app/workers/post_receive.rb
index 015a41b6e82e9..39f03983821fd 100644
--- a/app/workers/post_receive.rb
+++ b/app/workers/post_receive.rb
@@ -2,27 +2,24 @@ class PostReceive
   include Sidekiq::Worker
   include DedicatedSidekiqQueue
 
-  def perform(repo_path, identifier, changes)
-    repo_relative_path = Gitlab::RepoPath.strip_storage_path(repo_path)
+  def perform(project_identifier, identifier, changes)
+    project, is_wiki = parse_project_identifier(project_identifier)
+
+    if project.nil?
+      log("Triggered hook for non-existing project with identifier \"#{project_identifier}\"")
+      return false
+    end
 
     changes = Base64.decode64(changes) unless changes.include?(' ')
     # Use Sidekiq.logger so arguments can be correlated with execution
     # time and thread ID's.
     Sidekiq.logger.info "changes: #{changes.inspect}" if ENV['SIDEKIQ_LOG_ARGUMENTS']
-    post_received = Gitlab::GitPostReceive.new(repo_relative_path, identifier, changes)
-
-    if post_received.project.nil?
-      log("Triggered hook for non-existing project with full path \"#{repo_relative_path}\"")
-      return false
-    end
+    post_received = Gitlab::GitPostReceive.new(project, identifier, changes)
 
-    if post_received.wiki?
+    if is_wiki
       # Nothing defined here yet.
-    elsif post_received.regular_project?
-      process_project_changes(post_received)
     else
-      log("Triggered hook for unidentifiable repository type with full path \"#{repo_relative_path}\"")
-      false
+      process_project_changes(post_received)
     end
   end
 
@@ -47,6 +44,14 @@ def process_project_changes(post_received)
 
   private
 
+  def parse_project_identifier(project_identifier)
+    if project_identifier.start_with?('/')
+      Gitlab::RepoPath.parse(project_identifier)
+    else
+      Gitlab::GlRepository.parse(project_identifier)
+    end
+  end
+
   def log(message)
     Gitlab::GitLogger.error("POST-RECEIVE: #{message}")
   end
diff --git a/lib/gitlab/git_post_receive.rb b/lib/gitlab/git_post_receive.rb
index 6babea144c790..0e14253ab4e87 100644
--- a/lib/gitlab/git_post_receive.rb
+++ b/lib/gitlab/git_post_receive.rb
@@ -1,25 +1,12 @@
 module Gitlab
   class GitPostReceive
     include Gitlab::Identifier
-    attr_reader :repo_path, :identifier, :changes, :project
+    attr_reader :project, :identifier, :changes
 
-    def initialize(repo_path, identifier, changes)
-      repo_path.gsub!(/\.git\z/, '')
-      repo_path.gsub!(/\A\//, '')
-
-      @repo_path = repo_path
+    def initialize(project, identifier, changes)
+      @project = project
       @identifier = identifier
       @changes = deserialize_changes(changes)
-
-      retrieve_project_and_type
-    end
-
-    def wiki?
-      @type == :wiki
-    end
-
-    def regular_project?
-      @type == :project
     end
 
     def identify(revision)
@@ -28,16 +15,6 @@ def identify(revision)
 
     private
 
-    def retrieve_project_and_type
-      @type = :project
-      @project = Project.find_by_full_path(@repo_path)
-
-      if @repo_path.end_with?('.wiki') && !@project
-        @type = :wiki
-        @project = Project.find_by_full_path(@repo_path.gsub(/\.wiki\z/, ''))
-      end
-    end
-
     def deserialize_changes(changes)
       changes = utf8_encode_changes(changes)
       changes.lines
diff --git a/spec/workers/post_receive_spec.rb b/spec/workers/post_receive_spec.rb
index 5ab3c4a0e341b..0260416dbe2e5 100644
--- a/spec/workers/post_receive_spec.rb
+++ b/spec/workers/post_receive_spec.rb
@@ -5,6 +5,7 @@
   let(:wrongly_encoded_changes) { changes.encode("ISO-8859-1").force_encoding("UTF-8") }
   let(:base64_changes) { Base64.encode64(wrongly_encoded_changes) }
   let(:project) { create(:project, :repository) }
+  let(:project_identifier) { "project-#{project.id}" }
   let(:key) { create(:key, user: project.owner) }
   let(:key_id) { key.shell_id }
 
@@ -14,6 +15,26 @@
     end
   end
 
+  context 'with a non-existing project' do
+    let(:project_identifier) { "project-123456789" }
+    let(:error_message) do
+      "Triggered hook for non-existing project with identifier \"#{project_identifier}\""
+    end
+
+    it "returns false and logs an error" do
+      expect(Gitlab::GitLogger).to receive(:error).with("POST-RECEIVE: #{error_message}")
+      expect(described_class.new.perform(project_identifier, key_id, base64_changes)).to be(false)
+    end
+  end
+
+  context "with an absolute path as the project identifier" do
+    it "searches the project by full path" do
+      expect(Project).to receive(:find_by_full_path).with(project.full_path).and_call_original
+
+      described_class.new.perform(pwd(project), key_id, base64_changes)
+    end
+  end
+
   describe "#process_project_changes" do
     before do
       allow_any_instance_of(Gitlab::GitPostReceive).to receive(:identify).and_return(project.owner)
@@ -25,7 +46,7 @@
       it "calls GitTagPushService" do
         expect_any_instance_of(GitPushService).to receive(:execute).and_return(true)
         expect_any_instance_of(GitTagPushService).not_to receive(:execute)
-        described_class.new.perform(pwd(project), key_id, base64_changes)
+        described_class.new.perform(project_identifier, key_id, base64_changes)
       end
     end
 
@@ -35,7 +56,7 @@
       it "calls GitTagPushService" do
         expect_any_instance_of(GitPushService).not_to receive(:execute)
         expect_any_instance_of(GitTagPushService).to receive(:execute).and_return(true)
-        described_class.new.perform(pwd(project), key_id, base64_changes)
+        described_class.new.perform(project_identifier, key_id, base64_changes)
       end
     end
 
@@ -45,12 +66,12 @@
       it "does not call any of the services" do
         expect_any_instance_of(GitPushService).not_to receive(:execute)
         expect_any_instance_of(GitTagPushService).not_to receive(:execute)
-        described_class.new.perform(pwd(project), key_id, base64_changes)
+        described_class.new.perform(project_identifier, key_id, base64_changes)
       end
     end
 
     context "gitlab-ci.yml" do
-      subject { described_class.new.perform(pwd(project), key_id, base64_changes) }
+      subject { described_class.new.perform(project_identifier, key_id, base64_changes) }
 
       context "creates a Ci::Pipeline for every change" do
         before do
@@ -74,8 +95,8 @@
 
   context "webhook" do
     it "fetches the correct project" do
-      expect(Project).to receive(:find_by_full_path).with(project.path_with_namespace).and_return(project)
-      described_class.new.perform(pwd(project), key_id, base64_changes)
+      expect(Project).to receive(:find_by).with(id: project.id.to_s)
+      described_class.new.perform(project_identifier, key_id, base64_changes)
     end
 
     it "does not run if the author is not in the project" do
@@ -85,22 +106,22 @@
 
       expect(project).not_to receive(:execute_hooks)
 
-      expect(described_class.new.perform(pwd(project), key_id, base64_changes)).to be_falsey
+      expect(described_class.new.perform(project_identifier, key_id, base64_changes)).to be_falsey
     end
 
     it "asks the project to trigger all hooks" do
-      allow(Project).to receive(:find_by_full_path).and_return(project)
+      allow(Project).to receive(:find_by).and_return(project)
       expect(project).to receive(:execute_hooks).twice
       expect(project).to receive(:execute_services).twice
 
-      described_class.new.perform(pwd(project), key_id, base64_changes)
+      described_class.new.perform(project_identifier, key_id, base64_changes)
     end
 
     it "enqueues a UpdateMergeRequestsWorker job" do
-      allow(Project).to receive(:find_by_full_path).and_return(project)
+      allow(Project).to receive(:find_by).and_return(project)
       expect(UpdateMergeRequestsWorker).to receive(:perform_async).with(project.id, project.owner.id, any_args)
 
-      described_class.new.perform(pwd(project), key_id, base64_changes)
+      described_class.new.perform(project_identifier, key_id, base64_changes)
     end
   end
 
-- 
GitLab


From 8bc381db90c92bca6ba868d1588af1ad1a41073b Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Alejandro=20Rodr=C3=ADguez?= <alejorro70@gmail.com>
Date: Wed, 3 May 2017 18:07:54 -0300
Subject: [PATCH 287/363] Pass GL_REPOSITORY in Workhorse responses

---
 app/controllers/projects/git_http_controller.rb |  2 +-
 app/workers/post_receive.rb                     |  7 +++++++
 lib/api/internal.rb                             |  2 +-
 lib/gitlab/gl_repository.rb                     |  4 ++++
 lib/gitlab/workhorse.rb                         |  6 ++++--
 spec/lib/gitlab/workhorse_spec.rb               | 17 +++++++++++++++--
 6 files changed, 32 insertions(+), 6 deletions(-)

diff --git a/app/controllers/projects/git_http_controller.rb b/app/controllers/projects/git_http_controller.rb
index 10adddb463669..9e4edcae10107 100644
--- a/app/controllers/projects/git_http_controller.rb
+++ b/app/controllers/projects/git_http_controller.rb
@@ -59,7 +59,7 @@ def git_command
 
   def render_ok
     set_workhorse_internal_api_content_type
-    render json: Gitlab::Workhorse.git_http_ok(repository, user, action_name)
+    render json: Gitlab::Workhorse.git_http_ok(repository, wiki?, user, action_name)
   end
 
   def render_http_not_allowed
diff --git a/app/workers/post_receive.rb b/app/workers/post_receive.rb
index 39f03983821fd..127d8dfbb6171 100644
--- a/app/workers/post_receive.rb
+++ b/app/workers/post_receive.rb
@@ -44,6 +44,13 @@ def process_project_changes(post_received)
 
   private
 
+  # To maintain backwards compatibility, we accept both gl_repository or
+  # repository paths as project identifiers. Our plan is to migrate to
+  # gl_repository only with the following plan:
+  # 9.2: Handle both possible values. Keep Gitlab-Shell sending only repo paths
+  # 9.3 (or patch release): Make GitLab Shell pass gl_repository if present
+  # 9.4 (or patch release): Make GitLab Shell always pass gl_repository
+  # 9.5 (or patch release): Handle only gl_repository as project identifier on this method
   def parse_project_identifier(project_identifier)
     if project_identifier.start_with?('/')
       Gitlab::RepoPath.parse(project_identifier)
diff --git a/lib/api/internal.rb b/lib/api/internal.rb
index ddb2047f68621..2a11790b215d4 100644
--- a/lib/api/internal.rb
+++ b/lib/api/internal.rb
@@ -44,7 +44,7 @@ class Internal < Grape::API
 
           # Project id to pass between components that don't share/don't have
           # access to the same filesystem mounts
-          response[:gl_repository] = "#{wiki? ? 'wiki' : 'project'}-#{project.id}"
+          response[:gl_repository] = Gitlab::GlRepository.gl_repository(project, wiki?)
 
           # Return the repository full path so that gitlab-shell has it when
           # handling ssh commands
diff --git a/lib/gitlab/gl_repository.rb b/lib/gitlab/gl_repository.rb
index 6997546049ebb..07c0abcce235d 100644
--- a/lib/gitlab/gl_repository.rb
+++ b/lib/gitlab/gl_repository.rb
@@ -1,5 +1,9 @@
 module Gitlab
   module GlRepository
+    def self.gl_repository(project, is_wiki)
+      "#{is_wiki ? 'wiki' : 'project'}-#{project.id}"
+    end
+
     def self.parse(gl_repository)
       match_data = /\A(project|wiki)-([1-9][0-9]*)\z/.match(gl_repository)
       unless match_data
diff --git a/lib/gitlab/workhorse.rb b/lib/gitlab/workhorse.rb
index c551f939df19a..8c5ad01e8c2ee 100644
--- a/lib/gitlab/workhorse.rb
+++ b/lib/gitlab/workhorse.rb
@@ -16,15 +16,17 @@ class Workhorse
     SECRET_LENGTH = 32
 
     class << self
-      def git_http_ok(repository, user, action)
+      def git_http_ok(repository, is_wiki, user, action)
+        project = repository.project
         repo_path = repository.path_to_repo
         params = {
           GL_ID: Gitlab::GlId.gl_id(user),
+          GL_REPOSITORY: Gitlab::GlRepository.gl_repository(project, is_wiki),
           RepoPath: repo_path,
         }
 
         if Gitlab.config.gitaly.enabled
-          address = Gitlab::GitalyClient.get_address(repository.project.repository_storage)
+          address = Gitlab::GitalyClient.get_address(project.repository_storage)
           params[:Repository] = repository.gitaly_repository.to_h
 
           feature_enabled = case action.to_s
diff --git a/spec/lib/gitlab/workhorse_spec.rb b/spec/lib/gitlab/workhorse_spec.rb
index b703e9808a8c3..beb1791a42968 100644
--- a/spec/lib/gitlab/workhorse_spec.rb
+++ b/spec/lib/gitlab/workhorse_spec.rb
@@ -181,10 +181,23 @@ def call_verify(headers)
     let(:user) { create(:user) }
     let(:repo_path) { repository.path_to_repo }
     let(:action) { 'info_refs' }
+    let(:params) do
+      { GL_ID: "user-#{user.id}", GL_REPOSITORY: "project-#{project.id}", RepoPath: repo_path }
+    end
+
+    subject { described_class.git_http_ok(repository, false, user, action) }
+
+    it { expect(subject).to include(params) }
 
-    subject { described_class.git_http_ok(repository, user, action) }
+    context 'when is_wiki' do
+      let(:params) do
+        { GL_ID: "user-#{user.id}", GL_REPOSITORY: "wiki-#{project.id}", RepoPath: repo_path }
+      end
+
+      subject { described_class.git_http_ok(repository, true, user, action) }
 
-    it { expect(subject).to include({ GL_ID: "user-#{user.id}", RepoPath: repo_path }) }
+      it { expect(subject).to include(params) }
+    end
 
     context 'when Gitaly is enabled' do
       let(:gitaly_params) do
-- 
GitLab


From 79b8323d11cc06911b996f327c6e06fd29cafea4 Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Fri, 5 May 2017 16:57:27 +0300
Subject: [PATCH 288/363] [Multiple issue assignees] Fix issue atom feed

---
 app/views/issues/_issue.atom.builder | 11 +++++++++--
 spec/features/atom/issues_spec.rb    |  6 ++++--
 2 files changed, 13 insertions(+), 4 deletions(-)

diff --git a/app/views/issues/_issue.atom.builder b/app/views/issues/_issue.atom.builder
index 9ec765c45f927..2ed78bb3b6586 100644
--- a/app/views/issues/_issue.atom.builder
+++ b/app/views/issues/_issue.atom.builder
@@ -26,9 +26,16 @@ xml.entry do
   if issue.assignees.any?
     xml.assignees do
       issue.assignees.each do |assignee|
-        xml.name assignee.name
-        xml.email assignee.public_email
+        xml.assignee do
+          xml.name assignee.name
+          xml.email assignee.public_email
+        end
       end
     end
+
+    xml.assignee do
+      xml.name issue.assignees.first.name
+      xml.email issue.assignees.first.public_email
+    end
   end
 end
diff --git a/spec/features/atom/issues_spec.rb b/spec/features/atom/issues_spec.rb
index 78f8f46a04eb2..4f6754ad54105 100644
--- a/spec/features/atom/issues_spec.rb
+++ b/spec/features/atom/issues_spec.rb
@@ -22,7 +22,8 @@
           to have_content('application/atom+xml')
         expect(body).to have_selector('title', text: "#{project.name} issues")
         expect(body).to have_selector('author email', text: issue.author_public_email)
-        expect(body).to have_selector('assignees email', text: issue.author_public_email)
+        expect(body).to have_selector('assignees assignee email', text: issue.assignees.first.public_email)
+        expect(body).to have_selector('assignee email', text: issue.assignees.first.public_email)
         expect(body).to have_selector('entry summary', text: issue.title)
       end
     end
@@ -36,7 +37,8 @@
           to have_content('application/atom+xml')
         expect(body).to have_selector('title', text: "#{project.name} issues")
         expect(body).to have_selector('author email', text: issue.author_public_email)
-        expect(body).to have_selector('assignees email', text: issue.author_public_email)
+        expect(body).to have_selector('assignees assignee email', text: issue.assignees.first.public_email)
+        expect(body).to have_selector('assignee email', text: issue.assignees.first.public_email)
         expect(body).to have_selector('entry summary', text: issue.title)
       end
     end
-- 
GitLab


From 4c0adb9ee9ee6631b7cda6562d0e20b7daab8a9e Mon Sep 17 00:00:00 2001
From: James Edwards-Jones <jedwardsjones@gitlab.com>
Date: Sun, 16 Apr 2017 02:33:01 +0100
Subject: [PATCH 289/363] Build failures summary page for pipelines

---
 app/assets/javascripts/dispatcher.js          |  1 +
 app/assets/stylesheets/pages/pipelines.scss   | 26 +++++++++++++
 .../projects/pipelines_controller.rb          | 20 +++++++---
 app/helpers/builds_helper.rb                  |  8 ++++
 .../projects/pipelines/_with_tabs.html.haml   | 16 +++++++-
 .../24883-build-failure-summary-page.yml      |  4 ++
 config/routes/project.rb                      |  1 +
 .../projects/pipelines/pipeline_spec.rb       | 38 +++++++++++++++++++
 8 files changed, 106 insertions(+), 8 deletions(-)
 create mode 100644 changelogs/unreleased/24883-build-failure-summary-page.yml

diff --git a/app/assets/javascripts/dispatcher.js b/app/assets/javascripts/dispatcher.js
index f277e1dddc75a..32cc6623c7ede 100644
--- a/app/assets/javascripts/dispatcher.js
+++ b/app/assets/javascripts/dispatcher.js
@@ -232,6 +232,7 @@ const ShortcutsBlob = require('./shortcuts_blob');
           }
           break;
         case 'projects:pipelines:builds':
+        case 'projects:pipelines:failures':
         case 'projects:pipelines:show':
           const { controllerAction } = document.querySelector('.js-pipeline-container').dataset;
           const pipelineStatusUrl = `${document.querySelector('.js-pipeline-tab-link a').getAttribute('href')}/status.json`;
diff --git a/app/assets/stylesheets/pages/pipelines.scss b/app/assets/stylesheets/pages/pipelines.scss
index a4fe652b52fa6..22500698af3aa 100644
--- a/app/assets/stylesheets/pages/pipelines.scss
+++ b/app/assets/stylesheets/pages/pipelines.scss
@@ -316,6 +316,32 @@
   }
 }
 
+.build-failures {
+  .build-state {
+    padding: 20px 2px;
+
+    .build-name {
+      float: right;
+      font-weight: 500;
+    }
+
+    .ci-status-icon-failed svg {
+      vertical-align: middle;
+    }
+
+    .stage {
+      color: $gl-text-color-secondary;
+      font-weight: 500;
+      vertical-align: middle;
+    }
+  }
+
+  .build-log {
+    border: none;
+    line-height: initial;
+  }
+}
+
 // Pipeline graph
 .pipeline-graph {
   width: 100%;
diff --git a/app/controllers/projects/pipelines_controller.rb b/app/controllers/projects/pipelines_controller.rb
index 1780cc0233c6c..915f0bc63f768 100644
--- a/app/controllers/projects/pipelines_controller.rb
+++ b/app/controllers/projects/pipelines_controller.rb
@@ -1,6 +1,6 @@
 class Projects::PipelinesController < Projects::ApplicationController
   before_action :pipeline, except: [:index, :new, :create, :charts]
-  before_action :commit, only: [:show, :builds]
+  before_action :commit, only: [:show, :builds, :failures]
   before_action :authorize_read_pipeline!
   before_action :authorize_create_pipeline!, only: [:new, :create]
   before_action :authorize_update_pipeline!, only: [:retry, :cancel]
@@ -67,11 +67,11 @@ def show
   end
 
   def builds
-    respond_to do |format|
-      format.html do
-        render 'show'
-      end
-    end
+    render_show
+  end
+
+  def failures
+    render_show
   end
 
   def status
@@ -111,6 +111,14 @@ def charts
 
   private
 
+  def render_show
+    respond_to do |format|
+      format.html do
+        render 'show'
+      end
+    end
+  end
+
   def create_params
     params.require(:pipeline).permit(:ref)
   end
diff --git a/app/helpers/builds_helper.rb b/app/helpers/builds_helper.rb
index 2fcb7a59fc32a..0145029fb6005 100644
--- a/app/helpers/builds_helper.rb
+++ b/app/helpers/builds_helper.rb
@@ -1,4 +1,12 @@
 module BuildsHelper
+  def build_summary(build)
+    if build.has_trace?
+      build.trace.html(last_lines: 10).html_safe
+    else
+      "No job trace"
+    end
+  end
+
   def sidebar_build_class(build, current_build)
     build_class = ''
     build_class += ' active' if build.id === current_build.id
diff --git a/app/views/projects/pipelines/_with_tabs.html.haml b/app/views/projects/pipelines/_with_tabs.html.haml
index d7cefb8613e4a..76eb8533cc3b6 100644
--- a/app/views/projects/pipelines/_with_tabs.html.haml
+++ b/app/views/projects/pipelines/_with_tabs.html.haml
@@ -7,8 +7,10 @@
       = link_to builds_namespace_project_pipeline_path(@project.namespace, @project, @pipeline), data: {target: 'div#js-tab-builds', action: 'builds', toggle: 'tab' }, class: 'builds-tab' do
         Jobs
         %span.badge.js-builds-counter= pipeline.statuses.count
-
-
+    %li.js-failures-tab-link
+      = link_to failures_namespace_project_pipeline_path(@project.namespace, @project, @pipeline), data: {target: 'div#js-tab-failures', action: 'failures', toggle: 'tab' }, class: 'failures-tab' do
+        Failures
+        %span.badge.js-failures-counter= pipeline.statuses.latest.failed.count
 
 .tab-content
   #js-tab-pipeline.tab-pane
@@ -39,3 +41,13 @@
             %th Coverage
             %th
         = render partial: "projects/stage/stage", collection: pipeline.stages, as: :stage
+  #js-tab-failures.build-failures.tab-pane
+    - failed = @pipeline.statuses.latest.failed
+    - failed.each do |build|
+      .build-state
+        %span.ci-status-icon-failed= custom_icon('icon_status_failed')
+        %span.stage
+          = build.stage.titleize
+        %span.build-name
+          = render "notify/links/#{build.to_partial_path}", pipeline: @pipeline, build: build
+      %pre.build-log= build_summary(build)
diff --git a/changelogs/unreleased/24883-build-failure-summary-page.yml b/changelogs/unreleased/24883-build-failure-summary-page.yml
new file mode 100644
index 0000000000000..214cd3e2bc71f
--- /dev/null
+++ b/changelogs/unreleased/24883-build-failure-summary-page.yml
@@ -0,0 +1,4 @@
+---
+title: Added build failures summary page for pipelines
+merge_request: 10719
+author:
diff --git a/config/routes/project.rb b/config/routes/project.rb
index f5009186344eb..5f5b41afbf4c9 100644
--- a/config/routes/project.rb
+++ b/config/routes/project.rb
@@ -155,6 +155,7 @@
           post :cancel
           post :retry
           get :builds
+          get :failures
           get :status
         end
       end
diff --git a/spec/features/projects/pipelines/pipeline_spec.rb b/spec/features/projects/pipelines/pipeline_spec.rb
index 5a53e48f5f81d..ab7c002704d0e 100644
--- a/spec/features/projects/pipelines/pipeline_spec.rb
+++ b/spec/features/projects/pipelines/pipeline_spec.rb
@@ -254,4 +254,42 @@
       it { expect(build_manual.reload).to be_pending }
     end
   end
+
+  describe 'GET /:project/pipelines/:id/failures' do
+    let(:project) { create(:project) }
+    let(:pipeline) { create(:ci_pipeline, project: project, ref: 'master', sha: project.commit.id) }
+    let(:pipeline_failures_page) { failures_namespace_project_pipeline_path(project.namespace, project, pipeline) }
+    let!(:failed_build) { create(:ci_build, :failed, pipeline: pipeline) }
+
+    context 'with failed build' do
+      before do
+        failed_build.trace.set('4 examples, 1 failure')
+
+        visit pipeline_failures_page
+      end
+
+      it 'shows jobs tab pane as active' do
+        expect(page).to have_css('#js-tab-failures.active')
+      end
+
+      it 'lists failed builds' do
+        expect(page).to have_content(failed_build.name)
+        expect(page).to have_content(failed_build.stage)
+      end
+
+      it 'shows build failure logs' do
+        expect(page).to have_content('4 examples, 1 failure')
+      end
+    end
+
+    context 'when missing build logs' do
+      before do
+        visit pipeline_failures_page
+      end
+
+      it 'includes failed jobs' do
+        expect(page).to have_content('No job trace')
+      end
+    end
+  end
 end
-- 
GitLab


From ce418036c763219df8239632f71ef0e9782be7ea Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 16:16:02 +0200
Subject: [PATCH 290/363] add callbacks in bulk

---
 app/services/projects/propagate_service.rb    | 30 +++++++++++++++----
 .../projects/propagate_service_spec.rb        | 18 +++++++++++
 2 files changed, 42 insertions(+), 6 deletions(-)

diff --git a/app/services/projects/propagate_service.rb b/app/services/projects/propagate_service.rb
index c420f24fe028b..f4fae478609f1 100644
--- a/app/services/projects/propagate_service.rb
+++ b/app/services/projects/propagate_service.rb
@@ -35,12 +35,12 @@ def bulk_create_from_template(batch)
         service_hash.merge('project_id' => project_id).values
       end
 
-      # Project.transaction do
-      #   Service.create!(service_hash_list)
-      # end
-      Gitlab::SQL::BulkInsert.new(service_hash.keys + ['project_id'],
-                                  service_list,
-                                  'services').execute
+      Project.transaction do
+        Gitlab::SQL::BulkInsert.new(service_hash.keys + ['project_id'],
+                                    service_list,
+                                    'services').execute
+        run_callbacks(batch)
+      end
     end
 
     def project_ids_batch
@@ -72,5 +72,23 @@ def service_hash
           end
         end
     end
+
+    def run_callbacks(batch)
+      if active_external_issue_tracker?
+        Project.where(id: batch).update_all(has_external_issue_tracker: true)
+      end
+
+      if active_external_wiki?
+        Project.where(id: batch).update_all(has_external_wiki: true)
+      end
+    end
+
+    def active_external_issue_tracker?
+      @template['category'] == 'issue_tracker' && @template['active'] && !@template['default']
+    end
+
+    def active_external_wiki?
+      @template['type'] == 'ExternalWikiService' && @template['active']
+    end
   end
 end
diff --git a/spec/services/projects/propagate_service_spec.rb b/spec/services/projects/propagate_service_spec.rb
index ac25c8b3d561b..b8aa4de5bd17a 100644
--- a/spec/services/projects/propagate_service_spec.rb
+++ b/spec/services/projects/propagate_service_spec.rb
@@ -78,5 +78,23 @@
           to change { Service.count }.by(project_total + 1)
       end
     end
+
+    describe 'external tracker' do
+      it 'updates the project external tracker' do
+        service_template.update(category: 'issue_tracker', default: false)
+
+        expect { described_class.propagate(service_template) }.
+          to change { project.reload.has_external_issue_tracker }.to(true)
+      end
+    end
+
+    describe 'external wiki' do
+      it 'updates the project external tracker' do
+        service_template.update(type: 'ExternalWikiService')
+
+        expect { described_class.propagate(service_template) }.
+          to change { project.reload.has_external_wiki }.to(true)
+      end
+    end
   end
 end
-- 
GitLab


From 40f51c8e8172933ec511dd828a589abb52d83c45 Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Fri, 5 May 2017 09:19:58 -0500
Subject: [PATCH 291/363] [skip ci] Fix FE conflicts with master

---
 .../javascripts/boards/components/issue_card_inner.js  |  2 +-
 app/assets/javascripts/boards/models/assignee.js       |  9 ---------
 app/assets/javascripts/boards/models/issue.js          |  9 +--------
 .../sidebar/components/assignees/assignees.js          |  2 +-
 config/webpack.config.js                               |  5 -----
 spec/javascripts/boards/issue_card_spec.js             |  4 ++--
 spec/support/time_tracking_shared_examples.rb          | 10 ----------
 7 files changed, 5 insertions(+), 36 deletions(-)

diff --git a/app/assets/javascripts/boards/components/issue_card_inner.js b/app/assets/javascripts/boards/components/issue_card_inner.js
index 2f06d186c508b..710207db0c741 100644
--- a/app/assets/javascripts/boards/components/issue_card_inner.js
+++ b/app/assets/javascripts/boards/components/issue_card_inner.js
@@ -157,7 +157,7 @@ gl.issueBoards.IssueCardInner = Vue.extend({
           >
             <img
               class="avatar avatar-inline s20"
-              :src="assignee.avatarUrl"
+              :src="assignee.avatar"
               width="20"
               height="20"
               :alt="avatarUrlTitle(assignee)"
diff --git a/app/assets/javascripts/boards/models/assignee.js b/app/assets/javascripts/boards/models/assignee.js
index c07a840ec3787..05dd449e4fd8d 100644
--- a/app/assets/javascripts/boards/models/assignee.js
+++ b/app/assets/javascripts/boards/models/assignee.js
@@ -1,20 +1,11 @@
-<<<<<<< HEAD:app/assets/javascripts/boards/models/assignee.js
 /* eslint-disable no-unused-vars */
 
 class ListAssignee {
-  constructor(user) {
-    this.id = user.id;
-    this.name = user.name;
-    this.username = user.username;
-    this.avatarUrl = user.avatar_url;
-=======
-class ListUser {
   constructor(user, defaultAvatar) {
     this.id = user.id;
     this.name = user.name;
     this.username = user.username;
     this.avatar = user.avatar_url || defaultAvatar;
->>>>>>> 10c1bf2d77fd0ab21309d0b136cbc0ac11f56c77:app/assets/javascripts/boards/models/user.js
   }
 }
 
diff --git a/app/assets/javascripts/boards/models/issue.js b/app/assets/javascripts/boards/models/issue.js
index 33c2fab99c725..6c2d8a3781b38 100644
--- a/app/assets/javascripts/boards/models/issue.js
+++ b/app/assets/javascripts/boards/models/issue.js
@@ -18,13 +18,6 @@ class ListIssue {
     this.selected = false;
     this.position = obj.relative_position || Infinity;
 
-<<<<<<< HEAD
-=======
-    if (obj.assignee) {
-      this.assignee = new ListUser(obj.assignee, defaultAvatar);
-    }
-
->>>>>>> 10c1bf2d77fd0ab21309d0b136cbc0ac11f56c77
     if (obj.milestone) {
       this.milestone = new ListMilestone(obj.milestone);
     }
@@ -33,7 +26,7 @@ class ListIssue {
       this.labels.push(new ListLabel(label));
     });
 
-    this.assignees = obj.assignees.map(a => new ListAssignee(a));
+    this.assignees = obj.assignees.map(a => new ListAssignee(a, defaultAvatar));
   }
 
   addLabel (label) {
diff --git a/app/assets/javascripts/sidebar/components/assignees/assignees.js b/app/assets/javascripts/sidebar/components/assignees/assignees.js
index 88d7650f40ac8..7e5feac622c7a 100644
--- a/app/assets/javascripts/sidebar/components/assignees/assignees.js
+++ b/app/assets/javascripts/sidebar/components/assignees/assignees.js
@@ -84,7 +84,7 @@ export default {
       return !this.showLess || (index < this.defaultRenderCount && this.showLess);
     },
     avatarUrl(user) {
-      return user.avatarUrl || user.avatar_url;
+      return user.avatar || user.avatar_url;
     },
     assigneeUrl(user) {
       return `${this.rootPath}${user.username}`;
diff --git a/config/webpack.config.js b/config/webpack.config.js
index a539dc511fa1a..bc2aa1ebf0cca 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -34,11 +34,6 @@ var config = {
     graphs:               './graphs/graphs_bundle.js',
     group:                './group.js',
     groups_list:          './groups_list.js',
-<<<<<<< HEAD
-=======
-    issuable:             './issuable/issuable_bundle.js',
-    locale:               './locale/index.js',
->>>>>>> 10c1bf2d77fd0ab21309d0b136cbc0ac11f56c77
     issue_show:           './issue_show/index.js',
     locale:               './locale/index.js',
     main:                 './main.js',
diff --git a/spec/javascripts/boards/issue_card_spec.js b/spec/javascripts/boards/issue_card_spec.js
index f872ca1040b30..fddde799d01ac 100644
--- a/spec/javascripts/boards/issue_card_spec.js
+++ b/spec/javascripts/boards/issue_card_spec.js
@@ -148,11 +148,11 @@ describe('Issue card component', () => {
 
     describe('assignee default avatar', () => {
       beforeEach((done) => {
-        component.issue.assignee = new ListUser({
+        component.issue.assignees = [new ListAssignee({
           id: 1,
           name: 'testing 123',
           username: 'test',
-        }, 'default_avatar');
+        }, 'default_avatar')];
 
         Vue.nextTick(done);
       });
diff --git a/spec/support/time_tracking_shared_examples.rb b/spec/support/time_tracking_shared_examples.rb
index c8e7af19f892e..84ef46ffa27bf 100644
--- a/spec/support/time_tracking_shared_examples.rb
+++ b/spec/support/time_tracking_shared_examples.rb
@@ -37,12 +37,7 @@
     submit_time('/estimate 3w 1d 1h')
     submit_time('/remove_estimate')
 
-<<<<<<< HEAD
     page.within '.time-tracking-component-wrap' do
-=======
-    wait_for_ajax
-    page.within '#issuable-time-tracker' do
->>>>>>> 10c1bf2d77fd0ab21309d0b136cbc0ac11f56c77
       expect(page).to have_content 'No estimate or time spent'
     end
   end
@@ -51,12 +46,7 @@
     submit_time('/spend 3w 1d 1h')
     submit_time('/remove_time_spent')
 
-<<<<<<< HEAD
     page.within '.time-tracking-component-wrap' do
-=======
-    wait_for_ajax
-    page.within '#issuable-time-tracker' do
->>>>>>> 10c1bf2d77fd0ab21309d0b136cbc0ac11f56c77
       expect(page).to have_content 'No estimate or time spent'
     end
   end
-- 
GitLab


From 4a681bb19f672e2266f0863db73706977727523a Mon Sep 17 00:00:00 2001
From: Sean McGivern <sean@gitlab.com>
Date: Fri, 5 May 2017 15:31:14 +0100
Subject: [PATCH 292/363] Remove FE team label

---
 CONTRIBUTING.md | 9 ++++++---
 1 file changed, 6 insertions(+), 3 deletions(-)

diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index 600dad563a6a3..8b6c87ae51870 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -21,7 +21,7 @@ _This notice should stay as the first item in the CONTRIBUTING.MD file._
 - [Workflow labels](#workflow-labels)
   - [Type labels (~"feature proposal", ~bug, ~customer, etc.)](#type-labels-feature-proposal-bug-customer-etc)
   - [Subject labels (~wiki, ~"container registry", ~ldap, ~api, etc.)](#subject-labels-wiki-container-registry-ldap-api-etc)
-  - [Team labels (~CI, ~Discussion, ~Edge, ~Frontend, ~Platform, etc.)](#team-labels-ci-discussion-edge-frontend-platform-etc)
+  - [Team labels (~CI, ~Discussion, ~Edge, ~Platform, etc.)](#team-labels-ci-discussion-edge-platform-etc)
   - [Priority labels (~Deliverable and ~Stretch)](#priority-labels-deliverable-and-stretch)
   - [Label for community contributors (~"Accepting Merge Requests")](#label-for-community-contributors-accepting-merge-requests)
 - [Implement design & UI elements](#implement-design--ui-elements)
@@ -155,18 +155,21 @@ Examples of subject labels are ~wiki, ~"container registry", ~ldap, ~api,
 
 Subject labels are always all-lowercase.
 
-### Team labels (~CI, ~Discussion, ~Edge, ~Frontend, ~Platform, etc.)
+### Team labels (~CI, ~Discussion, ~Edge, ~Platform, etc.)
 
 Team labels specify what team is responsible for this issue.
 Assigning a team label makes sure issues get the attention of the appropriate
 people.
 
 The current team labels are ~Build, ~CI, ~Discussion, ~Documentation, ~Edge,
-~Frontend, ~Gitaly, ~Platform, ~Prometheus, ~Release, and ~"UX".
+~Gitaly, ~Platform, ~Prometheus, ~Release, and ~"UX".
 
 The descriptions on the [labels page][labels-page] explain what falls under the
 responsibility of each team.
 
+Within those team labels, we also have the ~backend and ~frontend labels to
+indicate if an issue needs backend work, frontend work, or both.
+
 Team labels are always capitalized so that they show up as the first label for
 any issue.
 
-- 
GitLab


From acd9cd0906046f3d11d95aa9dc9d1af0498ef2ef Mon Sep 17 00:00:00 2001
From: James Edwards-Jones <jedwardsjones@gitlab.com>
Date: Thu, 4 May 2017 17:50:09 +0100
Subject: [PATCH 293/363] =?UTF-8?q?Hides=20pipeline=20=E2=80=98Failed=20Jo?=
 =?UTF-8?q?bs=E2=80=99=20tab=20when=20no=20jobs=20have=20failed?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

---
 .../projects/pipelines_controller.rb          |  6 +++-
 .../projects/pipelines/_with_tabs.html.haml   | 31 ++++++++++---------
 .../projects/pipelines/pipeline_spec.rb       | 15 +++++++++
 3 files changed, 37 insertions(+), 15 deletions(-)

diff --git a/app/controllers/projects/pipelines_controller.rb b/app/controllers/projects/pipelines_controller.rb
index 915f0bc63f768..3a56be2735adf 100644
--- a/app/controllers/projects/pipelines_controller.rb
+++ b/app/controllers/projects/pipelines_controller.rb
@@ -71,7 +71,11 @@ def builds
   end
 
   def failures
-    render_show
+    if @pipeline.statuses.latest.failed.present?
+      render_show
+    else
+      redirect_to pipeline_path(@pipeline)
+    end
   end
 
   def status
diff --git a/app/views/projects/pipelines/_with_tabs.html.haml b/app/views/projects/pipelines/_with_tabs.html.haml
index 76eb8533cc3b6..ba2c71cfd887d 100644
--- a/app/views/projects/pipelines/_with_tabs.html.haml
+++ b/app/views/projects/pipelines/_with_tabs.html.haml
@@ -1,3 +1,5 @@
+- failed_builds = @pipeline.statuses.latest.failed
+
 .tabs-holder
   %ul.pipelines-tabs.nav-links.no-top.no-bottom
     %li.js-pipeline-tab-link
@@ -7,10 +9,11 @@
       = link_to builds_namespace_project_pipeline_path(@project.namespace, @project, @pipeline), data: {target: 'div#js-tab-builds', action: 'builds', toggle: 'tab' }, class: 'builds-tab' do
         Jobs
         %span.badge.js-builds-counter= pipeline.statuses.count
-    %li.js-failures-tab-link
-      = link_to failures_namespace_project_pipeline_path(@project.namespace, @project, @pipeline), data: {target: 'div#js-tab-failures', action: 'failures', toggle: 'tab' }, class: 'failures-tab' do
-        Failures
-        %span.badge.js-failures-counter= pipeline.statuses.latest.failed.count
+    - if failed_builds.present?
+      %li.js-failures-tab-link
+        = link_to failures_namespace_project_pipeline_path(@project.namespace, @project, @pipeline), data: {target: 'div#js-tab-failures', action: 'failures', toggle: 'tab' }, class: 'failures-tab' do
+          Failed Jobs
+          %span.badge.js-failures-counter= failed_builds.count
 
 .tab-content
   #js-tab-pipeline.tab-pane
@@ -41,13 +44,13 @@
             %th Coverage
             %th
         = render partial: "projects/stage/stage", collection: pipeline.stages, as: :stage
-  #js-tab-failures.build-failures.tab-pane
-    - failed = @pipeline.statuses.latest.failed
-    - failed.each do |build|
-      .build-state
-        %span.ci-status-icon-failed= custom_icon('icon_status_failed')
-        %span.stage
-          = build.stage.titleize
-        %span.build-name
-          = render "notify/links/#{build.to_partial_path}", pipeline: @pipeline, build: build
-      %pre.build-log= build_summary(build)
+  - if failed_builds.present?
+    #js-tab-failures.build-failures.tab-pane
+      - failed_builds.each do |build|
+        .build-state
+          %span.ci-status-icon-failed= custom_icon('icon_status_failed')
+          %span.stage
+            = build.stage.titleize
+          %span.build-name
+            = render "notify/links/#{build.to_partial_path}", pipeline: @pipeline, build: build
+        %pre.build-log= build_summary(build)
diff --git a/spec/features/projects/pipelines/pipeline_spec.rb b/spec/features/projects/pipelines/pipeline_spec.rb
index ab7c002704d0e..cfac54ef259ad 100644
--- a/spec/features/projects/pipelines/pipeline_spec.rb
+++ b/spec/features/projects/pipelines/pipeline_spec.rb
@@ -269,6 +269,7 @@
       end
 
       it 'shows jobs tab pane as active' do
+        expect(page).to have_content('Failed Jobs')
         expect(page).to have_css('#js-tab-failures.active')
       end
 
@@ -291,5 +292,19 @@
         expect(page).to have_content('No job trace')
       end
     end
+
+    context 'without failures' do
+      before do
+        failed_build.update!(status: :success)
+
+        visit pipeline_failures_page
+      end
+
+      it 'displays the pipeline graph' do
+        expect(current_path).to eq(pipeline_path(pipeline))
+        expect(page).not_to have_content('Failed Jobs')
+        expect(page).to have_selector('.pipeline-visualization')
+      end
+    end
   end
 end
-- 
GitLab


From 9f3f22c8959b430811102fb790895e7edd61d3f9 Mon Sep 17 00:00:00 2001
From: James Edwards-Jones <jedwardsjones@gitlab.com>
Date: Thu, 4 May 2017 19:30:13 +0100
Subject: [PATCH 294/363] Failed Jobs tab only shows 10 job traces for
 performance

---
 app/helpers/builds_helper.rb                      | 8 ++++++--
 app/views/projects/pipelines/_with_tabs.html.haml | 6 +++---
 2 files changed, 9 insertions(+), 5 deletions(-)

diff --git a/app/helpers/builds_helper.rb b/app/helpers/builds_helper.rb
index 0145029fb6005..2eb2c6c738945 100644
--- a/app/helpers/builds_helper.rb
+++ b/app/helpers/builds_helper.rb
@@ -1,7 +1,11 @@
 module BuildsHelper
-  def build_summary(build)
+  def build_summary(build, skip: false)
     if build.has_trace?
-      build.trace.html(last_lines: 10).html_safe
+      if skip
+        link_to "View job trace", pipeline_build_url(build.pipeline, build)
+      else
+        build.trace.html(last_lines: 10).html_safe
+      end
     else
       "No job trace"
     end
diff --git a/app/views/projects/pipelines/_with_tabs.html.haml b/app/views/projects/pipelines/_with_tabs.html.haml
index ba2c71cfd887d..1aa48bf98131b 100644
--- a/app/views/projects/pipelines/_with_tabs.html.haml
+++ b/app/views/projects/pipelines/_with_tabs.html.haml
@@ -46,11 +46,11 @@
         = render partial: "projects/stage/stage", collection: pipeline.stages, as: :stage
   - if failed_builds.present?
     #js-tab-failures.build-failures.tab-pane
-      - failed_builds.each do |build|
+      - failed_builds.each_with_index do |build, index|
         .build-state
           %span.ci-status-icon-failed= custom_icon('icon_status_failed')
           %span.stage
             = build.stage.titleize
           %span.build-name
-            = render "notify/links/#{build.to_partial_path}", pipeline: @pipeline, build: build
-        %pre.build-log= build_summary(build)
+            = link_to build.name, pipeline_build_url(pipeline, build)
+        %pre.build-log= build_summary(build, skip: index >= 10)
-- 
GitLab


From 505e883fc09f344a326437a0f457b1be6106a2b5 Mon Sep 17 00:00:00 2001
From: Phil Hughes <me@iamphill.com>
Date: Fri, 5 May 2017 15:42:01 +0100
Subject: [PATCH 295/363] Fixed labels layout on labels milestone tab

---
 app/assets/stylesheets/pages/labels.scss          |  2 +-
 app/views/shared/milestones/_labels_tab.html.haml | 14 ++++++--------
 2 files changed, 7 insertions(+), 9 deletions(-)

diff --git a/app/assets/stylesheets/pages/labels.scss b/app/assets/stylesheets/pages/labels.scss
index e1ef0b029a59a..c10588ac58e6a 100644
--- a/app/assets/stylesheets/pages/labels.scss
+++ b/app/assets/stylesheets/pages/labels.scss
@@ -116,7 +116,7 @@
 }
 
 .manage-labels-list {
-  > li:not(.empty-message) {
+  > li:not(.empty-message):not(.is-not-draggable) {
     background-color: $white-light;
     cursor: move;
     cursor: -webkit-grab;
diff --git a/app/views/shared/milestones/_labels_tab.html.haml b/app/views/shared/milestones/_labels_tab.html.haml
index 33f93dccd3c48..a26b3b8009e96 100644
--- a/app/views/shared/milestones/_labels_tab.html.haml
+++ b/app/views/shared/milestones/_labels_tab.html.haml
@@ -2,7 +2,7 @@
   - labels.each do |label|
     - options = { milestone_title: @milestone.title, label_name: label.title }
 
-    %li
+    %li.is-not-draggable
       %span.label-row
         %span.label-name
           = link_to milestones_label_path(options) do
@@ -10,10 +10,8 @@
         %span.prepend-description-left
           = markdown_field(label, :description)
 
-      .pull-info-right
-        %span.append-right-20
-          = link_to milestones_label_path(options.merge(state: 'opened')) do
-            - pluralize milestone_issues_by_label_count(@milestone, label, state: :opened), 'open issue'
-        %span.append-right-20
-          = link_to milestones_label_path(options.merge(state: 'closed')) do
-            - pluralize milestone_issues_by_label_count(@milestone, label, state: :closed), 'closed issue'
+      .pull-right.hidden-xs.hidden-sm.hidden-md
+        = link_to milestones_label_path(options.merge(state: 'opened')), class: 'btn btn-transparent btn-action' do
+          - pluralize milestone_issues_by_label_count(@milestone, label, state: :opened), 'open issue'
+        = link_to milestones_label_path(options.merge(state: 'closed')), class: 'btn btn-transparent btn-action' do
+          - pluralize milestone_issues_by_label_count(@milestone, label, state: :closed), 'closed issue'
-- 
GitLab


From cc826c0469202930be835861eeabe0c43390db47 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 16:45:46 +0200
Subject: [PATCH 296/363] workaround spec failure for mySQL invalid date issue

---
 lib/gitlab/import_export/import_export.yml | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/lib/gitlab/import_export/import_export.yml b/lib/gitlab/import_export/import_export.yml
index 3aac731e84448..5f757f99fb301 100644
--- a/lib/gitlab/import_export/import_export.yml
+++ b/lib/gitlab/import_export/import_export.yml
@@ -85,6 +85,8 @@ excluded_attributes:
     - :id
     - :star_count
     - :last_activity_at
+    - :last_repository_updated_at
+    - :last_repository_check_at
   snippets:
     - :expired_at
   merge_request_diff:
-- 
GitLab


From 513b96f4eedd864b5f56e63fd5342deba53cc870 Mon Sep 17 00:00:00 2001
From: Sam Rose <samrose3@gmail.com>
Date: Fri, 5 May 2017 14:47:02 +0000
Subject: [PATCH 297/363] Wrap note actions below timestamp on discussions

---
 app/assets/stylesheets/pages/notes.scss | 16 ++++++++++------
 1 file changed, 10 insertions(+), 6 deletions(-)

diff --git a/app/assets/stylesheets/pages/notes.scss b/app/assets/stylesheets/pages/notes.scss
index cfea52c6e5742..69c328d09ffab 100644
--- a/app/assets/stylesheets/pages/notes.scss
+++ b/app/assets/stylesheets/pages/notes.scss
@@ -130,12 +130,6 @@ ul.notes {
     }
 
     .note-header {
-      padding-bottom: 8px;
-      padding-right: 20px;
-
-      @media (min-width: $screen-sm-min) {
-        padding-right: 0;
-      }
 
       @media (max-width: $screen-xs-min) {
         .inline {
@@ -384,10 +378,15 @@ ul.notes {
 .note-header {
   display: flex;
   justify-content: space-between;
+
+  @media (max-width: $screen-xs-max) {
+    flex-flow: row wrap;
+  }
 }
 
 .note-header-info {
   min-width: 0;
+  padding-bottom: 5px;
 }
 
 .note-headline-light {
@@ -435,6 +434,11 @@ ul.notes {
   margin-left: 10px;
   color: $gray-darkest;
 
+  @media (max-width: $screen-xs-max) {
+    float: none;
+    margin-left: 0;
+  }
+
   .note-action-button {
     margin-left: 8px;
   }
-- 
GitLab


From e5a7ed3ac36aaa1045353e589dae98a29ca72f1e Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 5 May 2017 15:55:55 +0100
Subject: [PATCH 298/363] Review changes

---
 app/assets/javascripts/raven/index.js       | 2 +-
 lib/gitlab/gon_helper.rb                    | 1 -
 spec/javascripts/raven/index_spec.js        | 3 ++-
 spec/javascripts/raven/raven_config_spec.js | 4 ----
 4 files changed, 3 insertions(+), 7 deletions(-)

diff --git a/app/assets/javascripts/raven/index.js b/app/assets/javascripts/raven/index.js
index 3c5656040b9e1..5325e495815f7 100644
--- a/app/assets/javascripts/raven/index.js
+++ b/app/assets/javascripts/raven/index.js
@@ -5,7 +5,7 @@ const index = function index() {
     sentryDsn: gon.sentry_dsn,
     currentUserId: gon.current_user_id,
     whitelistUrls: [gon.gitlab_url],
-    isProduction: gon.is_production,
+    isProduction: process.env.NODE_ENV,
   });
 
   return RavenConfig;
diff --git a/lib/gitlab/gon_helper.rb b/lib/gitlab/gon_helper.rb
index 848b3352c63b7..26473f99bc351 100644
--- a/lib/gitlab/gon_helper.rb
+++ b/lib/gitlab/gon_helper.rb
@@ -12,7 +12,6 @@ def add_gon_variables
       gon.katex_js_url           = ActionController::Base.helpers.asset_path('katex.js')
       gon.sentry_dsn             = current_application_settings.clientside_sentry_dsn if current_application_settings.clientside_sentry_enabled
       gon.gitlab_url             = Gitlab.config.gitlab.url
-      gon.is_production          = Rails.env.production?
 
       if current_user
         gon.current_user_id = current_user.id
diff --git a/spec/javascripts/raven/index_spec.js b/spec/javascripts/raven/index_spec.js
index 85ec1de4e4efa..b5662cd0331c7 100644
--- a/spec/javascripts/raven/index_spec.js
+++ b/spec/javascripts/raven/index_spec.js
@@ -18,9 +18,10 @@ describe('RavenConfig options', () => {
       sentry_dsn: sentryDsn,
       current_user_id: currentUserId,
       gitlab_url: gitlabUrl,
-      is_production: isProduction,
     };
 
+    process.env.NODE_ENV = isProduction;
+
     spyOn(RavenConfig, 'init');
 
     indexReturnValue = index();
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
index 2a2b91fe9480c..a2d720760fce6 100644
--- a/spec/javascripts/raven/raven_config_spec.js
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -157,10 +157,6 @@ describe('RavenConfig', () => {
       RavenConfig.bindRavenErrors();
     });
 
-    it('should query for document using jquery', () => {
-      expect(window.$).toHaveBeenCalledWith(document);
-    });
-
     it('should call .on', function () {
       expect($document.on).toHaveBeenCalledWith('ajaxError.raven', RavenConfig.handleRavenErrors);
     });
-- 
GitLab


From b09460db7c5428ceaafb6d263bb134b4cbfb46a1 Mon Sep 17 00:00:00 2001
From: Lin Jen-Shin <godfat@godfat.org>
Date: Fri, 5 May 2017 15:13:17 +0000
Subject: [PATCH 299/363] Update CHANGELOG.md for 9.1.3

[ci skip]
---
 CHANGELOG.md | 1 +
 1 file changed, 1 insertion(+)

diff --git a/CHANGELOG.md b/CHANGELOG.md
index c9de0113e2452..e625278a7963a 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -4,6 +4,7 @@ entry.
 
 ## 9.1.3 (2017-05-05)
 
+- Do not show private groups on subgroups page if user doesn't have access to.
 - Enforce project features when searching blobs and wikis.
 - Fixed branches dropdown rendering branch names as HTML.
 - Make Asciidoc & other markup go through pipeline to prevent XSS.
-- 
GitLab


From 9c0f2485b5d7fed5b538313e5bf867110273c004 Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Fri, 5 May 2017 18:22:54 +0300
Subject: [PATCH 300/363] Multiple issue assignee: resolve conflicts after
 merging upstream

---
 .../slash_commands/interpret_service.rb       | 32 ++++++++-----------
 .../slash_commands/interpret_service_spec.rb  |  2 +-
 2 files changed, 14 insertions(+), 20 deletions(-)

diff --git a/app/services/slash_commands/interpret_service.rb b/app/services/slash_commands/interpret_service.rb
index 78a6e072f26c5..a7e13648b540f 100644
--- a/app/services/slash_commands/interpret_service.rb
+++ b/app/services/slash_commands/interpret_service.rb
@@ -91,41 +91,35 @@ def extractor
     end
 
     desc 'Assign'
-    explanation do |user|
-      "Assigns #{user.to_reference}." if user
+    explanation do |users|
+      "Assigns #{users.map(&:to_reference).to_sentence}." if users.any?
     end
     params '@user'
     condition do
       current_user.can?(:"admin_#{issuable.to_ability_name}", project)
     end
-<<<<<<< HEAD
-    command :assign do |assignee_param|
-      user_ids = extract_references(assignee_param, :user).map(&:id)
+    parse_params do |assignee_param|
+      users = extract_references(assignee_param, :user)
 
-      if user_ids.empty?
-        user_ids = User.where(username: assignee_param.split(' ').map(&:strip)).pluck(:id)
+      if users.empty?
+        users = User.where(username: assignee_param.split(' ').map(&:strip))
       end
 
-      next if user_ids.empty?
+      users
+    end
+    command :assign do |users|
+      next if users.empty?
 
       if issuable.is_a?(Issue)
-        @updates[:assignee_ids] = user_ids
+        @updates[:assignee_ids] = users.map(&:id)
       else
-        @updates[:assignee_id] = user_ids.last
+        @updates[:assignee_id] = users.last.id
       end
-=======
-    parse_params do |assignee_param|
-      extract_references(assignee_param, :user).first ||
-        User.find_by(username: assignee_param)
-    end
-    command :assign do |user|
-      @updates[:assignee_id] = user.id if user
->>>>>>> 10c1bf2d77fd0ab21309d0b136cbc0ac11f56c77
     end
 
     desc 'Remove assignee'
     explanation do
-      "Removes assignee #{issuable.assignee.to_reference}."
+      "Removes assignee #{issuable.assignees.first.to_reference}."
     end
     condition do
       issuable.persisted? &&
diff --git a/spec/services/slash_commands/interpret_service_spec.rb b/spec/services/slash_commands/interpret_service_spec.rb
index 24f2c1407aa54..e5e400ee281f6 100644
--- a/spec/services/slash_commands/interpret_service_spec.rb
+++ b/spec/services/slash_commands/interpret_service_spec.rb
@@ -874,7 +874,7 @@
 
     describe 'unassign command' do
       let(:content) { '/unassign' }
-      let(:issue) { create(:issue, project: project, assignee: developer) }
+      let(:issue) { create(:issue, project: project, assignees: [developer]) }
 
       it 'includes current assignee reference' do
         _, explanations = service.explain(content, issue)
-- 
GitLab


From 58b560fae0f75f5f5de0960d055791c471d58af7 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Fri, 5 May 2017 09:33:39 -0600
Subject: [PATCH 301/363] resolve discussion

---
 app/assets/javascripts/issue_show/index.js     |  4 ++--
 .../issue_show/issue_title_description.vue     | 18 +++++++++---------
 app/views/projects/issues/_issue.html.haml     |  4 ----
 app/views/projects/issues/show.html.haml       |  2 +-
 .../issue_show/issue_title_description_spec.js |  2 --
 5 files changed, 12 insertions(+), 18 deletions(-)

diff --git a/app/assets/javascripts/issue_show/index.js b/app/assets/javascripts/issue_show/index.js
index 2de072c9778d5..eb20a597bb5a5 100644
--- a/app/assets/javascripts/issue_show/index.js
+++ b/app/assets/javascripts/issue_show/index.js
@@ -4,13 +4,13 @@ import '../vue_shared/vue_resource_interceptor';
 
 (() => {
   const issueTitleData = document.querySelector('.issue-title-data').dataset;
-  const { canupdateissue, endpoint } = issueTitleData;
+  const { canUpdateTasksClass, endpoint } = issueTitleData;
 
   const vm = new Vue({
     el: '.issue-title-entrypoint',
     render: createElement => createElement(IssueTitle, {
       props: {
-        canUpdateIssue: canupdateissue,
+        canUpdateTasksClass,
         endpoint,
       },
     }),
diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index e88ab69455b54..f3437e2ef8b0e 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -10,7 +10,7 @@ export default {
       required: true,
       type: String,
     },
-    canUpdateIssue: {
+    canUpdateTasksClass: {
       required: true,
       type: String,
     },
@@ -112,19 +112,16 @@ export default {
     },
   },
   computed: {
-    descriptionClass() {
-      return `description ${this.canUpdateIssue} is-task-list-enabled`;
-    },
     titleAnimationCss() {
       return {
-        'title issue-realtime-pre-pulse': this.titleFlag.pre,
-        'title issue-realtime-trigger-pulse': this.titleFlag.pulse,
+        'issue-realtime-pre-pulse': this.titleFlag.pre,
+        'issue-realtime-trigger-pulse': this.titleFlag.pulse,
       };
     },
     descriptionAnimationCss() {
       return {
-        'wiki issue-realtime-pre-pulse': this.descriptionFlag.pre,
-        'wiki issue-realtime-trigger-pulse': this.descriptionFlag.pulse,
+        'issue-realtime-pre-pulse': this.descriptionFlag.pre,
+        'issue-realtime-trigger-pulse': this.descriptionFlag.pulse,
       };
     },
   },
@@ -165,16 +162,19 @@ export default {
 <template>
   <div>
     <h2
+      class="title"
       :class="titleAnimationCss"
       ref="issue-title"
       v-html="title"
     >
     </h2>
     <div
-      :class="descriptionClass"
+      class="description is-task-list-enabled"
+      :class="canUpdateTasksClass"
       v-if="description"
     >
       <div
+        class="wiki"
         :class="descriptionAnimationCss"
         v-html="description"
         ref="issue-content-container-gfm-entry"
diff --git a/app/views/projects/issues/_issue.html.haml b/app/views/projects/issues/_issue.html.haml
index 064a7bd1f2606..0e3902c066ad2 100644
--- a/app/views/projects/issues/_issue.html.haml
+++ b/app/views/projects/issues/_issue.html.haml
@@ -16,10 +16,6 @@
           - if issue.assignee
             %li
               = link_to_member(@project, issue.assignee, name: false, title: "Assigned to :name")
-          - if issue.tasks?
-            &nbsp;
-            %span.task-status
-              = issue.task_status
 
           = render 'shared/issuable_meta_data', issuable: issue
 
diff --git a/app/views/projects/issues/show.html.haml b/app/views/projects/issues/show.html.haml
index aa024be1c55d3..a7aefa08aa01b 100644
--- a/app/views/projects/issues/show.html.haml
+++ b/app/views/projects/issues/show.html.haml
@@ -52,7 +52,7 @@
 .issue-details.issuable-details
   .detail-page-description.content-block
     .issue-title-data.hidden{ "data" => { "endpoint" => rendered_title_namespace_project_issue_path(@project.namespace, @project, @issue),
-      "canUpdateIssue" => can?(current_user, :update_issue, @issue) ? 'js-task-list-container' : '',
+      "can-update-tasks-class" => can?(current_user, :update_issue, @issue) ? 'js-task-list-container' : '',
     } }
     .issue-title-entrypoint
 
diff --git a/spec/javascripts/issue_show/issue_title_description_spec.js b/spec/javascripts/issue_show/issue_title_description_spec.js
index 30455663e5097..1ec4fe58b08b9 100644
--- a/spec/javascripts/issue_show/issue_title_description_spec.js
+++ b/spec/javascripts/issue_show/issue_title_description_spec.js
@@ -39,7 +39,6 @@ describe('Issue Title', () => {
       },
     }).$mount();
 
-    // need setTimeout because of api call/v-html
     setTimeout(() => {
       expect(document.querySelector('title').innerText).toContain('this is a title (#1)');
       expect(issueShowComponent.$el.querySelector('.title').innerHTML).toContain('<p>this is a title</p>');
@@ -57,6 +56,5 @@ describe('Issue Title', () => {
         done();
       });
     });
-    // 10ms is just long enough for the update hook to fire
   });
 });
-- 
GitLab


From de59bab80989eba44c38a129a235900cf71ff0b5 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 5 May 2017 16:42:56 +0100
Subject: [PATCH 302/363] Another attempt at access_control_ce_spec

---
 spec/features/protected_tags/access_control_ce_spec.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/features/protected_tags/access_control_ce_spec.rb b/spec/features/protected_tags/access_control_ce_spec.rb
index a04fbcdd15fe1..12622cd548a55 100644
--- a/spec/features/protected_tags/access_control_ce_spec.rb
+++ b/spec/features/protected_tags/access_control_ce_spec.rb
@@ -9,7 +9,7 @@
         allowed_to_create_button = find(".js-allowed-to-create")
 
         unless allowed_to_create_button.text == access_type_name
-          allowed_to_create_button.click
+          allowed_to_create_button.trigger('click')
           find('.create_access_levels-container .dropdown-menu li', match: :first)
           within('.create_access_levels-container .dropdown-menu') { click_on access_type_name }
         end
-- 
GitLab


From 0cfe35f7279e7b6618c10be91908c29cd7ab110a Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Fri, 5 May 2017 18:55:16 +0300
Subject: [PATCH 303/363] Multiple issue assignee: CE restriction for multiple
 assignees

---
 app/services/issues/base_service.rb           |  3 +++
 spec/requests/api/issues_spec.rb              | 22 +++++++++++++++
 .../notes/slash_commands_service_spec.rb      | 27 +++++++++++++++++++
 3 files changed, 52 insertions(+)

diff --git a/app/services/issues/base_service.rb b/app/services/issues/base_service.rb
index eedbfa724ff66..bcd196bfa21b7 100644
--- a/app/services/issues/base_service.rb
+++ b/app/services/issues/base_service.rb
@@ -24,6 +24,9 @@ def execute_hooks(issue, action = 'open')
     def filter_assignee(issuable)
       return if params[:assignee_ids].blank?
 
+      # The number of assignees is limited by one for GitLab CE
+      params[:assignee_ids].slice!(0, 1)
+
       assignee_ids = params[:assignee_ids].select { |assignee_id| assignee_can_read?(issuable, assignee_id) }
 
       if params[:assignee_ids].map(&:to_s) == [IssuableFinder::NONE]
diff --git a/spec/requests/api/issues_spec.rb b/spec/requests/api/issues_spec.rb
index 9b7b5798b71e2..da2b56c040bc7 100644
--- a/spec/requests/api/issues_spec.rb
+++ b/spec/requests/api/issues_spec.rb
@@ -772,6 +772,17 @@
       end
     end
 
+    context 'CE restrictions' do
+      it 'creates a new project issue with no more than one assignee' do
+        post api("/projects/#{project.id}/issues", user),
+          title: 'new issue', assignee_ids: [user2.id, guest.id]
+
+        expect(response).to have_http_status(201)
+        expect(json_response['title']).to eq('new issue')
+        expect(json_response['assignees'].count).to eq(1)
+      end
+    end
+
     it 'creates a new project issue' do
       post api("/projects/#{project.id}/issues", user),
         title: 'new issue', labels: 'label, label2', weight: 3,
@@ -1111,6 +1122,17 @@
 
       expect(json_response['assignees'].first['name']).to eq(user2.name)
     end
+
+    context 'CE restrictions' do
+      it 'updates an issue with several assignee but only one has been applied' do
+        put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+          assignee_ids: [user2.id, guest.id]
+
+        expect(response).to have_http_status(200)
+
+        expect(json_response['assignees'].size).to eq(1)
+      end
+    end
   end
 
   describe 'PUT /projects/:id/issues/:issue_iid to update labels' do
diff --git a/spec/services/notes/slash_commands_service_spec.rb b/spec/services/notes/slash_commands_service_spec.rb
index 0edcc50ed7b22..b99f01162ee34 100644
--- a/spec/services/notes/slash_commands_service_spec.rb
+++ b/spec/services/notes/slash_commands_service_spec.rb
@@ -220,4 +220,31 @@
       let(:note) { build(:note_on_commit, project: project) }
     end
   end
+
+  context 'CE restriction for issue assignees' do
+    describe '/assign' do
+      let(:project) { create(:empty_project) }
+      let(:master) { create(:user).tap { |u| project.team << [u, :master] } }
+      let(:assignee) { create(:user) }
+      let(:master) { create(:user) }
+      let(:service) { described_class.new(project, master) }
+      let(:note) { create(:note_on_issue, note: note_text, project: project) }
+
+      let(:note_text) do
+        %(/assign @#{assignee.username} @#{master.username}\n")
+      end
+
+      before do
+        project.team << [master, :master]
+        project.team << [assignee, :master]
+      end
+
+      it 'adds only one assignee from the list' do
+        content, command_params = service.extract_commands(note)
+        service.execute(command_params, note)
+
+        expect(note.noteable.assignees.count).to eq(1)
+      end
+    end
+  end
 end
-- 
GitLab


From 6ecf16b8f70335e1e6868b7c918cd031f5eb2f8d Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 18:01:33 +0200
Subject: [PATCH 304/363] refactor code based on feedback

---
 app/controllers/admin/services_controller.rb  |  2 +-
 ...rvice.rb => propagate_service_template.rb} | 21 +++++++++++++------
 ...b => propagate_service_template_worker.rb} |  6 +++---
 config/sidekiq_queues.yml                     |  2 +-
 lib/gitlab/sql/bulk_insert.rb                 | 21 -------------------
 .../admin/services_controller_spec.rb         |  8 +++----
 ....rb => propagate_service_template_spec.rb} | 17 +++++++--------
 ...propagate_service_template_worker_spec.rb} |  4 ++--
 8 files changed, 34 insertions(+), 47 deletions(-)
 rename app/services/projects/{propagate_service.rb => propagate_service_template.rb} (77%)
 rename app/workers/{propagate_project_service_worker.rb => propagate_service_template_worker.rb} (63%)
 delete mode 100644 lib/gitlab/sql/bulk_insert.rb
 rename spec/services/projects/{propagate_service_spec.rb => propagate_service_template_spec.rb} (83%)
 rename spec/workers/{propagate_project_service_worker_spec.rb => propagate_service_template_worker_spec.rb} (79%)

diff --git a/app/controllers/admin/services_controller.rb b/app/controllers/admin/services_controller.rb
index e335fbfffedfd..a40ce3c24182b 100644
--- a/app/controllers/admin/services_controller.rb
+++ b/app/controllers/admin/services_controller.rb
@@ -16,7 +16,7 @@ def edit
 
   def update
     if service.update_attributes(service_params[:service])
-      PropagateProjectServiceWorker.perform_async(service.id) if  service.active?
+      PropagateServiceTemplateWorker.perform_async(service.id) if  service.active?
 
       redirect_to admin_application_settings_services_path,
         notice: 'Application settings saved successfully'
diff --git a/app/services/projects/propagate_service.rb b/app/services/projects/propagate_service_template.rb
similarity index 77%
rename from app/services/projects/propagate_service.rb
rename to app/services/projects/propagate_service_template.rb
index f4fae478609f1..32ad68673acd9 100644
--- a/app/services/projects/propagate_service.rb
+++ b/app/services/projects/propagate_service_template.rb
@@ -1,5 +1,5 @@
 module Projects
-  class PropagateService
+  class PropagateServiceTemplate
     BATCH_SIZE = 100
 
     def self.propagate(*args)
@@ -36,9 +36,7 @@ def bulk_create_from_template(batch)
       end
 
       Project.transaction do
-        Gitlab::SQL::BulkInsert.new(service_hash.keys + ['project_id'],
-                                    service_list,
-                                    'services').execute
+        bulk_insert_services(service_hash.keys + ['project_id'], service_list)
         run_callbacks(batch)
       end
     end
@@ -54,11 +52,22 @@ def project_ids_batch
             WHERE services.project_id = projects.id
             AND services.type = '#{@template.type}'
           )
+          AND projects.pending_delete = false
+          AND projects.archived = false
           LIMIT #{BATCH_SIZE}
       SQL
       )
     end
 
+    def bulk_insert_services(columns, values_array)
+      ActiveRecord::Base.connection.execute(
+        <<-SQL.strip_heredoc
+          INSERT INTO services (#{columns.join(', ')})
+          VALUES #{values_array.map { |tuple| "(#{tuple.join(', ')})" }.join(', ')}
+      SQL
+      )
+    end
+
     def service_hash
       @service_hash ||=
         begin
@@ -84,11 +93,11 @@ def run_callbacks(batch)
     end
 
     def active_external_issue_tracker?
-      @template['category'] == 'issue_tracker' && @template['active'] && !@template['default']
+      @template.category == :issue_tracker && !@template.default
     end
 
     def active_external_wiki?
-      @template['type'] == 'ExternalWikiService' && @template['active']
+      @template.type == 'ExternalWikiService'
     end
   end
 end
diff --git a/app/workers/propagate_project_service_worker.rb b/app/workers/propagate_service_template_worker.rb
similarity index 63%
rename from app/workers/propagate_project_service_worker.rb
rename to app/workers/propagate_service_template_worker.rb
index 5eabe4eaecdbc..f1fc7ccb955bf 100644
--- a/app/workers/propagate_project_service_worker.rb
+++ b/app/workers/propagate_service_template_worker.rb
@@ -1,5 +1,5 @@
 # Worker for updating any project specific caches.
-class PropagateProjectServiceWorker
+class PropagateServiceTemplateWorker
   include Sidekiq::Worker
   include DedicatedSidekiqQueue
 
@@ -10,14 +10,14 @@ class PropagateProjectServiceWorker
   def perform(template_id)
     return unless try_obtain_lease_for(template_id)
 
-    Projects::PropagateService.propagate(Service.find_by(id: template_id))
+    Projects::PropagateServiceTemplate.propagate(Service.find_by(id: template_id))
   end
 
   private
 
   def try_obtain_lease_for(template_id)
     Gitlab::ExclusiveLease.
-      new("propagate_project_service_worker:#{template_id}", timeout: LEASE_TIMEOUT).
+      new("propagate_service_template_worker:#{template_id}", timeout: LEASE_TIMEOUT).
       try_obtain
   end
 end
diff --git a/config/sidekiq_queues.yml b/config/sidekiq_queues.yml
index 91ea1c0f779ba..433381e79d3d5 100644
--- a/config/sidekiq_queues.yml
+++ b/config/sidekiq_queues.yml
@@ -53,4 +53,4 @@
   - [pages, 1]
   - [system_hook_push, 1]
   - [update_user_activity, 1]
-  - [propagate_project_service, 1]
+  - [propagate_service_template, 1]
diff --git a/lib/gitlab/sql/bulk_insert.rb b/lib/gitlab/sql/bulk_insert.rb
deleted file mode 100644
index 097f9ff237b1e..0000000000000
--- a/lib/gitlab/sql/bulk_insert.rb
+++ /dev/null
@@ -1,21 +0,0 @@
-module Gitlab
-  module SQL
-    # Class for building SQL bulk inserts
-    class BulkInsert
-      def initialize(columns, values_array, table)
-        @columns = columns
-        @values_array = values_array
-        @table = table
-      end
-
-      def execute
-        ActiveRecord::Base.connection.execute(
-          <<-SQL.strip_heredoc
-          INSERT INTO #{@table} (#{@columns.join(', ')})
-          VALUES #{@values_array.map { |tuple| "(#{tuple.join(', ')})" }.join(', ')}
-        SQL
-        )
-      end
-    end
-  end
-end
diff --git a/spec/controllers/admin/services_controller_spec.rb b/spec/controllers/admin/services_controller_spec.rb
index 808c98edb7f13..c94616d8508f4 100644
--- a/spec/controllers/admin/services_controller_spec.rb
+++ b/spec/controllers/admin/services_controller_spec.rb
@@ -39,16 +39,16 @@
       )
     end
 
-    it 'updates the service params successfully and calls the propagation worker' do
-      expect(PropagateProjectServiceWorker).to receive(:perform_async).with(service.id)
+    it 'calls the propagation worker when service is active' do
+      expect(PropagateServiceTemplateWorker).to receive(:perform_async).with(service.id)
 
       put :update, id: service.id, service: { active: true }
 
       expect(response).to have_http_status(302)
     end
 
-    it 'updates the service params successfully' do
-      expect(PropagateProjectServiceWorker).not_to receive(:perform_async)
+    it 'does not call the propagation worker when service is not active' do
+      expect(PropagateServiceTemplateWorker).not_to receive(:perform_async)
 
       put :update, id: service.id, service: { properties: {} }
 
diff --git a/spec/services/projects/propagate_service_spec.rb b/spec/services/projects/propagate_service_template_spec.rb
similarity index 83%
rename from spec/services/projects/propagate_service_spec.rb
rename to spec/services/projects/propagate_service_template_spec.rb
index b8aa4de5bd17a..331fb3c5ac53e 100644
--- a/spec/services/projects/propagate_service_spec.rb
+++ b/spec/services/projects/propagate_service_template_spec.rb
@@ -1,7 +1,7 @@
 require 'spec_helper'
 
-describe Projects::PropagateService, services: true do
-  describe '.propagate!' do
+describe Projects::PropagateServiceTemplate, services: true do
+  describe '.propagate' do
     let!(:service_template) do
       PushoverService.create(
         template: true,
@@ -23,9 +23,10 @@
     end
 
     it 'creates services for a project that has another service' do
-      other_service = BambooService.create(
+      BambooService.create(
         template: true,
         active: true,
+        project: project,
         properties: {
           bamboo_url: 'http://gitlab.com',
           username: 'mic',
@@ -34,8 +35,6 @@
         }
       )
 
-      Service.build_from_template(project.id, other_service).save!
-
       expect { described_class.propagate(service_template) }.
         to change { Service.count }.by(1)
     end
@@ -62,7 +61,7 @@
     it 'creates the service containing the template attributes' do
       described_class.propagate(service_template)
 
-      service = Service.find_by(type: service_template.type, template: false)
+      service = Service.find_by!(type: service_template.type, template: false)
 
       expect(service.properties).to eq(service_template.properties)
     end
@@ -70,7 +69,7 @@
     describe 'bulk update' do
       it 'creates services for all projects' do
         project_total = 5
-        stub_const 'Projects::PropagateService::BATCH_SIZE', 3
+        stub_const 'Projects::PropagateServiceTemplate::BATCH_SIZE', 3
 
         project_total.times { create(:empty_project) }
 
@@ -81,7 +80,7 @@
 
     describe 'external tracker' do
       it 'updates the project external tracker' do
-        service_template.update(category: 'issue_tracker', default: false)
+        service_template.update!(category: 'issue_tracker', default: false)
 
         expect { described_class.propagate(service_template) }.
           to change { project.reload.has_external_issue_tracker }.to(true)
@@ -90,7 +89,7 @@
 
     describe 'external wiki' do
       it 'updates the project external tracker' do
-        service_template.update(type: 'ExternalWikiService')
+        service_template.update!(type: 'ExternalWikiService')
 
         expect { described_class.propagate(service_template) }.
           to change { project.reload.has_external_wiki }.to(true)
diff --git a/spec/workers/propagate_project_service_worker_spec.rb b/spec/workers/propagate_service_template_worker_spec.rb
similarity index 79%
rename from spec/workers/propagate_project_service_worker_spec.rb
rename to spec/workers/propagate_service_template_worker_spec.rb
index 4c7edbfcd3e4a..7040d5ef81c97 100644
--- a/spec/workers/propagate_project_service_worker_spec.rb
+++ b/spec/workers/propagate_service_template_worker_spec.rb
@@ -1,6 +1,6 @@
 require 'spec_helper'
 
-describe PropagateProjectServiceWorker do
+describe PropagateServiceTemplateWorker do
   let!(:service_template) do
     PushoverService.create(
       template: true,
@@ -21,7 +21,7 @@
 
   describe '#perform' do
     it 'calls the propagate service with the template' do
-      expect(Projects::PropagateService).to receive(:propagate).with(service_template)
+      expect(Projects::PropagateServiceTemplate).to receive(:propagate).with(service_template)
 
       subject.perform(service_template.id)
     end
-- 
GitLab


From 7389bb98167551a850be52e6683ecc93c4734a05 Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Fri, 5 May 2017 11:02:40 -0500
Subject: [PATCH 305/363] Revert participants style change

---
 app/assets/stylesheets/pages/issuable.scss | 10 ++++++----
 1 file changed, 6 insertions(+), 4 deletions(-)

diff --git a/app/assets/stylesheets/pages/issuable.scss b/app/assets/stylesheets/pages/issuable.scss
index 485ea369f3d17..c4210ffd8230c 100644
--- a/app/assets/stylesheets/pages/issuable.scss
+++ b/app/assets/stylesheets/pages/issuable.scss
@@ -444,21 +444,23 @@
 }
 
 .participants-list {
-  display: flex;
-  flex-wrap: wrap;
-  justify-content: space-between;
   margin: -5px;
 }
 
+
 .user-list {
   display: flex;
   flex-wrap: wrap;
 }
 
 .participants-author {
-  flex-basis: 14%;
+  display: inline-block;
   padding: 5px;
 
+  &:nth-of-type(7n) {
+    padding-right: 0;
+  }
+
   .author_link {
     display: block;
   }
-- 
GitLab


From f15466bd5bd2ce5390e392785d7c750c176acbec Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 18:18:39 +0200
Subject: [PATCH 306/363] refactor code based on feedback

---
 app/controllers/admin/services_controller.rb        | 2 +-
 app/services/projects/propagate_service_template.rb | 2 +-
 2 files changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/controllers/admin/services_controller.rb b/app/controllers/admin/services_controller.rb
index a40ce3c24182b..4c3d336b3aff8 100644
--- a/app/controllers/admin/services_controller.rb
+++ b/app/controllers/admin/services_controller.rb
@@ -16,7 +16,7 @@ def edit
 
   def update
     if service.update_attributes(service_params[:service])
-      PropagateServiceTemplateWorker.perform_async(service.id) if  service.active?
+      PropagateServiceTemplateWorker.perform_async(service.id) if service.active?
 
       redirect_to admin_application_settings_services_path,
         notice: 'Application settings saved successfully'
diff --git a/app/services/projects/propagate_service_template.rb b/app/services/projects/propagate_service_template.rb
index 32ad68673acd9..2999e1af38505 100644
--- a/app/services/projects/propagate_service_template.rb
+++ b/app/services/projects/propagate_service_template.rb
@@ -11,7 +11,7 @@ def initialize(template)
     end
 
     def propagate
-      return unless @template&.active
+      return unless @template&.active?
 
       Rails.logger.info("Propagating services for template #{@template.id}")
 
-- 
GitLab


From 62f7b206b936bbd58d3f18021df37920a5c34ddd Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Fri, 5 May 2017 10:27:31 -0600
Subject: [PATCH 307/363] object in css not computed

---
 .../issue_show/issue_title_description.vue    | 45 ++++++++-----------
 1 file changed, 18 insertions(+), 27 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index f3437e2ef8b0e..9dc02bbee7f68 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -29,22 +29,23 @@ export default {
       },
     });
 
-    const defaultFlags = {
-      pre: true,
-      pulse: false,
-    };
-
     return {
       poll,
       apiData: {},
       tasks: '0 of 0',
       title: null,
       titleText: '',
-      titleFlag: defaultFlags,
+      titleFlag: {
+        pre: true,
+        pulse: false,
+      },
       description: null,
       descriptionText: '',
       descriptionChange: false,
-      descriptionFlag: defaultFlags,
+      descriptionFlag: {
+        pre: true,
+        pulse: false,
+      },
       timeAgoEl: $('.issue_edited_ago'),
       titleEl: document.querySelector('title'),
     };
@@ -60,7 +61,8 @@ export default {
     elementsToVisualize(noTitleChange, noDescriptionChange) {
       if (!noTitleChange) {
         this.titleText = this.apiData.title_text;
-        this.titleFlag = { pre: true, pulse: false };
+        this.titleFlag.pre = true;
+        this.titleFlag.pulse = false;
       }
 
       if (!noDescriptionChange) {
@@ -68,7 +70,8 @@ export default {
         this.descriptionChange = true;
         this.updateTaskHTML();
         this.tasks = this.apiData.task_status;
-        this.descriptionFlag = { pre: true, pulse: false };
+        this.descriptionFlag.pre = true;
+        this.descriptionFlag.pulse = false;
       }
     },
     setTabTitle() {
@@ -82,8 +85,10 @@ export default {
       this.setTabTitle();
 
       this.$nextTick(() => {
-        this.titleFlag = { pre: false, pulse: true };
-        this.descriptionFlag = { pre: false, pulse: true };
+        this.titleFlag.pre = false;
+        this.titleFlag.pulse = true;
+        this.descriptionFlag.pre = false;
+        this.descriptionFlag.pulse = true;
       });
     },
     triggerAnimation() {
@@ -111,20 +116,6 @@ export default {
       this.timeAgoEl.attr('title', toolTipTime).tooltip('fixTitle');
     },
   },
-  computed: {
-    titleAnimationCss() {
-      return {
-        'issue-realtime-pre-pulse': this.titleFlag.pre,
-        'issue-realtime-trigger-pulse': this.titleFlag.pulse,
-      };
-    },
-    descriptionAnimationCss() {
-      return {
-        'issue-realtime-pre-pulse': this.descriptionFlag.pre,
-        'issue-realtime-trigger-pulse': this.descriptionFlag.pulse,
-      };
-    },
-  },
   created() {
     if (!Visibility.hidden()) {
       this.poll.makeRequest();
@@ -163,7 +154,7 @@ export default {
   <div>
     <h2
       class="title"
-      :class="titleAnimationCss"
+      :class="{ 'issue-realtime-pre-pulse': titleFlag.pre, 'issue-realtime-trigger-pulse': titleFlag.pulse }"
       ref="issue-title"
       v-html="title"
     >
@@ -175,7 +166,7 @@ export default {
     >
       <div
         class="wiki"
-        :class="descriptionAnimationCss"
+        :class="{ 'issue-realtime-pre-pulse': descriptionFlag.pre, 'issue-realtime-trigger-pulse': descriptionFlag.pulse }"
         v-html="description"
         ref="issue-content-container-gfm-entry"
       >
-- 
GitLab


From 166e3cc5b012cc564707ab9c8a69c6b357dd5567 Mon Sep 17 00:00:00 2001
From: tauriedavis <taurie@gitlab.com>
Date: Thu, 4 May 2017 15:58:36 -0700
Subject: [PATCH 308/363] 30903 Vertically align mini pipeline stage container

---
 app/assets/stylesheets/pages/pipelines.scss                   | 1 +
 .../unreleased/30903-vertically-align-mini-pipeline.yml       | 4 ++++
 2 files changed, 5 insertions(+)
 create mode 100644 changelogs/unreleased/30903-vertically-align-mini-pipeline.yml

diff --git a/app/assets/stylesheets/pages/pipelines.scss b/app/assets/stylesheets/pages/pipelines.scss
index 9115d26c77960..f514fdafc7b16 100644
--- a/app/assets/stylesheets/pages/pipelines.scss
+++ b/app/assets/stylesheets/pages/pipelines.scss
@@ -273,6 +273,7 @@
   .stage-container {
     display: inline-block;
     position: relative;
+    vertical-align: middle;
     height: 22px;
     margin: 3px 6px 3px 0;
 
diff --git a/changelogs/unreleased/30903-vertically-align-mini-pipeline.yml b/changelogs/unreleased/30903-vertically-align-mini-pipeline.yml
new file mode 100644
index 0000000000000..af87e5ce39f17
--- /dev/null
+++ b/changelogs/unreleased/30903-vertically-align-mini-pipeline.yml
@@ -0,0 +1,4 @@
+---
+title: Vertically align mini pipeline stage container
+merge_request:
+author:
-- 
GitLab


From 17730a6e1fde0643719c03218e8e582a52e7ad94 Mon Sep 17 00:00:00 2001
From: Annabel Dunstone Gray <annabel.dunstone@gmail.com>
Date: Fri, 5 May 2017 08:49:19 -0500
Subject: [PATCH 309/363] Change text of inline issue creation tooltip

---
 app/views/projects/boards/components/_board.html.haml | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/views/projects/boards/components/_board.html.haml b/app/views/projects/boards/components/_board.html.haml
index 5a4eaf92b1613..bc5c727bf0df2 100644
--- a/app/views/projects/boards/components/_board.html.haml
+++ b/app/views/projects/boards/components/_board.html.haml
@@ -13,8 +13,8 @@
             %button.btn.btn-small.btn-default.pull-right.has-tooltip{ type: "button",
               "@click" => "showNewIssueForm",
               "v-if" => 'list.type !== "closed"',
-              "aria-label" => "Add an issue",
-              "title" => "Add an issue",
+              "aria-label" => "New issue",
+              "title" => "New issue",
               data: { placement: "top", container: "body" } }
               = icon("plus")
         - if can?(current_user, :admin_list, @project)
-- 
GitLab


From d32e0b31e01b653c3dc2f5fe307119fd4fbff6f4 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 5 May 2017 17:51:18 +0100
Subject: [PATCH 310/363] Trigger click instead of actual click to make sure
 the right button is always hit

---
 spec/features/issues/issue_sidebar_spec.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/features/issues/issue_sidebar_spec.rb b/spec/features/issues/issue_sidebar_spec.rb
index 82b80a69bed5f..e980e78f9950a 100644
--- a/spec/features/issues/issue_sidebar_spec.rb
+++ b/spec/features/issues/issue_sidebar_spec.rb
@@ -152,7 +152,7 @@ def visit_issue(project, issue)
   end
 
   def open_issue_sidebar
-    find('aside.right-sidebar.right-sidebar-collapsed .js-sidebar-toggle').click
+    find('aside.right-sidebar.right-sidebar-collapsed .js-sidebar-toggle').trigger('click')
     find('aside.right-sidebar.right-sidebar-expanded')
   end
 end
-- 
GitLab


From 856a511b4804a0b78294a29bbba86ac111d960f8 Mon Sep 17 00:00:00 2001
From: James Lopez <james@jameslopez.es>
Date: Fri, 5 May 2017 18:57:52 +0200
Subject: [PATCH 311/363] refactor code based on feedback

---
 .../projects/propagate_service_template.rb     | 12 ++++++------
 .../propagate_service_template_worker.rb       |  2 --
 .../propagate_service_template_spec.rb         | 18 +++++++++++-------
 3 files changed, 17 insertions(+), 15 deletions(-)

diff --git a/app/services/projects/propagate_service_template.rb b/app/services/projects/propagate_service_template.rb
index 2999e1af38505..a8ef2108492e7 100644
--- a/app/services/projects/propagate_service_template.rb
+++ b/app/services/projects/propagate_service_template.rb
@@ -11,7 +11,7 @@ def initialize(template)
     end
 
     def propagate
-      return unless @template&.active?
+      return unless @template.active?
 
       Rails.logger.info("Propagating services for template #{@template.id}")
 
@@ -32,11 +32,11 @@ def propagate_projects_with_template
 
     def bulk_create_from_template(batch)
       service_list = batch.map do |project_id|
-        service_hash.merge('project_id' => project_id).values
+        service_hash.values << project_id
       end
 
       Project.transaction do
-        bulk_insert_services(service_hash.keys + ['project_id'], service_list)
+        bulk_insert_services(service_hash.keys << 'project_id', service_list)
         run_callbacks(batch)
       end
     end
@@ -75,9 +75,9 @@ def service_hash
 
           template_hash.each_with_object({}) do |(key, value), service_hash|
             value = value.is_a?(Hash) ? value.to_json : value
-            key = Gitlab::Database.postgresql? ? "\"#{key}\"" : "`#{key}`"
 
-            service_hash[key] = ActiveRecord::Base.sanitize(value)
+            service_hash[ActiveRecord::Base.connection.quote_column_name(key)] =
+              ActiveRecord::Base.sanitize(value)
           end
         end
     end
@@ -93,7 +93,7 @@ def run_callbacks(batch)
     end
 
     def active_external_issue_tracker?
-      @template.category == :issue_tracker && !@template.default
+      @template.issue_tracker? && !@template.default
     end
 
     def active_external_wiki?
diff --git a/app/workers/propagate_service_template_worker.rb b/app/workers/propagate_service_template_worker.rb
index f1fc7ccb955bf..5ce0e0405d0ac 100644
--- a/app/workers/propagate_service_template_worker.rb
+++ b/app/workers/propagate_service_template_worker.rb
@@ -3,8 +3,6 @@ class PropagateServiceTemplateWorker
   include Sidekiq::Worker
   include DedicatedSidekiqQueue
 
-  sidekiq_options retry: 3
-
   LEASE_TIMEOUT = 4.hours.to_i
 
   def perform(template_id)
diff --git a/spec/services/projects/propagate_service_template_spec.rb b/spec/services/projects/propagate_service_template_spec.rb
index 331fb3c5ac53e..90eff3bbc1e0b 100644
--- a/spec/services/projects/propagate_service_template_spec.rb
+++ b/spec/services/projects/propagate_service_template_spec.rb
@@ -18,8 +18,11 @@
     let!(:project) { create(:empty_project) }
 
     it 'creates services for projects' do
-      expect { described_class.propagate(service_template) }.
-        to change { Service.count }.by(1)
+      expect(project.pushover_service).to be_nil
+
+      described_class.propagate(service_template)
+
+      expect(project.reload.pushover_service).to be_present
     end
 
     it 'creates services for a project that has another service' do
@@ -35,8 +38,11 @@
         }
       )
 
-      expect { described_class.propagate(service_template) }.
-        to change { Service.count }.by(1)
+      expect(project.pushover_service).to be_nil
+
+      described_class.propagate(service_template)
+
+      expect(project.reload.pushover_service).to be_present
     end
 
     it 'does not create the service if it exists already' do
@@ -61,9 +67,7 @@
     it 'creates the service containing the template attributes' do
       described_class.propagate(service_template)
 
-      service = Service.find_by!(type: service_template.type, template: false)
-
-      expect(service.properties).to eq(service_template.properties)
+      expect(project.pushover_service.properties).to eq(service_template.properties)
     end
 
     describe 'bulk update' do
-- 
GitLab


From 541c8da0103d008471b8c6389451e6370a3992f3 Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Fri, 5 May 2017 11:07:19 -0600
Subject: [PATCH 312/363] make toggle switch for flags

---
 .../issue_show/issue_title_description.vue       | 16 ++++++++--------
 1 file changed, 8 insertions(+), 8 deletions(-)

diff --git a/app/assets/javascripts/issue_show/issue_title_description.vue b/app/assets/javascripts/issue_show/issue_title_description.vue
index 9dc02bbee7f68..dc3ba2550c5a5 100644
--- a/app/assets/javascripts/issue_show/issue_title_description.vue
+++ b/app/assets/javascripts/issue_show/issue_title_description.vue
@@ -51,6 +51,10 @@ export default {
     };
   },
   methods: {
+    updateFlag(key, toggle) {
+      this[key].pre = toggle;
+      this[key].pulse = !toggle;
+    },
     renderResponse(res) {
       this.apiData = res.json();
       this.triggerAnimation();
@@ -61,8 +65,7 @@ export default {
     elementsToVisualize(noTitleChange, noDescriptionChange) {
       if (!noTitleChange) {
         this.titleText = this.apiData.title_text;
-        this.titleFlag.pre = true;
-        this.titleFlag.pulse = false;
+        this.updateFlag('titleFlag', true);
       }
 
       if (!noDescriptionChange) {
@@ -70,8 +73,7 @@ export default {
         this.descriptionChange = true;
         this.updateTaskHTML();
         this.tasks = this.apiData.task_status;
-        this.descriptionFlag.pre = true;
-        this.descriptionFlag.pulse = false;
+        this.updateFlag('descriptionFlag', true);
       }
     },
     setTabTitle() {
@@ -85,10 +87,8 @@ export default {
       this.setTabTitle();
 
       this.$nextTick(() => {
-        this.titleFlag.pre = false;
-        this.titleFlag.pulse = true;
-        this.descriptionFlag.pre = false;
-        this.descriptionFlag.pulse = true;
+        this.updateFlag('titleFlag', false);
+        this.updateFlag('descriptionFlag', false);
       });
     },
     triggerAnimation() {
-- 
GitLab


From 0b7aabe302195348e06cea68937457c0c905de6c Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Fri, 5 May 2017 20:06:22 +0300
Subject: [PATCH 313/363] Multiple issue assignee: fix for CE restrictions

---
 app/services/issues/base_service.rb         | 2 +-
 spec/services/issues/update_service_spec.rb | 4 ++--
 2 files changed, 3 insertions(+), 3 deletions(-)

diff --git a/app/services/issues/base_service.rb b/app/services/issues/base_service.rb
index bcd196bfa21b7..34199eb5d1373 100644
--- a/app/services/issues/base_service.rb
+++ b/app/services/issues/base_service.rb
@@ -25,7 +25,7 @@ def filter_assignee(issuable)
       return if params[:assignee_ids].blank?
 
       # The number of assignees is limited by one for GitLab CE
-      params[:assignee_ids].slice!(0, 1)
+      params[:assignee_ids] = params[:assignee_ids][0, 1]
 
       assignee_ids = params[:assignee_ids].select { |assignee_id| assignee_can_read?(issuable, assignee_id) }
 
diff --git a/spec/services/issues/update_service_spec.rb b/spec/services/issues/update_service_spec.rb
index 1797a23ee8afd..6633ac10236e4 100644
--- a/spec/services/issues/update_service_spec.rb
+++ b/spec/services/issues/update_service_spec.rb
@@ -40,7 +40,7 @@ def update_issue(opts)
         {
           title: 'New title',
           description: 'Also please fix',
-          assignee_ids: [user2.id, user3.id],
+          assignee_ids: [user2.id],
           state_event: 'close',
           label_ids: [label.id],
           due_date: Date.tomorrow
@@ -53,7 +53,7 @@ def update_issue(opts)
         expect(issue).to be_valid
         expect(issue.title).to eq 'New title'
         expect(issue.description).to eq 'Also please fix'
-        expect(issue.assignees).to match_array([user2, user3])
+        expect(issue.assignees).to match_array([user2])
         expect(issue).to be_closed
         expect(issue.labels).to match_array [label]
         expect(issue.due_date).to eq Date.tomorrow
-- 
GitLab


From fc121cca5ba87abd24afbc8da2f76e14e386e4c8 Mon Sep 17 00:00:00 2001
From: Grzegorz Bizon <grzesiek.bizon@gmail.com>
Date: Fri, 5 May 2017 19:35:32 +0200
Subject: [PATCH 314/363] Do not reprocess actions when user retries pipeline

User who is not allowed to trigger manual actions should not be
allowed to reprocess / retrigger / retry these actions.
---
 app/services/ci/retry_pipeline_service.rb     |  2 +
 .../ci/retry_pipeline_service_spec.rb         | 44 ++++++++++++++++++-
 2 files changed, 45 insertions(+), 1 deletion(-)

diff --git a/app/services/ci/retry_pipeline_service.rb b/app/services/ci/retry_pipeline_service.rb
index ecc6173a96a55..5b2071573453e 100644
--- a/app/services/ci/retry_pipeline_service.rb
+++ b/app/services/ci/retry_pipeline_service.rb
@@ -8,6 +8,8 @@ def execute(pipeline)
       end
 
       pipeline.retryable_builds.find_each do |build|
+        next unless can?(current_user, :update_build, build)
+
         Ci::RetryBuildService.new(project, current_user)
           .reprocess(build)
       end
diff --git a/spec/services/ci/retry_pipeline_service_spec.rb b/spec/services/ci/retry_pipeline_service_spec.rb
index f1b2d3a47985a..40e151545c995 100644
--- a/spec/services/ci/retry_pipeline_service_spec.rb
+++ b/spec/services/ci/retry_pipeline_service_spec.rb
@@ -7,7 +7,9 @@
   let(:service) { described_class.new(project, user) }
 
   context 'when user has ability to modify pipeline' do
-    let(:user) { create(:admin) }
+    before do
+      project.add_master(user)
+    end
 
     context 'when there are already retried jobs present' do
       before do
@@ -227,6 +229,46 @@
     end
   end
 
+  context 'when user is not allowed to trigger manual action' do
+    before do
+      project.add_developer(user)
+    end
+
+    context 'when there is a failed manual action present' do
+      before do
+        create_build('test', :failed, 0)
+        create_build('deploy', :failed, 0, when: :manual)
+        create_build('verify', :canceled, 1)
+      end
+
+      it 'does not reprocess manual action' do
+        service.execute(pipeline)
+
+        expect(build('test')).to be_pending
+        expect(build('deploy')).to be_failed
+        expect(build('verify')).to be_created
+        expect(pipeline.reload).to be_running
+      end
+    end
+
+    context 'when there is a failed manual action in later stage' do
+      before do
+        create_build('test', :failed, 0)
+        create_build('deploy', :failed, 1, when: :manual)
+        create_build('verify', :canceled, 2)
+      end
+
+      it 'does not reprocess manual action' do
+        service.execute(pipeline)
+
+        expect(build('test')).to be_pending
+        expect(build('deploy')).to be_failed
+        expect(build('verify')).to be_created
+        expect(pipeline.reload).to be_running
+      end
+    end
+  end
+
   def statuses
     pipeline.reload.statuses
   end
-- 
GitLab


From 8a5eaee6e51fa176f53bb1fdb1ecfe3f92e781ba Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Fri, 5 May 2017 20:39:03 +0300
Subject: [PATCH 315/363] Remove wrong changelog

---
 changelogs/unreleased/update-issue-board-cards-design.yml | 4 ----
 1 file changed, 4 deletions(-)
 delete mode 100644 changelogs/unreleased/update-issue-board-cards-design.yml

diff --git a/changelogs/unreleased/update-issue-board-cards-design.yml b/changelogs/unreleased/update-issue-board-cards-design.yml
deleted file mode 100644
index 5ef94a74e8a92..0000000000000
--- a/changelogs/unreleased/update-issue-board-cards-design.yml
+++ /dev/null
@@ -1,4 +0,0 @@
----
-title: Update issue board cards design
-merge_request: 10353
-author:
-- 
GitLab


From 933447e0da19f9d0be8574185500cabb5d7ab012 Mon Sep 17 00:00:00 2001
From: Valery Sizov <valery@gitlab.com>
Date: Fri, 5 May 2017 20:44:19 +0300
Subject: [PATCH 316/363] Address static analyzer warning

---
 spec/services/notes/slash_commands_service_spec.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/spec/services/notes/slash_commands_service_spec.rb b/spec/services/notes/slash_commands_service_spec.rb
index b99f01162ee34..c9954dc360351 100644
--- a/spec/services/notes/slash_commands_service_spec.rb
+++ b/spec/services/notes/slash_commands_service_spec.rb
@@ -240,7 +240,7 @@
       end
 
       it 'adds only one assignee from the list' do
-        content, command_params = service.extract_commands(note)
+        _, command_params = service.extract_commands(note)
         service.execute(command_params, note)
 
         expect(note.noteable.assignees.count).to eq(1)
-- 
GitLab


From bef42d9a36bf465878e0edc19c9e154f73f12b13 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Fri, 5 May 2017 17:59:41 +0000
Subject: [PATCH 317/363] Fallback localstorage cases

---
 app/assets/javascripts/autosave.js            |  44 +++---
 .../behaviors/gl_emoji/unicode_support_map.js |  17 ++-
 .../recent_searches_dropdown_content.js       |  12 +-
 .../filtered_search_manager.js                |  11 +-
 .../filtered_search_visual_tokens.js          |   3 +
 .../filtered_search/recent_searches_root.js   |   7 +-
 .../services/recent_searches_service.js       |  14 ++
 .../services/recent_searches_service_error.js |  11 ++
 app/assets/javascripts/lib/utils/accessor.js  |  47 ++++++
 .../javascripts/signin_tabs_memoizer.js       |  12 +-
 spec/javascripts/autosave_spec.js             | 134 ++++++++++++++++++
 .../gl_emoji/unicode_support_map_spec.js      |  47 ++++++
 .../recent_searches_dropdown_content_spec.js  |  20 +++
 .../filtered_search_manager_spec.js           |  34 +++++
 .../recent_searches_root_spec.js              |  31 ++++
 .../recent_searches_service_error_spec.js     |  18 +++
 .../services/recent_searches_service_spec.js  |  95 ++++++++++++-
 spec/javascripts/lib/utils/accessor_spec.js   |  78 ++++++++++
 spec/javascripts/signin_tabs_memoizer_spec.js |  90 ++++++++++++
 19 files changed, 684 insertions(+), 41 deletions(-)
 create mode 100644 app/assets/javascripts/filtered_search/services/recent_searches_service_error.js
 create mode 100644 app/assets/javascripts/lib/utils/accessor.js
 create mode 100644 spec/javascripts/autosave_spec.js
 create mode 100644 spec/javascripts/behaviors/gl_emoji/unicode_support_map_spec.js
 create mode 100644 spec/javascripts/filtered_search/recent_searches_root_spec.js
 create mode 100644 spec/javascripts/filtered_search/services/recent_searches_service_error_spec.js
 create mode 100644 spec/javascripts/lib/utils/accessor_spec.js

diff --git a/app/assets/javascripts/autosave.js b/app/assets/javascripts/autosave.js
index 8630b18a73f96..cfab6c40b34f7 100644
--- a/app/assets/javascripts/autosave.js
+++ b/app/assets/javascripts/autosave.js
@@ -1,8 +1,11 @@
 /* eslint-disable func-names, space-before-function-paren, wrap-iife, no-param-reassign, quotes, prefer-template, no-var, one-var, no-unused-vars, one-var-declaration-per-line, no-void, consistent-return, no-empty, max-len */
+import AccessorUtilities from './lib/utils/accessor';
 
 window.Autosave = (function() {
   function Autosave(field, key) {
     this.field = field;
+    this.isLocalStorageAvailable = AccessorUtilities.isLocalStorageAccessSafe();
+
     if (key.join != null) {
       key = key.join("/");
     }
@@ -17,16 +20,12 @@ window.Autosave = (function() {
   }
 
   Autosave.prototype.restore = function() {
-    var e, text;
-    if (window.localStorage == null) {
-      return;
-    }
-    try {
-      text = window.localStorage.getItem(this.key);
-    } catch (error) {
-      e = error;
-      return;
-    }
+    var text;
+
+    if (!this.isLocalStorageAvailable) return;
+
+    text = window.localStorage.getItem(this.key);
+
     if ((text != null ? text.length : void 0) > 0) {
       this.field.val(text);
     }
@@ -35,27 +34,22 @@ window.Autosave = (function() {
 
   Autosave.prototype.save = function() {
     var text;
-    if (window.localStorage == null) {
-      return;
-    }
     text = this.field.val();
-    if ((text != null ? text.length : void 0) > 0) {
-      try {
-        return window.localStorage.setItem(this.key, text);
-      } catch (error) {}
-    } else {
-      return this.reset();
+
+    if (this.isLocalStorageAvailable && (text != null ? text.length : void 0) > 0) {
+      return window.localStorage.setItem(this.key, text);
     }
+
+    return this.reset();
   };
 
   Autosave.prototype.reset = function() {
-    if (window.localStorage == null) {
-      return;
-    }
-    try {
-      return window.localStorage.removeItem(this.key);
-    } catch (error) {}
+    if (!this.isLocalStorageAvailable) return;
+
+    return window.localStorage.removeItem(this.key);
   };
 
   return Autosave;
 })();
+
+export default window.Autosave;
diff --git a/app/assets/javascripts/behaviors/gl_emoji/unicode_support_map.js b/app/assets/javascripts/behaviors/gl_emoji/unicode_support_map.js
index aa522e20c3603..257df55e54fb4 100644
--- a/app/assets/javascripts/behaviors/gl_emoji/unicode_support_map.js
+++ b/app/assets/javascripts/behaviors/gl_emoji/unicode_support_map.js
@@ -1,3 +1,5 @@
+import AccessorUtilities from '../../lib/utils/accessor';
+
 const unicodeSupportTestMap = {
   // man, student (emojione does not have any of these yet), http://emojipedia.org/emoji-zwj-sequences/
   // occupationZwj: '\u{1F468}\u{200D}\u{1F393}',
@@ -140,16 +142,25 @@ function generateUnicodeSupportMap(testMap) {
 
 function getUnicodeSupportMap() {
   let unicodeSupportMap;
-  const userAgentFromCache = window.localStorage.getItem('gl-emoji-user-agent');
+  let userAgentFromCache;
+
+  const isLocalStorageAvailable = AccessorUtilities.isLocalStorageAccessSafe();
+
+  if (isLocalStorageAvailable) userAgentFromCache = window.localStorage.getItem('gl-emoji-user-agent');
+
   try {
     unicodeSupportMap = JSON.parse(window.localStorage.getItem('gl-emoji-unicode-support-map'));
   } catch (err) {
     // swallow
   }
+
   if (!unicodeSupportMap || userAgentFromCache !== navigator.userAgent) {
     unicodeSupportMap = generateUnicodeSupportMap(unicodeSupportTestMap);
-    window.localStorage.setItem('gl-emoji-user-agent', navigator.userAgent);
-    window.localStorage.setItem('gl-emoji-unicode-support-map', JSON.stringify(unicodeSupportMap));
+
+    if (isLocalStorageAvailable) {
+      window.localStorage.setItem('gl-emoji-user-agent', navigator.userAgent);
+      window.localStorage.setItem('gl-emoji-unicode-support-map', JSON.stringify(unicodeSupportMap));
+    }
   }
 
   return unicodeSupportMap;
diff --git a/app/assets/javascripts/filtered_search/components/recent_searches_dropdown_content.js b/app/assets/javascripts/filtered_search/components/recent_searches_dropdown_content.js
index 9126422b33547..15052dbd362f5 100644
--- a/app/assets/javascripts/filtered_search/components/recent_searches_dropdown_content.js
+++ b/app/assets/javascripts/filtered_search/components/recent_searches_dropdown_content.js
@@ -8,6 +8,11 @@ export default {
       type: Array,
       required: true,
     },
+    isLocalStorageAvailable: {
+      type: Boolean,
+      required: false,
+      default: true,
+    },
   },
 
   computed: {
@@ -47,7 +52,12 @@ export default {
 
   template: `
     <div>
-      <ul v-if="hasItems">
+      <div
+        v-if="!isLocalStorageAvailable"
+        class="dropdown-info-note">
+        This feature requires local storage to be enabled
+      </div>
+      <ul v-else-if="hasItems">
         <li
           v-for="(item, index) in processedItems"
           :key="index">
diff --git a/app/assets/javascripts/filtered_search/filtered_search_manager.js b/app/assets/javascripts/filtered_search/filtered_search_manager.js
index 36af0674ac668..9fea563370f77 100644
--- a/app/assets/javascripts/filtered_search/filtered_search_manager.js
+++ b/app/assets/javascripts/filtered_search/filtered_search_manager.js
@@ -1,5 +1,3 @@
-/* global Flash */
-
 import FilteredSearchContainer from './container';
 import RecentSearchesRoot from './recent_searches_root';
 import RecentSearchesStore from './stores/recent_searches_store';
@@ -15,7 +13,9 @@ class FilteredSearchManager {
     this.tokensContainer = this.container.querySelector('.tokens-container');
     this.filteredSearchTokenKeys = gl.FilteredSearchTokenKeys;
 
-    this.recentSearchesStore = new RecentSearchesStore();
+    this.recentSearchesStore = new RecentSearchesStore({
+      isLocalStorageAvailable: RecentSearchesService.isAvailable(),
+    });
     let recentSearchesKey = 'issue-recent-searches';
     if (page === 'merge_requests') {
       recentSearchesKey = 'merge-request-recent-searches';
@@ -24,9 +24,10 @@ class FilteredSearchManager {
 
     // Fetch recent searches from localStorage
     this.fetchingRecentSearchesPromise = this.recentSearchesService.fetch()
-      .catch(() => {
+      .catch((error) => {
+        if (error.name === 'RecentSearchesServiceError') return undefined;
         // eslint-disable-next-line no-new
-        new Flash('An error occured while parsing recent searches');
+        new window.Flash('An error occured while parsing recent searches');
         // Gracefully fail to empty array
         return [];
       })
diff --git a/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js b/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
index 453ecccc6fc8d..59ce0587e1e40 100644
--- a/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
+++ b/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
@@ -183,6 +183,9 @@ class FilteredSearchVisualTokens {
 
   static moveInputToTheRight() {
     const input = FilteredSearchContainer.container.querySelector('.filtered-search');
+
+    if (!input) return;
+
     const inputLi = input.parentElement;
     const tokenContainer = FilteredSearchContainer.container.querySelector('.tokens-container');
 
diff --git a/app/assets/javascripts/filtered_search/recent_searches_root.js b/app/assets/javascripts/filtered_search/recent_searches_root.js
index 4e38409e12a5a..b2e6f63aacf34 100644
--- a/app/assets/javascripts/filtered_search/recent_searches_root.js
+++ b/app/assets/javascripts/filtered_search/recent_searches_root.js
@@ -29,12 +29,15 @@ class RecentSearchesRoot {
   }
 
   render() {
+    const state = this.store.state;
     this.vm = new Vue({
       el: this.wrapperElement,
-      data: this.store.state,
+      data() { return state; },
       template: `
         <recent-searches-dropdown-content
-          :items="recentSearches" />
+          :items="recentSearches"
+          :is-local-storage-available="isLocalStorageAvailable"
+          />
       `,
       components: {
         'recent-searches-dropdown-content': RecentSearchesDropdownContent,
diff --git a/app/assets/javascripts/filtered_search/services/recent_searches_service.js b/app/assets/javascripts/filtered_search/services/recent_searches_service.js
index 3e402d5aed006..a056dea928dcc 100644
--- a/app/assets/javascripts/filtered_search/services/recent_searches_service.js
+++ b/app/assets/javascripts/filtered_search/services/recent_searches_service.js
@@ -1,9 +1,17 @@
+import RecentSearchesServiceError from './recent_searches_service_error';
+import AccessorUtilities from '../../lib/utils/accessor';
+
 class RecentSearchesService {
   constructor(localStorageKey = 'issuable-recent-searches') {
     this.localStorageKey = localStorageKey;
   }
 
   fetch() {
+    if (!RecentSearchesService.isAvailable()) {
+      const error = new RecentSearchesServiceError();
+      return Promise.reject(error);
+    }
+
     const input = window.localStorage.getItem(this.localStorageKey);
 
     let searches = [];
@@ -19,8 +27,14 @@ class RecentSearchesService {
   }
 
   save(searches = []) {
+    if (!RecentSearchesService.isAvailable()) return;
+
     window.localStorage.setItem(this.localStorageKey, JSON.stringify(searches));
   }
+
+  static isAvailable() {
+    return AccessorUtilities.isLocalStorageAccessSafe();
+  }
 }
 
 export default RecentSearchesService;
diff --git a/app/assets/javascripts/filtered_search/services/recent_searches_service_error.js b/app/assets/javascripts/filtered_search/services/recent_searches_service_error.js
new file mode 100644
index 0000000000000..5917b223d63fd
--- /dev/null
+++ b/app/assets/javascripts/filtered_search/services/recent_searches_service_error.js
@@ -0,0 +1,11 @@
+class RecentSearchesServiceError {
+  constructor(message) {
+    this.name = 'RecentSearchesServiceError';
+    this.message = message || 'Recent Searches Service is unavailable';
+  }
+}
+
+// Can't use `extends` for builtin prototypes and get true inheritance yet
+RecentSearchesServiceError.prototype = Error.prototype;
+
+export default RecentSearchesServiceError;
diff --git a/app/assets/javascripts/lib/utils/accessor.js b/app/assets/javascripts/lib/utils/accessor.js
new file mode 100644
index 0000000000000..1d18992af6325
--- /dev/null
+++ b/app/assets/javascripts/lib/utils/accessor.js
@@ -0,0 +1,47 @@
+function isPropertyAccessSafe(base, property) {
+  let safe;
+
+  try {
+    safe = !!base[property];
+  } catch (error) {
+    safe = false;
+  }
+
+  return safe;
+}
+
+function isFunctionCallSafe(base, functionName, ...args) {
+  let safe = true;
+
+  try {
+    base[functionName](...args);
+  } catch (error) {
+    safe = false;
+  }
+
+  return safe;
+}
+
+function isLocalStorageAccessSafe() {
+  let safe;
+
+  const TEST_KEY = 'isLocalStorageAccessSafe';
+  const TEST_VALUE = 'true';
+
+  safe = isPropertyAccessSafe(window, 'localStorage');
+  if (!safe) return safe;
+
+  safe = isFunctionCallSafe(window.localStorage, 'setItem', TEST_KEY, TEST_VALUE);
+
+  if (safe) window.localStorage.removeItem(TEST_KEY);
+
+  return safe;
+}
+
+const AccessorUtilities = {
+  isPropertyAccessSafe,
+  isFunctionCallSafe,
+  isLocalStorageAccessSafe,
+};
+
+export default AccessorUtilities;
diff --git a/app/assets/javascripts/signin_tabs_memoizer.js b/app/assets/javascripts/signin_tabs_memoizer.js
index d811d1cd53abf..2587facc58225 100644
--- a/app/assets/javascripts/signin_tabs_memoizer.js
+++ b/app/assets/javascripts/signin_tabs_memoizer.js
@@ -1,5 +1,7 @@
 /* eslint no-param-reassign: ["error", { "props": false }]*/
 /* eslint no-new: "off" */
+import AccessorUtilities from './lib/utils/accessor';
+
 ((global) => {
   /**
    * Memorize the last selected tab after reloading a page.
@@ -9,6 +11,8 @@
     constructor({ currentTabKey = 'current_signin_tab', tabSelector = 'ul.nav-tabs' } = {}) {
       this.currentTabKey = currentTabKey;
       this.tabSelector = tabSelector;
+      this.isLocalStorageAvailable = AccessorUtilities.isLocalStorageAccessSafe();
+
       this.bootstrap();
     }
 
@@ -37,11 +41,15 @@
     }
 
     saveData(val) {
-      localStorage.setItem(this.currentTabKey, val);
+      if (!this.isLocalStorageAvailable) return undefined;
+
+      return window.localStorage.setItem(this.currentTabKey, val);
     }
 
     readData() {
-      return localStorage.getItem(this.currentTabKey);
+      if (!this.isLocalStorageAvailable) return null;
+
+      return window.localStorage.getItem(this.currentTabKey);
     }
   }
 
diff --git a/spec/javascripts/autosave_spec.js b/spec/javascripts/autosave_spec.js
new file mode 100644
index 0000000000000..9f9acc392c286
--- /dev/null
+++ b/spec/javascripts/autosave_spec.js
@@ -0,0 +1,134 @@
+import Autosave from '~/autosave';
+import AccessorUtilities from '~/lib/utils/accessor';
+
+describe('Autosave', () => {
+  let autosave;
+
+  describe('class constructor', () => {
+    const key = 'key';
+    const field = jasmine.createSpyObj('field', ['data', 'on']);
+
+    beforeEach(() => {
+      spyOn(AccessorUtilities, 'isLocalStorageAccessSafe').and.returnValue(true);
+      spyOn(Autosave.prototype, 'restore');
+
+      autosave = new Autosave(field, key);
+    });
+
+    it('should set .isLocalStorageAvailable', () => {
+      expect(AccessorUtilities.isLocalStorageAccessSafe).toHaveBeenCalled();
+      expect(autosave.isLocalStorageAvailable).toBe(true);
+    });
+  });
+
+  describe('restore', () => {
+    const key = 'key';
+    const field = jasmine.createSpyObj('field', ['trigger']);
+
+    beforeEach(() => {
+      autosave = {
+        field,
+        key,
+      };
+
+      spyOn(window.localStorage, 'getItem');
+    });
+
+    describe('if .isLocalStorageAvailable is `false`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = false;
+
+        Autosave.prototype.restore.call(autosave);
+      });
+
+      it('should not call .getItem', () => {
+        expect(window.localStorage.getItem).not.toHaveBeenCalled();
+      });
+    });
+
+    describe('if .isLocalStorageAvailable is `true`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = true;
+
+        Autosave.prototype.restore.call(autosave);
+      });
+
+      it('should call .getItem', () => {
+        expect(window.localStorage.getItem).toHaveBeenCalledWith(key);
+      });
+    });
+  });
+
+  describe('save', () => {
+    const field = jasmine.createSpyObj('field', ['val']);
+
+    beforeEach(() => {
+      autosave = jasmine.createSpyObj('autosave', ['reset']);
+      autosave.field = field;
+
+      field.val.and.returnValue('value');
+
+      spyOn(window.localStorage, 'setItem');
+    });
+
+    describe('if .isLocalStorageAvailable is `false`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = false;
+
+        Autosave.prototype.save.call(autosave);
+      });
+
+      it('should not call .setItem', () => {
+        expect(window.localStorage.setItem).not.toHaveBeenCalled();
+      });
+    });
+
+    describe('if .isLocalStorageAvailable is `true`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = true;
+
+        Autosave.prototype.save.call(autosave);
+      });
+
+      it('should call .setItem', () => {
+        expect(window.localStorage.setItem).toHaveBeenCalled();
+      });
+    });
+  });
+
+  describe('reset', () => {
+    const key = 'key';
+
+    beforeEach(() => {
+      autosave = {
+        key,
+      };
+
+      spyOn(window.localStorage, 'removeItem');
+    });
+
+    describe('if .isLocalStorageAvailable is `false`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = false;
+
+        Autosave.prototype.reset.call(autosave);
+      });
+
+      it('should not call .removeItem', () => {
+        expect(window.localStorage.removeItem).not.toHaveBeenCalled();
+      });
+    });
+
+    describe('if .isLocalStorageAvailable is `true`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = true;
+
+        Autosave.prototype.reset.call(autosave);
+      });
+
+      it('should call .removeItem', () => {
+        expect(window.localStorage.removeItem).toHaveBeenCalledWith(key);
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/behaviors/gl_emoji/unicode_support_map_spec.js b/spec/javascripts/behaviors/gl_emoji/unicode_support_map_spec.js
new file mode 100644
index 0000000000000..1ed96a6747813
--- /dev/null
+++ b/spec/javascripts/behaviors/gl_emoji/unicode_support_map_spec.js
@@ -0,0 +1,47 @@
+import { getUnicodeSupportMap } from '~/behaviors/gl_emoji/unicode_support_map';
+import AccessorUtilities from '~/lib/utils/accessor';
+
+describe('Unicode Support Map', () => {
+  describe('getUnicodeSupportMap', () => {
+    const stringSupportMap = 'stringSupportMap';
+
+    beforeEach(() => {
+      spyOn(AccessorUtilities, 'isLocalStorageAccessSafe');
+      spyOn(window.localStorage, 'getItem');
+      spyOn(window.localStorage, 'setItem');
+      spyOn(JSON, 'parse');
+      spyOn(JSON, 'stringify').and.returnValue(stringSupportMap);
+    });
+
+    describe('if isLocalStorageAvailable is `true`', function () {
+      beforeEach(() => {
+        AccessorUtilities.isLocalStorageAccessSafe.and.returnValue(true);
+
+        getUnicodeSupportMap();
+      });
+
+      it('should call .getItem and .setItem', () => {
+        const allArgs = window.localStorage.setItem.calls.allArgs();
+
+        expect(window.localStorage.getItem).toHaveBeenCalledWith('gl-emoji-user-agent');
+        expect(allArgs[0][0]).toBe('gl-emoji-user-agent');
+        expect(allArgs[0][1]).toBe(navigator.userAgent);
+        expect(allArgs[1][0]).toBe('gl-emoji-unicode-support-map');
+        expect(allArgs[1][1]).toBe(stringSupportMap);
+      });
+    });
+
+    describe('if isLocalStorageAvailable is `false`', function () {
+      beforeEach(() => {
+        AccessorUtilities.isLocalStorageAccessSafe.and.returnValue(false);
+
+        getUnicodeSupportMap();
+      });
+
+      it('should not call .getItem or .setItem', () => {
+        expect(window.localStorage.getItem.calls.count()).toBe(1);
+        expect(window.localStorage.setItem).not.toHaveBeenCalled();
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/filtered_search/components/recent_searches_dropdown_content_spec.js b/spec/javascripts/filtered_search/components/recent_searches_dropdown_content_spec.js
index 2722882375f44..d0f09a561d5ab 100644
--- a/spec/javascripts/filtered_search/components/recent_searches_dropdown_content_spec.js
+++ b/spec/javascripts/filtered_search/components/recent_searches_dropdown_content_spec.js
@@ -76,6 +76,26 @@ describe('RecentSearchesDropdownContent', () => {
     });
   });
 
+  describe('if isLocalStorageAvailable is `false`', () => {
+    let el;
+
+    beforeEach(() => {
+      const props = Object.assign({ isLocalStorageAvailable: false }, propsDataWithItems);
+
+      vm = createComponent(props);
+      el = vm.$el;
+    });
+
+    it('should render an info note', () => {
+      const note = el.querySelector('.dropdown-info-note');
+      const items = el.querySelectorAll('.filtered-search-history-dropdown-item');
+
+      expect(note).toBeDefined();
+      expect(note.innerText.trim()).toBe('This feature requires local storage to be enabled');
+      expect(items.length).toEqual(propsDataWithoutItems.items.length);
+    });
+  });
+
   describe('computed', () => {
     describe('processedItems', () => {
       it('with items', () => {
diff --git a/spec/javascripts/filtered_search/filtered_search_manager_spec.js b/spec/javascripts/filtered_search/filtered_search_manager_spec.js
index e747aa497c20b..063d547d00c93 100644
--- a/spec/javascripts/filtered_search/filtered_search_manager_spec.js
+++ b/spec/javascripts/filtered_search/filtered_search_manager_spec.js
@@ -1,3 +1,7 @@
+import * as recentSearchesStoreSrc from '~/filtered_search/stores/recent_searches_store';
+import RecentSearchesService from '~/filtered_search/services/recent_searches_service';
+import RecentSearchesServiceError from '~/filtered_search/services/recent_searches_service_error';
+
 require('~/lib/utils/url_utility');
 require('~/lib/utils/common_utils');
 require('~/filtered_search/filtered_search_token_keys');
@@ -60,6 +64,36 @@ describe('Filtered Search Manager', () => {
     manager.cleanup();
   });
 
+  describe('class constructor', () => {
+    const isLocalStorageAvailable = 'isLocalStorageAvailable';
+    let filteredSearchManager;
+
+    beforeEach(() => {
+      spyOn(RecentSearchesService, 'isAvailable').and.returnValue(isLocalStorageAvailable);
+      spyOn(recentSearchesStoreSrc, 'default');
+
+      filteredSearchManager = new gl.FilteredSearchManager();
+
+      return filteredSearchManager;
+    });
+
+    it('should instantiate RecentSearchesStore with isLocalStorageAvailable', () => {
+      expect(RecentSearchesService.isAvailable).toHaveBeenCalled();
+      expect(recentSearchesStoreSrc.default).toHaveBeenCalledWith({
+        isLocalStorageAvailable,
+      });
+    });
+
+    it('should not instantiate Flash if an RecentSearchesServiceError is caught', () => {
+      spyOn(RecentSearchesService.prototype, 'fetch').and.callFake(() => Promise.reject(new RecentSearchesServiceError()));
+      spyOn(window, 'Flash');
+
+      filteredSearchManager = new gl.FilteredSearchManager();
+
+      expect(window.Flash).not.toHaveBeenCalled();
+    });
+  });
+
   describe('search', () => {
     const defaultParams = '?scope=all&utf8=%E2%9C%93&state=opened';
 
diff --git a/spec/javascripts/filtered_search/recent_searches_root_spec.js b/spec/javascripts/filtered_search/recent_searches_root_spec.js
new file mode 100644
index 0000000000000..d8ba6de5f45d9
--- /dev/null
+++ b/spec/javascripts/filtered_search/recent_searches_root_spec.js
@@ -0,0 +1,31 @@
+import RecentSearchesRoot from '~/filtered_search/recent_searches_root';
+import * as vueSrc from 'vue';
+
+describe('RecentSearchesRoot', () => {
+  describe('render', () => {
+    let recentSearchesRoot;
+    let data;
+    let template;
+
+    beforeEach(() => {
+      recentSearchesRoot = {
+        store: {
+          state: 'state',
+        },
+      };
+
+      spyOn(vueSrc, 'default').and.callFake((options) => {
+        data = options.data;
+        template = options.template;
+      });
+
+      RecentSearchesRoot.prototype.render.call(recentSearchesRoot);
+    });
+
+    it('should instantiate Vue', () => {
+      expect(vueSrc.default).toHaveBeenCalled();
+      expect(data()).toBe(recentSearchesRoot.store.state);
+      expect(template).toContain(':is-local-storage-available="isLocalStorageAvailable"');
+    });
+  });
+});
diff --git a/spec/javascripts/filtered_search/services/recent_searches_service_error_spec.js b/spec/javascripts/filtered_search/services/recent_searches_service_error_spec.js
new file mode 100644
index 0000000000000..ea7c146fa4f01
--- /dev/null
+++ b/spec/javascripts/filtered_search/services/recent_searches_service_error_spec.js
@@ -0,0 +1,18 @@
+import RecentSearchesServiceError from '~/filtered_search/services/recent_searches_service_error';
+
+describe('RecentSearchesServiceError', () => {
+  let recentSearchesServiceError;
+
+  beforeEach(() => {
+    recentSearchesServiceError = new RecentSearchesServiceError();
+  });
+
+  it('instantiates an instance of RecentSearchesServiceError and not an Error', () => {
+    expect(recentSearchesServiceError).toEqual(jasmine.any(RecentSearchesServiceError));
+    expect(recentSearchesServiceError.name).toBe('RecentSearchesServiceError');
+  });
+
+  it('should set a default message', () => {
+    expect(recentSearchesServiceError.message).toBe('Recent Searches Service is unavailable');
+  });
+});
diff --git a/spec/javascripts/filtered_search/services/recent_searches_service_spec.js b/spec/javascripts/filtered_search/services/recent_searches_service_spec.js
index c255bf7c93996..31fa478804af9 100644
--- a/spec/javascripts/filtered_search/services/recent_searches_service_spec.js
+++ b/spec/javascripts/filtered_search/services/recent_searches_service_spec.js
@@ -1,6 +1,7 @@
 /* eslint-disable promise/catch-or-return */
 
 import RecentSearchesService from '~/filtered_search/services/recent_searches_service';
+import AccessorUtilities from '~/lib/utils/accessor';
 
 describe('RecentSearchesService', () => {
   let service;
@@ -11,6 +12,10 @@ describe('RecentSearchesService', () => {
   });
 
   describe('fetch', () => {
+    beforeEach(() => {
+      spyOn(RecentSearchesService, 'isAvailable').and.returnValue(true);
+    });
+
     it('should default to empty array', (done) => {
       const fetchItemsPromise = service.fetch();
 
@@ -29,11 +34,21 @@ describe('RecentSearchesService', () => {
       const fetchItemsPromise = service.fetch();
 
       fetchItemsPromise
-        .catch(() => {
+        .catch((error) => {
+          expect(error).toEqual(jasmine.any(SyntaxError));
           done();
         });
     });
 
+    it('should reject when service is unavailable', (done) => {
+      RecentSearchesService.isAvailable.and.returnValue(false);
+
+      service.fetch().catch((error) => {
+        expect(error).toEqual(jasmine.any(Error));
+        done();
+      });
+    });
+
     it('should return items from localStorage', (done) => {
       window.localStorage.setItem(service.localStorageKey, '["foo", "bar"]');
       const fetchItemsPromise = service.fetch();
@@ -44,15 +59,89 @@ describe('RecentSearchesService', () => {
           done();
         });
     });
+
+    describe('if .isAvailable returns `false`', () => {
+      beforeEach(() => {
+        RecentSearchesService.isAvailable.and.returnValue(false);
+
+        spyOn(window.localStorage, 'getItem');
+
+        RecentSearchesService.prototype.fetch();
+      });
+
+      it('should not call .getItem', () => {
+        expect(window.localStorage.getItem).not.toHaveBeenCalled();
+      });
+    });
   });
 
   describe('setRecentSearches', () => {
+    beforeEach(() => {
+      spyOn(RecentSearchesService, 'isAvailable').and.returnValue(true);
+    });
+
     it('should save things in localStorage', () => {
       const items = ['foo', 'bar'];
       service.save(items);
-      const newLocalStorageValue =
-        window.localStorage.getItem(service.localStorageKey);
+      const newLocalStorageValue = window.localStorage.getItem(service.localStorageKey);
       expect(JSON.parse(newLocalStorageValue)).toEqual(items);
     });
   });
+
+  describe('save', () => {
+    beforeEach(() => {
+      spyOn(window.localStorage, 'setItem');
+      spyOn(RecentSearchesService, 'isAvailable');
+    });
+
+    describe('if .isAvailable returns `true`', () => {
+      const searchesString = 'searchesString';
+      const localStorageKey = 'localStorageKey';
+      const recentSearchesService = {
+        localStorageKey,
+      };
+
+      beforeEach(() => {
+        RecentSearchesService.isAvailable.and.returnValue(true);
+
+        spyOn(JSON, 'stringify').and.returnValue(searchesString);
+
+        RecentSearchesService.prototype.save.call(recentSearchesService);
+      });
+
+      it('should call .setItem', () => {
+        expect(window.localStorage.setItem).toHaveBeenCalledWith(localStorageKey, searchesString);
+      });
+    });
+
+    describe('if .isAvailable returns `false`', () => {
+      beforeEach(() => {
+        RecentSearchesService.isAvailable.and.returnValue(false);
+
+        RecentSearchesService.prototype.save();
+      });
+
+      it('should not call .setItem', () => {
+        expect(window.localStorage.setItem).not.toHaveBeenCalled();
+      });
+    });
+  });
+
+  describe('isAvailable', () => {
+    let isAvailable;
+
+    beforeEach(() => {
+      spyOn(AccessorUtilities, 'isLocalStorageAccessSafe').and.callThrough();
+
+      isAvailable = RecentSearchesService.isAvailable();
+    });
+
+    it('should call .isLocalStorageAccessSafe', () => {
+      expect(AccessorUtilities.isLocalStorageAccessSafe).toHaveBeenCalled();
+    });
+
+    it('should return a boolean', () => {
+      expect(typeof isAvailable).toBe('boolean');
+    });
+  });
 });
diff --git a/spec/javascripts/lib/utils/accessor_spec.js b/spec/javascripts/lib/utils/accessor_spec.js
new file mode 100644
index 0000000000000..b768d6f2a68b4
--- /dev/null
+++ b/spec/javascripts/lib/utils/accessor_spec.js
@@ -0,0 +1,78 @@
+import AccessorUtilities from '~/lib/utils/accessor';
+
+describe('AccessorUtilities', () => {
+  const testError = new Error('test error');
+
+  describe('isPropertyAccessSafe', () => {
+    let base;
+
+    it('should return `true` if access is safe', () => {
+      base = { testProp: 'testProp' };
+
+      expect(AccessorUtilities.isPropertyAccessSafe(base, 'testProp')).toBe(true);
+    });
+
+    it('should return `false` if access throws an error', () => {
+      base = { get testProp() { throw testError; } };
+
+      expect(AccessorUtilities.isPropertyAccessSafe(base, 'testProp')).toBe(false);
+    });
+
+    it('should return `false` if property is undefined', () => {
+      base = {};
+
+      expect(AccessorUtilities.isPropertyAccessSafe(base, 'testProp')).toBe(false);
+    });
+  });
+
+  describe('isFunctionCallSafe', () => {
+    const base = {};
+
+    it('should return `true` if calling is safe', () => {
+      base.func = () => {};
+
+      expect(AccessorUtilities.isFunctionCallSafe(base, 'func')).toBe(true);
+    });
+
+    it('should return `false` if calling throws an error', () => {
+      base.func = () => { throw new Error('test error'); };
+
+      expect(AccessorUtilities.isFunctionCallSafe(base, 'func')).toBe(false);
+    });
+
+    it('should return `false` if function is undefined', () => {
+      base.func = undefined;
+
+      expect(AccessorUtilities.isFunctionCallSafe(base, 'func')).toBe(false);
+    });
+  });
+
+  describe('isLocalStorageAccessSafe', () => {
+    beforeEach(() => {
+      spyOn(window.localStorage, 'setItem');
+      spyOn(window.localStorage, 'removeItem');
+    });
+
+    it('should return `true` if access is safe', () => {
+      expect(AccessorUtilities.isLocalStorageAccessSafe()).toBe(true);
+    });
+
+    it('should return `false` if access to .setItem isnt safe', () => {
+      window.localStorage.setItem.and.callFake(() => { throw testError; });
+
+      expect(AccessorUtilities.isLocalStorageAccessSafe()).toBe(false);
+    });
+
+    it('should set a test item if access is safe', () => {
+      AccessorUtilities.isLocalStorageAccessSafe();
+
+      expect(window.localStorage.setItem).toHaveBeenCalledWith('isLocalStorageAccessSafe', 'true');
+    });
+
+    it('should remove the test item if access is safe', () => {
+      AccessorUtilities.isLocalStorageAccessSafe();
+
+      expect(window.localStorage.removeItem).toHaveBeenCalledWith('isLocalStorageAccessSafe');
+    });
+  });
+});
diff --git a/spec/javascripts/signin_tabs_memoizer_spec.js b/spec/javascripts/signin_tabs_memoizer_spec.js
index d83d9a57b4246..5b4f5933b34de 100644
--- a/spec/javascripts/signin_tabs_memoizer_spec.js
+++ b/spec/javascripts/signin_tabs_memoizer_spec.js
@@ -1,3 +1,5 @@
+import AccessorUtilities from '~/lib/utils/accessor';
+
 require('~/signin_tabs_memoizer');
 
 ((global) => {
@@ -19,6 +21,8 @@ require('~/signin_tabs_memoizer');
 
     beforeEach(() => {
       loadFixtures(fixtureTemplate);
+
+      spyOn(AccessorUtilities, 'isLocalStorageAccessSafe').and.returnValue(true);
     });
 
     it('does nothing if no tab was previously selected', () => {
@@ -49,5 +53,91 @@ require('~/signin_tabs_memoizer');
 
       expect(memo.readData()).toEqual('#standard');
     });
+
+    describe('class constructor', () => {
+      beforeEach(() => {
+        memo = createMemoizer();
+      });
+
+      it('should set .isLocalStorageAvailable', () => {
+        expect(AccessorUtilities.isLocalStorageAccessSafe).toHaveBeenCalled();
+        expect(memo.isLocalStorageAvailable).toBe(true);
+      });
+    });
+
+    describe('saveData', () => {
+      beforeEach(() => {
+        memo = {
+          currentTabKey,
+        };
+
+        spyOn(localStorage, 'setItem');
+      });
+
+      describe('if .isLocalStorageAvailable is `false`', () => {
+        beforeEach(function () {
+          memo.isLocalStorageAvailable = false;
+
+          global.ActiveTabMemoizer.prototype.saveData.call(memo);
+        });
+
+        it('should not call .setItem', () => {
+          expect(localStorage.setItem).not.toHaveBeenCalled();
+        });
+      });
+
+      describe('if .isLocalStorageAvailable is `true`', () => {
+        const value = 'value';
+
+        beforeEach(function () {
+          memo.isLocalStorageAvailable = true;
+
+          global.ActiveTabMemoizer.prototype.saveData.call(memo, value);
+        });
+
+        it('should call .setItem', () => {
+          expect(localStorage.setItem).toHaveBeenCalledWith(currentTabKey, value);
+        });
+      });
+    });
+
+    describe('readData', () => {
+      const itemValue = 'itemValue';
+      let readData;
+
+      beforeEach(() => {
+        memo = {
+          currentTabKey,
+        };
+
+        spyOn(localStorage, 'getItem').and.returnValue(itemValue);
+      });
+
+      describe('if .isLocalStorageAvailable is `false`', () => {
+        beforeEach(function () {
+          memo.isLocalStorageAvailable = false;
+
+          readData = global.ActiveTabMemoizer.prototype.readData.call(memo);
+        });
+
+        it('should not call .getItem and should return `null`', () => {
+          expect(localStorage.getItem).not.toHaveBeenCalled();
+          expect(readData).toBe(null);
+        });
+      });
+
+      describe('if .isLocalStorageAvailable is `true`', () => {
+        beforeEach(function () {
+          memo.isLocalStorageAvailable = true;
+
+          readData = global.ActiveTabMemoizer.prototype.readData.call(memo);
+        });
+
+        it('should call .getItem and return the localStorage value', () => {
+          expect(window.localStorage.getItem).toHaveBeenCalledWith(currentTabKey);
+          expect(readData).toBe(itemValue);
+        });
+      });
+    });
   });
 })(window);
-- 
GitLab


From 4b4fc943a3f4e6379cd5f59d2a3879b8d1ebb703 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Fri, 5 May 2017 13:10:46 -0500
Subject: [PATCH 318/363] Add ruby_parser gem for all environments.

It was only available for test and development environments and we need
it to run the `rake gettext:pack` command in omnibus which doesn't have
the test/dev gems installed.
---
 Gemfile      | 1 +
 Gemfile.lock | 1 +
 2 files changed, 2 insertions(+)

diff --git a/Gemfile b/Gemfile
index 9426a55861283..5dbb0bdfddb5d 100644
--- a/Gemfile
+++ b/Gemfile
@@ -257,6 +257,7 @@ gem 'sentry-raven', '~> 2.4.0'
 gem 'premailer-rails', '~> 1.9.0'
 
 # I18n
+gem 'ruby_parser', '~> 3.8.4', require: false
 gem 'gettext_i18n_rails', '~> 1.8.0'
 gem 'gettext_i18n_rails_js', '~> 1.2.0'
 gem 'gettext', '~> 3.2.2', require: false, group: :development
diff --git a/Gemfile.lock b/Gemfile.lock
index c7e3f9935dad2..01c35a935f225 100644
--- a/Gemfile.lock
+++ b/Gemfile.lock
@@ -1012,6 +1012,7 @@ DEPENDENCIES
   rubocop-rspec (~> 1.15.0)
   ruby-fogbugz (~> 0.2.1)
   ruby-prof (~> 0.16.2)
+  ruby_parser (~> 3.8.4)
   rufus-scheduler (~> 3.1.10)
   rugged (~> 0.25.1.1)
   sanitize (~> 2.0)
-- 
GitLab


From c6d8a517dfa1da2f9180073c6aebcdb05ae2553b Mon Sep 17 00:00:00 2001
From: Nick Thomas <nick@gitlab.com>
Date: Fri, 5 May 2017 19:16:04 +0100
Subject: [PATCH 319/363] Use GitLab Pages v0.4.2

---
 GITLAB_PAGES_VERSION | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/GITLAB_PAGES_VERSION b/GITLAB_PAGES_VERSION
index 267577d47e497..2b7c5ae01848a 100644
--- a/GITLAB_PAGES_VERSION
+++ b/GITLAB_PAGES_VERSION
@@ -1 +1 @@
-0.4.1
+0.4.2
-- 
GitLab


From 48e4991907aa9046c25323af53afc3b7eaa895c4 Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Fri, 5 May 2017 13:23:31 -0500
Subject: [PATCH 320/363] Add sidebar specs

---
 .../time_tracking/sidebar_time_tracking.js    |  20 +-
 .../javascripts/sidebar/sidebar_bundle.js     |   7 +-
 .../javascripts/sidebar/sidebar_mediator.js   |   4 +-
 .../sidebar/stores/sidebar_store.js           |   4 +-
 .../sidebar/assignee_title_spec.js            |  80 ++++++
 spec/javascripts/sidebar/assignees_spec.js    | 272 ++++++++++++++++++
 spec/javascripts/sidebar/mock_data.js         | 109 +++++++
 .../sidebar/sidebar_assignees_spec.js         |  45 +++
 .../sidebar/sidebar_bundle_spec.js            |  42 +++
 .../sidebar/sidebar_mediator_spec.js          |  38 +++
 .../sidebar/sidebar_service_spec.js           |  28 ++
 .../javascripts/sidebar/sidebar_store_spec.js |  80 ++++++
 spec/javascripts/sidebar/user_mock_data.js    |  16 ++
 13 files changed, 732 insertions(+), 13 deletions(-)
 create mode 100644 spec/javascripts/sidebar/assignee_title_spec.js
 create mode 100644 spec/javascripts/sidebar/assignees_spec.js
 create mode 100644 spec/javascripts/sidebar/mock_data.js
 create mode 100644 spec/javascripts/sidebar/sidebar_assignees_spec.js
 create mode 100644 spec/javascripts/sidebar/sidebar_bundle_spec.js
 create mode 100644 spec/javascripts/sidebar/sidebar_mediator_spec.js
 create mode 100644 spec/javascripts/sidebar/sidebar_service_spec.js
 create mode 100644 spec/javascripts/sidebar/sidebar_store_spec.js
 create mode 100644 spec/javascripts/sidebar/user_mock_data.js

diff --git a/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js b/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
index e2dba1fb0c2e7..244b67b3ad926 100644
--- a/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
+++ b/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
@@ -17,15 +17,21 @@ export default {
   },
   methods: {
     listenForSlashCommands() {
-      $(document).on('ajax:success', '.gfm-form', (e, data) => {
-        const subscribedCommands = ['spend_time', 'time_estimate'];
-        const changedCommands = data.commands_changes
+      $(document).on('ajax:success', '.gfm-form', this.slashCommandListened);
+    },
+    slashCommandListened(e, data) {
+      const subscribedCommands = ['spend_time', 'time_estimate'];
+      let changedCommands;
+      if (data !== undefined) {
+        changedCommands = data.commands_changes
           ? Object.keys(data.commands_changes)
           : [];
-        if (changedCommands && _.intersection(subscribedCommands, changedCommands).length) {
-          this.mediator.fetch();
-        }
-      });
+      } else {
+        changedCommands = [];
+      }
+      if (changedCommands && _.intersection(subscribedCommands, changedCommands).length) {
+        this.mediator.fetch();
+      }
     },
   },
   mounted() {
diff --git a/app/assets/javascripts/sidebar/sidebar_bundle.js b/app/assets/javascripts/sidebar/sidebar_bundle.js
index 2ce53c2ed30d5..2b02af87d8ac4 100644
--- a/app/assets/javascripts/sidebar/sidebar_bundle.js
+++ b/app/assets/javascripts/sidebar/sidebar_bundle.js
@@ -4,7 +4,7 @@ import sidebarAssignees from './components/assignees/sidebar_assignees';
 
 import Mediator from './sidebar_mediator';
 
-document.addEventListener('DOMContentLoaded', () => {
+function domContentLoaded() {
   const mediator = new Mediator(gl.sidebarOptions);
   mediator.fetch();
 
@@ -17,5 +17,8 @@ document.addEventListener('DOMContentLoaded', () => {
   }
 
   new Vue(sidebarTimeTracking).$mount('#issuable-time-tracker');
-});
+}
 
+document.addEventListener('DOMContentLoaded', domContentLoaded);
+
+export default domContentLoaded;
diff --git a/app/assets/javascripts/sidebar/sidebar_mediator.js b/app/assets/javascripts/sidebar/sidebar_mediator.js
index c13f3391f0d0e..5ccfb4ee9c1d3 100644
--- a/app/assets/javascripts/sidebar/sidebar_mediator.js
+++ b/app/assets/javascripts/sidebar/sidebar_mediator.js
@@ -30,8 +30,8 @@ export default class SidebarMediator {
     this.service.get()
       .then((response) => {
         const data = response.json();
-        this.store.processAssigneeData(data);
-        this.store.processTimeTrackingData(data);
+        this.store.setAssigneeData(data);
+        this.store.setTimeTrackingData(data);
       })
       .catch(() => new Flash('Error occured when fetching sidebar data'));
   }
diff --git a/app/assets/javascripts/sidebar/stores/sidebar_store.js b/app/assets/javascripts/sidebar/stores/sidebar_store.js
index 94408c4d71547..2d44c05bb8d24 100644
--- a/app/assets/javascripts/sidebar/stores/sidebar_store.js
+++ b/app/assets/javascripts/sidebar/stores/sidebar_store.js
@@ -17,13 +17,13 @@ export default class SidebarStore {
     return SidebarStore.singleton;
   }
 
-  processAssigneeData(data) {
+  setAssigneeData(data) {
     if (data.assignees) {
       this.assignees = data.assignees;
     }
   }
 
-  processTimeTrackingData(data) {
+  setTimeTrackingData(data) {
     this.timeEstimate = data.time_estimate;
     this.totalTimeSpent = data.total_time_spent;
     this.humanTimeEstimate = data.human_time_estimate;
diff --git a/spec/javascripts/sidebar/assignee_title_spec.js b/spec/javascripts/sidebar/assignee_title_spec.js
new file mode 100644
index 0000000000000..5b5b1bf414010
--- /dev/null
+++ b/spec/javascripts/sidebar/assignee_title_spec.js
@@ -0,0 +1,80 @@
+import Vue from 'vue';
+import AssigneeTitle from '~/sidebar/components/assignees/assignee_title';
+
+describe('AssigneeTitle component', () => {
+  let component;
+  let AssigneeTitleComponent;
+
+  beforeEach(() => {
+    AssigneeTitleComponent = Vue.extend(AssigneeTitle);
+  });
+
+  describe('assignee title', () => {
+    it('renders assignee', () => {
+      component = new AssigneeTitleComponent({
+        propsData: {
+          numberOfAssignees: 1,
+          editable: false,
+        },
+      }).$mount();
+
+      expect(component.$el.innerText.trim()).toEqual('Assignee');
+    });
+
+    it('renders 2 assignees', () => {
+      component = new AssigneeTitleComponent({
+        propsData: {
+          numberOfAssignees: 2,
+          editable: false,
+        },
+      }).$mount();
+
+      expect(component.$el.innerText.trim()).toEqual('2 Assignees');
+    });
+  });
+
+  it('does not render spinner by default', () => {
+    component = new AssigneeTitleComponent({
+      propsData: {
+        numberOfAssignees: 0,
+        editable: false,
+      },
+    }).$mount();
+
+    expect(component.$el.querySelector('.fa')).toBeNull();
+  });
+
+  it('renders spinner when loading', () => {
+    component = new AssigneeTitleComponent({
+      propsData: {
+        loading: true,
+        numberOfAssignees: 0,
+        editable: false,
+      },
+    }).$mount();
+
+    expect(component.$el.querySelector('.fa')).not.toBeNull();
+  });
+
+  it('does not render edit link when not editable', () => {
+    component = new AssigneeTitleComponent({
+      propsData: {
+        numberOfAssignees: 0,
+        editable: false,
+      },
+    }).$mount();
+
+    expect(component.$el.querySelector('.edit-link')).toBeNull();
+  });
+
+  it('renders edit link when editable', () => {
+    component = new AssigneeTitleComponent({
+      propsData: {
+        numberOfAssignees: 0,
+        editable: true,
+      },
+    }).$mount();
+
+    expect(component.$el.querySelector('.edit-link')).not.toBeNull();
+  });
+});
diff --git a/spec/javascripts/sidebar/assignees_spec.js b/spec/javascripts/sidebar/assignees_spec.js
new file mode 100644
index 0000000000000..cf47940ef1c18
--- /dev/null
+++ b/spec/javascripts/sidebar/assignees_spec.js
@@ -0,0 +1,272 @@
+import Vue from 'vue';
+import Assignee from '~/sidebar/components/assignees/assignees';
+import UsersMock from './mock_data';
+import UsersMockHelper from '../test_helpers/user_mock_data';
+
+describe('Assignee component', () => {
+  let component;
+  let AssigneeComponent;
+
+  beforeEach(() => {
+    AssigneeComponent = Vue.extend(Assignee);
+  });
+
+  describe('No assignees/users', () => {
+    it('displays no assignee icon when collapsed', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [],
+          editable: false,
+        },
+      }).$mount();
+
+      const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
+      expect(collapsed.childElementCount).toEqual(1);
+      expect(collapsed.children[0].getAttribute('aria-label')).toEqual('No Assignee');
+      expect(collapsed.children[0].classList.contains('fa')).toEqual(true);
+      expect(collapsed.children[0].classList.contains('fa-user')).toEqual(true);
+    });
+
+    it('displays only "No assignee" when no users are assigned and the issue is read-only', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [],
+          editable: false,
+        },
+      }).$mount();
+      const componentTextNoUsers = component.$el.querySelector('.assign-yourself').innerText.trim();
+
+      expect(componentTextNoUsers).toBe('No assignee');
+      expect(componentTextNoUsers.indexOf('assign yourself')).toEqual(-1);
+    });
+
+    it('displays only "No assignee" when no users are assigned and the issue can be edited', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [],
+          editable: true,
+        },
+      }).$mount();
+      const componentTextNoUsers = component.$el.querySelector('.assign-yourself').innerText.trim();
+
+      expect(componentTextNoUsers.indexOf('No assignee')).toEqual(0);
+      expect(componentTextNoUsers.indexOf('assign yourself')).toBeGreaterThan(0);
+    });
+
+    it('emits the assign-self event when "assign yourself" is clicked', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [],
+          editable: true,
+        },
+      }).$mount();
+
+      spyOn(component, '$emit');
+      component.$el.querySelector('.assign-yourself .btn-link').click();
+      expect(component.$emit).toHaveBeenCalledWith('assign-self');
+    });
+  });
+
+  describe('One assignee/user', () => {
+    it('displays one assignee icon when collapsed', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [
+            UsersMock.user,
+          ],
+          editable: false,
+        },
+      }).$mount();
+
+      const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
+      const assignee = collapsed.children[0];
+      expect(collapsed.childElementCount).toEqual(1);
+      expect(assignee.querySelector('.avatar').getAttribute('src')).toEqual(UsersMock.user.avatarUrl);
+      expect(assignee.querySelector('.avatar').getAttribute('alt')).toEqual(`${UsersMock.user.name}'s avatar`);
+      expect(assignee.querySelector('.author').innerText.trim()).toEqual(UsersMock.user.name);
+    });
+
+    it('Shows one user with avatar, username and author name', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000/',
+          users: [
+            UsersMock.user,
+          ],
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.$el.querySelector('.author_link')).not.toBeNull();
+      // The image
+      expect(component.$el.querySelector('.author_link img').getAttribute('src')).toEqual(UsersMock.user.avatarUrl);
+      // Author name
+      expect(component.$el.querySelector('.author_link .author').innerText.trim()).toEqual(UsersMock.user.name);
+      // Username
+      expect(component.$el.querySelector('.author_link .username').innerText.trim()).toEqual(`@${UsersMock.user.username}`);
+    });
+
+    it('has the root url present in the assigneeUrl method', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000/',
+          users: [
+            UsersMock.user,
+          ],
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.assigneeUrl(UsersMock.user).indexOf('http://localhost:3000/')).not.toEqual(-1);
+    });
+  });
+
+  describe('Two or more assignees/users', () => {
+    it('displays two assignee icons when collapsed', () => {
+      const users = UsersMockHelper.createNumberRandomUsers(2);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: false,
+        },
+      }).$mount();
+
+      const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
+      expect(collapsed.childElementCount).toEqual(2);
+
+      const first = collapsed.children[0];
+      expect(first.querySelector('.avatar').getAttribute('src')).toEqual(users[0].avatarUrl);
+      expect(first.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[0].name}'s avatar`);
+      expect(first.querySelector('.author').innerText.trim()).toEqual(users[0].name);
+
+      const second = collapsed.children[1];
+      expect(second.querySelector('.avatar').getAttribute('src')).toEqual(users[1].avatarUrl);
+      expect(second.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[1].name}'s avatar`);
+      expect(second.querySelector('.author').innerText.trim()).toEqual(users[1].name);
+    });
+
+    it('displays one assignee icon and counter when collapsed', () => {
+      const users = UsersMockHelper.createNumberRandomUsers(3);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: false,
+        },
+      }).$mount();
+
+      const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
+      expect(collapsed.childElementCount).toEqual(2);
+
+      const first = collapsed.children[0];
+      expect(first.querySelector('.avatar').getAttribute('src')).toEqual(users[0].avatarUrl);
+      expect(first.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[0].name}'s avatar`);
+      expect(first.querySelector('.author').innerText.trim()).toEqual(users[0].name);
+
+      const second = collapsed.children[1];
+      expect(second.querySelector('.avatar-counter').innerText.trim()).toEqual('+2');
+    });
+
+    it('Shows two assignees', () => {
+      const users = UsersMockHelper.createNumberRandomUsers(2);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.$el.querySelectorAll('.user-item').length).toEqual(users.length);
+      expect(component.$el.querySelector('.user-list-more')).toBe(null);
+    });
+
+    it('Shows the "show-less" assignees label', (done) => {
+      const users = UsersMockHelper.createNumberRandomUsers(6);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.$el.querySelectorAll('.user-item').length).toEqual(component.defaultRenderCount);
+      expect(component.$el.querySelector('.user-list-more')).not.toBe(null);
+      const usersLabelExpectation = users.length - component.defaultRenderCount;
+      expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+        .not.toBe(`+${usersLabelExpectation} more`);
+      component.toggleShowLess();
+      Vue.nextTick(() => {
+        expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+          .toBe('- show less');
+        done();
+      });
+    });
+
+    it('Shows the "show-less" when "n+ more " label is clicked', (done) => {
+      const users = UsersMockHelper.createNumberRandomUsers(6);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: true,
+        },
+      }).$mount();
+
+      component.$el.querySelector('.user-list-more .btn-link').click();
+      Vue.nextTick(() => {
+        expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+          .toBe('- show less');
+        done();
+      });
+    });
+
+    it('gets the count of avatar via a computed property ', () => {
+      const users = UsersMockHelper.createNumberRandomUsers(6);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.sidebarAvatarCounter).toEqual(`+${users.length - 1}`);
+    });
+
+    describe('n+ more label', () => {
+      beforeEach(() => {
+        const users = UsersMockHelper.createNumberRandomUsers(6);
+        component = new AssigneeComponent({
+          propsData: {
+            rootPath: 'http://localhost:3000',
+            users,
+            editable: true,
+          },
+        }).$mount();
+      });
+
+      it('shows "+1 more" label', () => {
+        expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+          .toBe('+ 1 more');
+      });
+
+      it('shows "show less" label', (done) => {
+        component.toggleShowLess();
+
+        Vue.nextTick(() => {
+          expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+            .toBe('- show less');
+          done();
+        });
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/sidebar/mock_data.js b/spec/javascripts/sidebar/mock_data.js
new file mode 100644
index 0000000000000..0599a63912c71
--- /dev/null
+++ b/spec/javascripts/sidebar/mock_data.js
@@ -0,0 +1,109 @@
+/* eslint-disable quote-props*/
+
+const sidebarMockData = {
+  'GET': {
+    '/gitlab-org/gitlab-shell/issues/5.json': {
+      id: 45,
+      iid: 5,
+      author_id: 23,
+      description: 'Nulla ullam commodi delectus adipisci quis sit.',
+      lock_version: null,
+      milestone_id: 21,
+      position: 0,
+      state: 'closed',
+      title: 'Vel et nulla voluptatibus corporis dolor iste saepe laborum.',
+      updated_by_id: 1,
+      created_at: '2017-02-02T21: 49: 49.664Z',
+      updated_at: '2017-05-03T22: 26: 03.760Z',
+      deleted_at: null,
+      time_estimate: 0,
+      total_time_spent: 0,
+      human_time_estimate: null,
+      human_total_time_spent: null,
+      branch_name: null,
+      confidential: false,
+      assignees: [
+        {
+          name: 'User 0',
+          username: 'user0',
+          id: 22,
+          state: 'active',
+          avatar_url: 'http: //www.gravatar.com/avatar/52e4ce24a915fb7e51e1ad3b57f4b00a?s=80\u0026d=identicon',
+          web_url: 'http: //localhost:3001/user0',
+        },
+        {
+          name: 'Marguerite Bartell',
+          username: 'tajuana',
+          id: 18,
+          state: 'active',
+          avatar_url: 'http: //www.gravatar.com/avatar/4852a41fb41616bf8f140d3701673f53?s=80\u0026d=identicon',
+          web_url: 'http: //localhost:3001/tajuana',
+        },
+        {
+          name: 'Laureen Ritchie',
+          username: 'michaele.will',
+          id: 16,
+          state: 'active',
+          avatar_url: 'http: //www.gravatar.com/avatar/e301827eb03be955c9c172cb9a8e4e8a?s=80\u0026d=identicon',
+          web_url: 'http: //localhost:3001/michaele.will',
+        },
+      ],
+      due_date: null,
+      moved_to_id: null,
+      project_id: 4,
+      weight: null,
+      milestone: {
+        id: 21,
+        iid: 1,
+        project_id: 4,
+        title: 'v0.0',
+        description: 'Molestiae commodi laboriosam odio sunt eaque reprehenderit.',
+        state: 'active',
+        created_at: '2017-02-02T21: 49: 30.530Z',
+        updated_at: '2017-02-02T21: 49: 30.530Z',
+        due_date: null,
+        start_date: null,
+      },
+      labels: [],
+    },
+  },
+  'PUT': {
+    '/gitlab-org/gitlab-shell/issues/5.json': {
+      data: {},
+    },
+  },
+};
+
+export default {
+  mediator: {
+    endpoint: '/gitlab-org/gitlab-shell/issues/5.json',
+    editable: true,
+    currentUser: {
+      id: 1,
+      name: 'Administrator',
+      username: 'root',
+      avatar_url: 'http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+    },
+    rootPath: '/',
+  },
+  time: {
+    time_estimate: 3600,
+    total_time_spent: 0,
+    human_time_estimate: '1h',
+    human_total_time_spent: null,
+  },
+  user: {
+    avatarUrl: 'http://gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+    id: 1,
+    name: 'Administrator',
+    username: 'root',
+  },
+
+  sidebarMockInterceptor(request, next) {
+    const body = sidebarMockData[request.method.toUpperCase()][request.url];
+
+    next(request.respondWith(JSON.stringify(body), {
+      status: 200,
+    }));
+  },
+};
diff --git a/spec/javascripts/sidebar/sidebar_assignees_spec.js b/spec/javascripts/sidebar/sidebar_assignees_spec.js
new file mode 100644
index 0000000000000..e0df0a3228f9f
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_assignees_spec.js
@@ -0,0 +1,45 @@
+import Vue from 'vue';
+import SidebarAssignees from '~/sidebar/components/assignees/sidebar_assignees';
+import SidebarMediator from '~/sidebar/sidebar_mediator';
+import SidebarService from '~/sidebar/services/sidebar_service';
+import SidebarStore from '~/sidebar/stores/sidebar_store';
+import Mock from './mock_data';
+
+describe('sidebar assignees', () => {
+  let component;
+  let SidebarAssigneeComponent;
+  preloadFixtures('issues/open-issue.html.raw');
+
+  beforeEach(() => {
+    Vue.http.interceptors.push(Mock.sidebarMockInterceptor);
+    SidebarAssigneeComponent = Vue.extend(SidebarAssignees);
+    spyOn(SidebarMediator.prototype, 'saveAssignees').and.callThrough();
+    spyOn(SidebarMediator.prototype, 'assignYourself').and.callThrough();
+    this.mediator = new SidebarMediator(Mock.mediator);
+    loadFixtures('issues/open-issue.html.raw');
+    this.sidebarAssigneesEl = document.querySelector('#js-vue-sidebar-assignees');
+  });
+
+  afterEach(() => {
+    SidebarService.singleton = null;
+    SidebarStore.singleton = null;
+    SidebarMediator.singleton = null;
+  });
+
+  it('calls the mediator when saves the assignees', () => {
+    component = new SidebarAssigneeComponent()
+      .$mount(this.sidebarAssigneesEl);
+    component.saveAssignees();
+
+    expect(SidebarMediator.prototype.saveAssignees).toHaveBeenCalled();
+  });
+
+  it('calls the mediator when "assignSelf" method is called', () => {
+    component = new SidebarAssigneeComponent()
+      .$mount(this.sidebarAssigneesEl);
+    component.assignSelf();
+
+    expect(SidebarMediator.prototype.assignYourself).toHaveBeenCalled();
+    expect(this.mediator.store.assignees.length).toEqual(1);
+  });
+});
diff --git a/spec/javascripts/sidebar/sidebar_bundle_spec.js b/spec/javascripts/sidebar/sidebar_bundle_spec.js
new file mode 100644
index 0000000000000..7760b34e07198
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_bundle_spec.js
@@ -0,0 +1,42 @@
+import Vue from 'vue';
+import SidebarBundleDomContentLoaded from '~/sidebar/sidebar_bundle';
+import SidebarTimeTracking from '~/sidebar/components/time_tracking/sidebar_time_tracking';
+import SidebarMediator from '~/sidebar/sidebar_mediator';
+import SidebarService from '~/sidebar/services/sidebar_service';
+import SidebarStore from '~/sidebar/stores/sidebar_store';
+import Mock from './mock_data';
+
+describe('sidebar bundle', () => {
+  gl.sidebarOptions = Mock.mediator;
+
+  beforeEach(() => {
+    spyOn(SidebarTimeTracking.methods, 'listenForSlashCommands').and.callFake(() => { });
+    preloadFixtures('issues/open-issue.html.raw');
+    Vue.http.interceptors.push(Mock.sidebarMockInterceptor);
+    loadFixtures('issues/open-issue.html.raw');
+    spyOn(Vue.prototype, '$mount');
+    SidebarBundleDomContentLoaded();
+    this.mediator = new SidebarMediator();
+  });
+
+  afterEach(() => {
+    SidebarService.singleton = null;
+    SidebarStore.singleton = null;
+    SidebarMediator.singleton = null;
+  });
+
+  it('the mediator should be already defined with some data', () => {
+    SidebarBundleDomContentLoaded();
+
+    expect(this.mediator.store).toBeDefined();
+    expect(this.mediator.service).toBeDefined();
+    expect(this.mediator.store.currentUser).toEqual(Mock.mediator.currentUser);
+    expect(this.mediator.store.rootPath).toEqual(Mock.mediator.rootPath);
+    expect(this.mediator.store.endPoint).toEqual(Mock.mediator.endPoint);
+    expect(this.mediator.store.editable).toEqual(Mock.mediator.editable);
+  });
+
+  it('the sidebar time tracking and assignees components to have been mounted', () => {
+    expect(Vue.prototype.$mount).toHaveBeenCalledTimes(2);
+  });
+});
diff --git a/spec/javascripts/sidebar/sidebar_mediator_spec.js b/spec/javascripts/sidebar/sidebar_mediator_spec.js
new file mode 100644
index 0000000000000..9bfca0c099103
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_mediator_spec.js
@@ -0,0 +1,38 @@
+import Vue from 'vue';
+import SidebarMediator from '~/sidebar/sidebar_mediator';
+import SidebarStore from '~/sidebar/stores/sidebar_store';
+import SidebarService from '~/sidebar/services/sidebar_service';
+import Mock from './mock_data';
+
+describe('Sidebar mediator', () => {
+  beforeEach(() => {
+    Vue.http.interceptors.push(Mock.sidebarMockInterceptor);
+    this.mediator = new SidebarMediator(Mock.mediator);
+  });
+
+  afterEach(() => {
+    SidebarService.singleton = null;
+    SidebarStore.singleton = null;
+    SidebarMediator.singleton = null;
+  });
+
+  it('assigns yourself ', () => {
+    this.mediator.assignYourself();
+
+    expect(this.mediator.store.currentUser).toEqual(Mock.mediator.currentUser);
+    expect(this.mediator.store.assignees[0]).toEqual(Mock.mediator.currentUser);
+  });
+
+  it('saves assignees', (done) => {
+    this.mediator.saveAssignees('issue[assignee_ids]').then((resp) => {
+      expect(resp.status).toEqual(200);
+      done();
+    });
+  });
+
+  it('fetches the data', () => {
+    spyOn(this.mediator.service, 'get').and.callThrough();
+    this.mediator.fetch();
+    expect(this.mediator.service.get).toHaveBeenCalled();
+  });
+});
diff --git a/spec/javascripts/sidebar/sidebar_service_spec.js b/spec/javascripts/sidebar/sidebar_service_spec.js
new file mode 100644
index 0000000000000..ed7be76dbcc8d
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_service_spec.js
@@ -0,0 +1,28 @@
+import Vue from 'vue';
+import SidebarService from '~/sidebar/services/sidebar_service';
+import Mock from './mock_data';
+
+describe('Sidebar service', () => {
+  beforeEach(() => {
+    Vue.http.interceptors.push(Mock.sidebarMockInterceptor);
+    this.service = new SidebarService('/gitlab-org/gitlab-shell/issues/5.json');
+  });
+
+  afterEach(() => {
+    SidebarService.singleton = null;
+  });
+
+  it('gets the data', (done) => {
+    this.service.get().then((resp) => {
+      expect(resp).toBeDefined();
+      done();
+    });
+  });
+
+  it('updates the data', (done) => {
+    this.service.update('issue[assignee_ids]', [1]).then((resp) => {
+      expect(resp).toBeDefined();
+      done();
+    });
+  });
+});
diff --git a/spec/javascripts/sidebar/sidebar_store_spec.js b/spec/javascripts/sidebar/sidebar_store_spec.js
new file mode 100644
index 0000000000000..6c0f2b6a793f8
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_store_spec.js
@@ -0,0 +1,80 @@
+import SidebarStore from '~/sidebar/stores/sidebar_store';
+import Mock from './mock_data';
+import UsersMockHelper from '../test_helpers/user_mock_data';
+
+describe('Sidebar store', () => {
+  const assignee = {
+    id: 2,
+    name: 'gitlab user 2',
+    username: 'gitlab2',
+    avatar_url: 'http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+  };
+
+  const anotherAssignee = {
+    id: 3,
+    name: 'gitlab user 3',
+    username: 'gitlab3',
+    avatar_url: 'http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+  };
+
+  beforeEach(() => {
+    this.store = new SidebarStore({
+      currentUser: {
+        id: 1,
+        name: 'Administrator',
+        username: 'root',
+        avatar_url: 'http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+      },
+      editable: true,
+      rootPath: '/',
+      endpoint: '/gitlab-org/gitlab-shell/issues/5.json',
+    });
+  });
+
+  afterEach(() => {
+    SidebarStore.singleton = null;
+  });
+
+  it('adds a new assignee', () => {
+    this.store.addAssignee(assignee);
+    expect(this.store.assignees.length).toEqual(1);
+  });
+
+  it('removes an assignee', () => {
+    this.store.removeAssignee(assignee);
+    expect(this.store.assignees.length).toEqual(0);
+  });
+
+  it('finds an existent assignee', () => {
+    let foundAssignee;
+
+    this.store.addAssignee(assignee);
+    foundAssignee = this.store.findAssignee(assignee);
+    expect(foundAssignee).toBeDefined();
+    expect(foundAssignee).toEqual(assignee);
+    foundAssignee = this.store.findAssignee(anotherAssignee);
+    expect(foundAssignee).toBeUndefined();
+  });
+
+  it('removes all assignees', () => {
+    this.store.removeAllAssignees();
+    expect(this.store.assignees.length).toEqual(0);
+  });
+
+  it('set assigned data', () => {
+    const users = {
+      assignees: UsersMockHelper.createNumberRandomUsers(3),
+    };
+
+    this.store.setAssigneeData(users);
+    expect(this.store.assignees.length).toEqual(3);
+  });
+
+  it('set time tracking data', () => {
+    this.store.setTimeTrackingData(Mock.time);
+    expect(this.store.timeEstimate).toEqual(Mock.time.time_estimate);
+    expect(this.store.totalTimeSpent).toEqual(Mock.time.total_time_spent);
+    expect(this.store.humanTimeEstimate).toEqual(Mock.time.human_time_estimate);
+    expect(this.store.humanTotalTimeSpent).toEqual(Mock.time.human_total_time_spent);
+  });
+});
diff --git a/spec/javascripts/sidebar/user_mock_data.js b/spec/javascripts/sidebar/user_mock_data.js
new file mode 100644
index 0000000000000..9e7b834623878
--- /dev/null
+++ b/spec/javascripts/sidebar/user_mock_data.js
@@ -0,0 +1,16 @@
+export default {
+  createNumberRandomUsers(numberUsers) {
+    const users = [];
+    for (let i = 0; i < numberUsers; i = i += 1) {
+      users.push(
+        {
+          avatarUrl: 'http://gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+          id: (i + 1),
+          name: `GitLab User ${i}`,
+          username: `gitlab${i}`,
+        },
+      );
+    }
+    return users;
+  },
+};
-- 
GitLab


From 8985ea1b9c649183e997a76c32aca927a258b51e Mon Sep 17 00:00:00 2001
From: Regis <boudinot.regis@yahoo.com>
Date: Fri, 5 May 2017 12:28:06 -0600
Subject: [PATCH 321/363] add changelog [ci skip]

---
 changelogs/unreleased/issue-title-description-realtime.yml | 4 ++++
 1 file changed, 4 insertions(+)
 create mode 100644 changelogs/unreleased/issue-title-description-realtime.yml

diff --git a/changelogs/unreleased/issue-title-description-realtime.yml b/changelogs/unreleased/issue-title-description-realtime.yml
new file mode 100644
index 0000000000000..003e1a4ab334e
--- /dev/null
+++ b/changelogs/unreleased/issue-title-description-realtime.yml
@@ -0,0 +1,4 @@
+---
+title: Add realtime descriptions to issue show pages
+merge_request:
+author:
-- 
GitLab


From f4a2dfb46f168d3fd7309aca8631cf680456fa82 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Fri, 21 Apr 2017 14:05:19 -0700
Subject: [PATCH 322/363] Add happy path feature tests for redirect behavior

---
 spec/features/groups/group_settings_spec.rb   |  77 +++++++++
 spec/features/profiles/account_spec.rb        |  60 +++++++
 .../projects/project_settings_spec.rb         | 146 ++++++++++++++----
 3 files changed, 255 insertions(+), 28 deletions(-)
 create mode 100644 spec/features/groups/group_settings_spec.rb
 create mode 100644 spec/features/profiles/account_spec.rb

diff --git a/spec/features/groups/group_settings_spec.rb b/spec/features/groups/group_settings_spec.rb
new file mode 100644
index 0000000000000..f18f2f2310f03
--- /dev/null
+++ b/spec/features/groups/group_settings_spec.rb
@@ -0,0 +1,77 @@
+require 'spec_helper'
+
+feature 'Edit group settings', feature: true do
+  given(:user)  { create(:user) }
+  given(:group) { create(:group, path: 'foo') }
+
+  background do
+    group.add_owner(user)
+    login_as(user)
+  end
+
+  describe 'when the group path is changed' do
+    let(:new_group_path) { 'bar' }
+    let(:old_group_full_path) { "/#{group.path}" }
+    let(:new_group_full_path) { "/#{new_group_path}" }
+
+    scenario 'the group is accessible via the new path' do
+      update_path(new_group_path)
+      visit new_group_full_path
+      expect(current_path).to eq(new_group_full_path)
+      expect(find('h1.group-title')).to have_content(new_group_path)
+    end
+
+    scenario 'the old group path redirects to the new path' do
+      update_path(new_group_path)
+      visit old_group_full_path
+      expect(current_path).to eq(new_group_full_path)
+      expect(find('h1.group-title')).to have_content(new_group_path)
+    end
+
+    context 'with a subgroup' do
+      given!(:subgroup) { create(:group, parent: group, path: 'subgroup') }
+      given(:old_subgroup_full_path) { "/#{group.path}/#{subgroup.path}" }
+      given(:new_subgroup_full_path) { "/#{new_group_path}/#{subgroup.path}" }
+
+      scenario 'the subgroup is accessible via the new path' do
+        update_path(new_group_path)
+        visit new_subgroup_full_path
+        expect(current_path).to eq(new_subgroup_full_path)
+        expect(find('h1.group-title')).to have_content(subgroup.path)
+      end
+
+      scenario 'the old subgroup path redirects to the new path' do
+        update_path(new_group_path)
+        visit old_subgroup_full_path
+        expect(current_path).to eq(new_subgroup_full_path)
+        expect(find('h1.group-title')).to have_content(subgroup.path)
+      end
+    end
+
+    context 'with a project' do
+      given!(:project) { create(:project, group: group, path: 'project') }
+      given(:old_project_full_path) { "/#{group.path}/#{project.path}" }
+      given(:new_project_full_path) { "/#{new_group_path}/#{project.path}" }
+
+      scenario 'the project is accessible via the new path' do
+        update_path(new_group_path)
+        visit new_project_full_path
+        expect(current_path).to eq(new_project_full_path)
+        expect(find('h1.project-title')).to have_content(project.name)
+      end
+
+      scenario 'the old project path redirects to the new path' do
+        update_path(new_group_path)
+        visit old_project_full_path
+        expect(current_path).to eq(new_project_full_path)
+        expect(find('h1.project-title')).to have_content(project.name)
+      end
+    end
+  end
+end
+
+def update_path(new_group_path)
+  visit edit_group_path(group)
+  fill_in 'group_path', with: new_group_path
+  click_button 'Save group'
+end
diff --git a/spec/features/profiles/account_spec.rb b/spec/features/profiles/account_spec.rb
new file mode 100644
index 0000000000000..b3902a6b37f7b
--- /dev/null
+++ b/spec/features/profiles/account_spec.rb
@@ -0,0 +1,60 @@
+require 'rails_helper'
+
+feature 'Profile > Account', feature: true do
+  given(:user) { create(:user, username: 'foo') }
+
+  before do
+    login_as(user)
+  end
+
+  describe 'Change username' do
+    given(:new_username) { 'bar' }
+    given(:new_user_path) { "/#{new_username}" }
+    given(:old_user_path) { "/#{user.username}" }
+
+    scenario 'the user is accessible via the new path' do
+      update_username(new_username)
+      visit new_user_path
+      expect(current_path).to eq(new_user_path)
+      expect(find('.user-info')).to have_content(new_username)
+    end
+
+    scenario 'the old user path redirects to the new path' do
+      update_username(new_username)
+      visit old_user_path
+      expect(current_path).to eq(new_user_path)
+      expect(find('.user-info')).to have_content(new_username)
+    end
+
+    context 'with a project' do
+      given!(:project) { create(:project, namespace: user.namespace, path: 'project') }
+      given(:new_project_path) { "/#{new_username}/#{project.path}" }
+      given(:old_project_path) { "/#{user.username}/#{project.path}" }
+
+      after do
+        TestEnv.clean_test_path
+      end
+
+      scenario 'the project is accessible via the new path' do
+        update_username(new_username)
+        visit new_project_path
+        expect(current_path).to eq(new_project_path)
+        expect(find('h1.project-title')).to have_content(project.name)
+      end
+
+      scenario 'the old project path redirects to the new path' do
+        update_username(new_username)
+        visit old_project_path
+        expect(current_path).to eq(new_project_path)
+        expect(find('h1.project-title')).to have_content(project.name)
+      end
+    end
+  end
+end
+
+def update_username(new_username)
+  allow(user.namespace).to receive(:move_dir)
+  visit profile_account_path
+  fill_in 'user_username', with: new_username
+  click_button 'Update username'
+end
diff --git a/spec/features/projects/project_settings_spec.rb b/spec/features/projects/project_settings_spec.rb
index 5d0314d5c0987..364cc8ef0f29a 100644
--- a/spec/features/projects/project_settings_spec.rb
+++ b/spec/features/projects/project_settings_spec.rb
@@ -1,64 +1,154 @@
 require 'spec_helper'
 
 describe 'Edit Project Settings', feature: true do
+  include Select2Helper
+
   let(:user) { create(:user) }
-  let(:project) { create(:empty_project, path: 'gitlab', name: 'sample') }
+  let(:project) { create(:empty_project, namespace: user.namespace, path: 'gitlab', name: 'sample') }
 
   before do
     login_as(user)
-    project.team << [user, :master]
   end
 
-  describe 'Project settings', js: true do
+  describe 'Project settings section', js: true do
     it 'shows errors for invalid project name' do
       visit edit_namespace_project_path(project.namespace, project)
-
       fill_in 'project_name_edit', with: 'foo&bar'
-
       click_button 'Save changes'
-
       expect(page).to have_field 'project_name_edit', with: 'foo&bar'
       expect(page).to have_content "Name can contain only letters, digits, emojis, '_', '.', dash, space. It must start with letter, digit, emoji or '_'."
       expect(page).to have_button 'Save changes'
     end
 
-    scenario 'shows a successful notice when the project is updated' do
+    it 'shows a successful notice when the project is updated' do
       visit edit_namespace_project_path(project.namespace, project)
-
       fill_in 'project_name_edit', with: 'hello world'
-
       click_button 'Save changes'
-
       expect(page).to have_content "Project 'hello world' was successfully updated."
     end
   end
 
-  describe 'Rename repository' do
-    it 'shows errors for invalid project path/name' do
-      visit edit_namespace_project_path(project.namespace, project)
-
-      fill_in 'project_name', with: 'foo&bar'
-      fill_in 'Path', with: 'foo&bar'
+  describe 'Rename repository section' do
+    context 'with invalid characters' do
+      it 'shows errors for invalid project path/name' do
+        rename_project(project, name: 'foo&bar', path: 'foo&bar')
+        expect(page).to have_field 'Project name', with: 'foo&bar'
+        expect(page).to have_field 'Path', with: 'foo&bar'
+        expect(page).to have_content "Name can contain only letters, digits, emojis, '_', '.', dash, space. It must start with letter, digit, emoji or '_'."
+        expect(page).to have_content "Path can contain only letters, digits, '_', '-' and '.'. Cannot start with '-', end in '.git' or end in '.atom'"
+      end
+    end
 
-      click_button 'Rename project'
+    context 'when changing project name' do
+      it 'renames the repository' do
+        rename_project(project, name: 'bar')
+        expect(find('h1.title')).to have_content(project.name)
+      end
+
+      context 'with emojis' do
+        it 'shows error for invalid project name' do
+          rename_project(project, name: '🚀 foo bar ☁️')
+          expect(page).to have_field 'Project name', with: '🚀 foo bar ☁️'
+          expect(page).not_to have_content "Name can contain only letters, digits, emojis '_', '.', dash and space. It must start with letter, digit, emoji or '_'."
+        end
+      end
+    end
 
-      expect(page).to have_field 'Project name', with: 'foo&bar'
-      expect(page).to have_field 'Path', with: 'foo&bar'
-      expect(page).to have_content "Name can contain only letters, digits, emojis, '_', '.', dash, space. It must start with letter, digit, emoji or '_'."
-      expect(page).to have_content "Path can contain only letters, digits, '_', '-' and '.'. Cannot start with '-', end in '.git' or end in '.atom'"
+    context 'when changing project path' do
+      # Not using empty project because we need a repo to exist
+      let(:project) { create(:project, namespace: user.namespace, name: 'gitlabhq') }
+
+      specify 'the project is accessible via the new path' do
+        rename_project(project, path: 'bar')
+        new_path = namespace_project_path(project.namespace, 'bar')
+        visit new_path
+        expect(current_path).to eq(new_path)
+        expect(find('h1.title')).to have_content(project.name)
+      end
+
+      specify 'the project is accessible via a redirect from the old path' do
+        old_path = namespace_project_path(project.namespace, project)
+        rename_project(project, path: 'bar')
+        new_path = namespace_project_path(project.namespace, 'bar')
+        visit old_path
+        expect(current_path).to eq(new_path)
+        expect(find('h1.title')).to have_content(project.name)
+      end
+
+      context 'and a new project is added with the same path' do
+        it 'overrides the redirect' do
+          old_path = namespace_project_path(project.namespace, project)
+          rename_project(project, path: 'bar')
+          new_project = create(:empty_project, namespace: user.namespace, path: 'gitlabhq', name: 'quz')
+          visit old_path
+          expect(current_path).to eq(old_path)
+          expect(find('h1.title')).to have_content(new_project.name)
+        end
+      end
     end
   end
 
-  describe 'Rename repository name with emojis' do
-    it 'shows error for invalid project name' do
-      visit edit_namespace_project_path(project.namespace, project)
+  describe 'Transfer project section', js: true do
+    # Not using empty project because we need a repo to exist
+    let(:project) { create(:project, namespace: user.namespace, name: 'gitlabhq') }
+    let(:group) { create(:group) }
+    before do
+      group.add_owner(user)
+    end
 
-      fill_in 'project_name', with: '🚀 foo bar ☁️'
+    specify 'the project is accessible via the new path' do
+      transfer_project(project, group)
+      new_path = namespace_project_path(group, project)
+      visit new_path
+      expect(current_path).to eq(new_path)
+      expect(find('h1.title')).to have_content(project.name)
+    end
 
-      click_button 'Rename project'
+    specify 'the project is accessible via a redirect from the old path' do
+      old_path = namespace_project_path(project.namespace, project)
+      transfer_project(project, group)
+      new_path = namespace_project_path(group, project)
+      visit old_path
+      expect(current_path).to eq(new_path)
+      expect(find('h1.title')).to have_content(project.name)
+    end
 
-      expect(page).to have_field 'Project name', with: '🚀 foo bar ☁️'
-      expect(page).not_to have_content "Name can contain only letters, digits, emojis '_', '.', dash and space. It must start with letter, digit, emoji or '_'."
+    context 'and a new project is added with the same path' do
+      it 'overrides the redirect' do
+        old_path = namespace_project_path(project.namespace, project)
+        transfer_project(project, group)
+        new_project = create(:empty_project, namespace: user.namespace, path: 'gitlabhq', name: 'quz')
+        visit old_path
+        expect(current_path).to eq(old_path)
+        expect(find('h1.title')).to have_content(new_project.name)
+      end
     end
   end
 end
+
+def rename_project(project, name: nil, path: nil)
+  visit edit_namespace_project_path(project.namespace, project)
+  fill_in('project_name', with: name) if name
+  fill_in('Path', with: path) if path
+  click_button('Rename project')
+  wait_for_edit_project_page_reload
+  project.reload
+end
+
+def transfer_project(project, namespace)
+  visit edit_namespace_project_path(project.namespace, project)
+  select2(namespace.id, from: '#new_namespace_id')
+  click_button('Transfer project')
+  confirm_transfer_modal
+  wait_for_edit_project_page_reload
+  project.reload
+end
+
+def confirm_transfer_modal
+  fill_in('confirm_name_input', with: project.path)
+  click_button 'Confirm'
+end
+
+def wait_for_edit_project_page_reload
+  expect(find('.project-edit-container')).to have_content('Rename repository')
+end
-- 
GitLab


From 7d02bcd2e0165a90a9f2c1edb34b064ff76afd69 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Mon, 1 May 2017 13:46:30 -0700
Subject: [PATCH 323/363] Redirect from redirect routes to canonical routes

---
 app/controllers/application_controller.rb     |   2 +-
 app/controllers/concerns/routable_actions.rb  |  11 ++
 .../groups/application_controller.rb          |  21 +-
 app/controllers/groups_controller.rb          |   2 +-
 .../projects/application_controller.rb        |  14 +-
 app/controllers/users_controller.rb           |  12 +-
 app/models/concerns/routable.rb               |  24 ++-
 app/models/redirect_route.rb                  |  10 +
 app/models/user.rb                            |   5 +
 .../20170427215854_create_redirect_routes.rb  |  15 ++
 db/schema.rb                                  |   8 +
 lib/constraints/group_url_constrainer.rb      |   2 +-
 lib/constraints/project_url_constrainer.rb    |   2 +-
 lib/constraints/user_url_constrainer.rb       |   2 +-
 .../application_controller_spec.rb            |   5 +-
 spec/controllers/groups_controller_spec.rb    |  92 ++++++++-
 spec/controllers/projects_controller_spec.rb  |  84 +++++++-
 spec/controllers/users_controller_spec.rb     | 181 +++++++++++++++++-
 spec/features/groups/group_settings_spec.rb   |   3 +
 spec/features/profiles/account_spec.rb        |   5 +-
 .../projects/features_visibility_spec.rb      |  37 ++--
 .../projects/project_settings_spec.rb         |  10 +-
 .../constraints/group_url_constrainer_spec.rb |  32 +++-
 .../project_url_constrainer_spec.rb           |  21 +-
 .../constraints/user_url_constrainer_spec.rb  |  21 +-
 spec/lib/gitlab/import_export/all_models.yml  |   1 +
 spec/models/concerns/routable_spec.rb         |  52 ++++-
 spec/models/redirect_route_spec.rb            |  16 ++
 spec/models/user_spec.rb                      |  59 ++++++
 spec/routing/project_routing_spec.rb          |   6 +-
 30 files changed, 668 insertions(+), 87 deletions(-)
 create mode 100644 app/controllers/concerns/routable_actions.rb
 create mode 100644 app/models/redirect_route.rb
 create mode 100644 db/migrate/20170427215854_create_redirect_routes.rb
 create mode 100644 spec/models/redirect_route_spec.rb

diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index d2c13da691741..65a1f640a764d 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -58,7 +58,7 @@ def route_not_found
     if current_user
       not_found
     else
-      redirect_to new_user_session_path
+      authenticate_user!
     end
   end
 
diff --git a/app/controllers/concerns/routable_actions.rb b/app/controllers/concerns/routable_actions.rb
new file mode 100644
index 0000000000000..6f16377a156f6
--- /dev/null
+++ b/app/controllers/concerns/routable_actions.rb
@@ -0,0 +1,11 @@
+module RoutableActions
+  extend ActiveSupport::Concern
+
+  def ensure_canonical_path(routable, requested_path)
+    return unless request.get?
+
+    if routable.full_path != requested_path
+      redirect_to request.original_url.sub(requested_path, routable.full_path)
+    end
+  end
+end
diff --git a/app/controllers/groups/application_controller.rb b/app/controllers/groups/application_controller.rb
index 29ffaeb19c166..209d8b1a08afe 100644
--- a/app/controllers/groups/application_controller.rb
+++ b/app/controllers/groups/application_controller.rb
@@ -1,4 +1,6 @@
 class Groups::ApplicationController < ApplicationController
+  include RoutableActions
+
   layout 'group'
 
   skip_before_action :authenticate_user!
@@ -8,18 +10,15 @@ class Groups::ApplicationController < ApplicationController
 
   def group
     unless @group
-      id = params[:group_id] || params[:id]
-      @group = Group.find_by_full_path(id)
-      @group_merge_requests = MergeRequestsFinder.new(current_user, group_id: @group.id).execute
+      given_path = params[:group_id] || params[:id]
+      @group = Group.find_by_full_path(given_path, follow_redirects: request.get?)
 
-      unless @group && can?(current_user, :read_group, @group)
+      if @group && can?(current_user, :read_group, @group)
+        ensure_canonical_path(@group, given_path)
+      else
         @group = nil
 
-        if current_user.nil?
-          authenticate_user!
-        else
-          render_404
-        end
+        route_not_found
       end
     end
 
@@ -30,6 +29,10 @@ def group_projects
     @projects ||= GroupProjectsFinder.new(group: group, current_user: current_user).execute
   end
 
+  def group_merge_requests
+    @group_merge_requests = MergeRequestsFinder.new(current_user, group_id: @group.id).execute
+  end
+
   def authorize_admin_group!
     unless can?(current_user, :admin_group, group)
       return render_404
diff --git a/app/controllers/groups_controller.rb b/app/controllers/groups_controller.rb
index 593001e6396dc..46c3ff1069409 100644
--- a/app/controllers/groups_controller.rb
+++ b/app/controllers/groups_controller.rb
@@ -12,8 +12,8 @@ class GroupsController < Groups::ApplicationController
   before_action :authorize_admin_group!, only: [:edit, :update, :destroy, :projects]
   before_action :authorize_create_group!, only: [:new, :create]
 
-  # Load group projects
   before_action :group_projects, only: [:projects, :activity, :issues, :merge_requests]
+  before_action :group_merge_requests, only: [:merge_requests]
   before_action :event_filter, only: [:activity]
 
   before_action :user_actions, only: [:show, :subgroups]
diff --git a/app/controllers/projects/application_controller.rb b/app/controllers/projects/application_controller.rb
index 89f1128ec36ad..dbdf68776f1d7 100644
--- a/app/controllers/projects/application_controller.rb
+++ b/app/controllers/projects/application_controller.rb
@@ -1,4 +1,6 @@
 class Projects::ApplicationController < ApplicationController
+  include RoutableActions
+
   skip_before_action :authenticate_user!
   before_action :project
   before_action :repository
@@ -24,20 +26,14 @@ def project
       end
 
       project_path = "#{namespace}/#{id}"
-      @project = Project.find_by_full_path(project_path)
+      @project = Project.find_by_full_path(project_path, follow_redirects: request.get?)
 
       if can?(current_user, :read_project, @project) && !@project.pending_delete?
-        if @project.path_with_namespace != project_path
-          redirect_to request.original_url.gsub(project_path, @project.path_with_namespace)
-        end
+        ensure_canonical_path(@project, project_path)
       else
         @project = nil
 
-        if current_user.nil?
-          authenticate_user!
-        else
-          render_404
-        end
+        route_not_found
       end
     end
 
diff --git a/app/controllers/users_controller.rb b/app/controllers/users_controller.rb
index a452bbba422bb..6b6b3b20a8ddf 100644
--- a/app/controllers/users_controller.rb
+++ b/app/controllers/users_controller.rb
@@ -1,7 +1,9 @@
 class UsersController < ApplicationController
+  include RoutableActions
+
   skip_before_action :authenticate_user!
   before_action :user, except: [:exists]
-  before_action :authorize_read_user!, only: [:show]
+  before_action :authorize_read_user!, except: [:exists]
 
   def show
     respond_to do |format|
@@ -92,11 +94,15 @@ def exists
   private
 
   def authorize_read_user!
-    render_404 unless can?(current_user, :read_user, user)
+    if can?(current_user, :read_user, user)
+      ensure_canonical_path(user.namespace, params[:username])
+    else
+      render_404
+    end
   end
 
   def user
-    @user ||= User.find_by_username!(params[:username])
+    @user ||= User.find_by_full_path(params[:username], follow_redirects: true)
   end
 
   def contributed_projects
diff --git a/app/models/concerns/routable.rb b/app/models/concerns/routable.rb
index b28e05d0c280d..e351dbb45ddb8 100644
--- a/app/models/concerns/routable.rb
+++ b/app/models/concerns/routable.rb
@@ -5,6 +5,7 @@ module Routable
 
   included do
     has_one :route, as: :source, autosave: true, dependent: :destroy
+    has_many :redirect_routes, as: :source, autosave: true, dependent: :destroy
 
     validates_associated :route
     validates :route, presence: true
@@ -26,16 +27,31 @@ module Routable
     #     Klass.find_by_full_path('gitlab-org/gitlab-ce')
     #
     # Returns a single object, or nil.
-    def find_by_full_path(path)
+    def find_by_full_path(path, follow_redirects: false)
       # On MySQL we want to ensure the ORDER BY uses a case-sensitive match so
       # any literal matches come first, for this we have to use "BINARY".
       # Without this there's still no guarantee in what order MySQL will return
       # rows.
+      #
+      # Why do we do this?
+      #
+      # Even though we have Rails validation on Route for unique paths
+      # (case-insensitive), there are old projects in our DB (and possibly
+      # clients' DBs) that have the same path with different cases.
+      # See https://gitlab.com/gitlab-org/gitlab-ce/issues/18603. Also note that
+      # our unique index is case-sensitive in Postgres.
       binary = Gitlab::Database.mysql? ? 'BINARY' : ''
-
       order_sql = "(CASE WHEN #{binary} routes.path = #{connection.quote(path)} THEN 0 ELSE 1 END)"
-
-      where_full_path_in([path]).reorder(order_sql).take
+      found = where_full_path_in([path]).reorder(order_sql).take
+      return found if found
+
+      if follow_redirects
+        if Gitlab::Database.postgresql?
+          joins(:redirect_routes).find_by("LOWER(redirect_routes.path) = LOWER(?)", path)
+        else
+          joins(:redirect_routes).find_by(path: path)
+        end
+      end
     end
 
     # Builds a relation to find multiple objects by their full paths.
diff --git a/app/models/redirect_route.rb b/app/models/redirect_route.rb
new file mode 100644
index 0000000000000..e36ca988701f1
--- /dev/null
+++ b/app/models/redirect_route.rb
@@ -0,0 +1,10 @@
+class RedirectRoute < ActiveRecord::Base
+  belongs_to :source, polymorphic: true
+
+  validates :source, presence: true
+
+  validates :path,
+    length: { within: 1..255 },
+    presence: true,
+    uniqueness: { case_sensitive: false }
+end
diff --git a/app/models/user.rb b/app/models/user.rb
index 43c5fdc038d47..c6354b989ade2 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -333,6 +333,11 @@ def find_by_ssh_key_id(key_id)
       find_by(id: Key.unscoped.select(:user_id).where(id: key_id))
     end
 
+    def find_by_full_path(path, follow_redirects: false)
+      namespace = Namespace.find_by_full_path(path, follow_redirects: follow_redirects)
+      namespace.owner if namespace && namespace.owner
+    end
+
     def reference_prefix
       '@'
     end
diff --git a/db/migrate/20170427215854_create_redirect_routes.rb b/db/migrate/20170427215854_create_redirect_routes.rb
new file mode 100644
index 0000000000000..57b79a0267fd3
--- /dev/null
+++ b/db/migrate/20170427215854_create_redirect_routes.rb
@@ -0,0 +1,15 @@
+class CreateRedirectRoutes < ActiveRecord::Migration
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  def change
+    create_table :redirect_routes do |t|
+      t.integer :source_id, null: false
+      t.string :source_type, null: false
+      t.string :path, null: false
+
+      t.timestamps null: false
+    end
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 588e2082ae038..3a8970b2235e8 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -1040,6 +1040,14 @@
 
   add_index "protected_tags", ["project_id"], name: "index_protected_tags_on_project_id", using: :btree
 
+  create_table "redirect_routes", force: :cascade do |t|
+    t.integer "source_id", null: false
+    t.string "source_type", null: false
+    t.string "path", null: false
+    t.datetime "created_at", null: false
+    t.datetime "updated_at", null: false
+  end
+
   create_table "releases", force: :cascade do |t|
     t.string "tag"
     t.text "description"
diff --git a/lib/constraints/group_url_constrainer.rb b/lib/constraints/group_url_constrainer.rb
index 1501f64d537af..5f379756c11ee 100644
--- a/lib/constraints/group_url_constrainer.rb
+++ b/lib/constraints/group_url_constrainer.rb
@@ -4,6 +4,6 @@ def matches?(request)
 
     return false unless DynamicPathValidator.valid?(id)
 
-    Group.find_by_full_path(id).present?
+    Group.find_by_full_path(id, follow_redirects: request.get?).present?
   end
 end
diff --git a/lib/constraints/project_url_constrainer.rb b/lib/constraints/project_url_constrainer.rb
index d0ce2caffffc4..6f542f63f9807 100644
--- a/lib/constraints/project_url_constrainer.rb
+++ b/lib/constraints/project_url_constrainer.rb
@@ -6,6 +6,6 @@ def matches?(request)
 
     return false unless DynamicPathValidator.valid?(full_path)
 
-    Project.find_by_full_path(full_path).present?
+    Project.find_by_full_path(full_path, follow_redirects: request.get?).present?
   end
 end
diff --git a/lib/constraints/user_url_constrainer.rb b/lib/constraints/user_url_constrainer.rb
index 9ab5bcb12ffc3..28159dc0decee 100644
--- a/lib/constraints/user_url_constrainer.rb
+++ b/lib/constraints/user_url_constrainer.rb
@@ -1,5 +1,5 @@
 class UserUrlConstrainer
   def matches?(request)
-    User.find_by_username(request.params[:username]).present?
+    User.find_by_full_path(request.params[:username], follow_redirects: request.get?).present?
   end
 end
diff --git a/spec/controllers/application_controller_spec.rb b/spec/controllers/application_controller_spec.rb
index 1bf0533ca2487..d40aae04fc3aa 100644
--- a/spec/controllers/application_controller_spec.rb
+++ b/spec/controllers/application_controller_spec.rb
@@ -106,10 +106,9 @@ def index
       controller.send(:route_not_found)
     end
 
-    it 'does redirect to login page if not authenticated' do
+    it 'does redirect to login page via authenticate_user! if not authenticated' do
       allow(controller).to receive(:current_user).and_return(nil)
-      expect(controller).to receive(:redirect_to)
-      expect(controller).to receive(:new_user_session_path)
+      expect(controller).to receive(:authenticate_user!)
       controller.send(:route_not_found)
     end
   end
diff --git a/spec/controllers/groups_controller_spec.rb b/spec/controllers/groups_controller_spec.rb
index cad82a34fb0e9..3398878f33058 100644
--- a/spec/controllers/groups_controller_spec.rb
+++ b/spec/controllers/groups_controller_spec.rb
@@ -49,6 +49,24 @@
         expect(assigns(:issues)).to eq [issue_2, issue_1]
       end
     end
+
+    context 'when requesting the canonical path with different casing' do
+      it 'redirects to the correct casing' do
+        get :issues, id: group.to_param.upcase
+
+        expect(response).to redirect_to(issues_group_path(group.to_param))
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let(:redirect_route) { group.redirect_routes.create(path: 'old-path') }
+
+      it 'redirects to the canonical path' do
+        get :issues, id: redirect_route.path
+
+        expect(response).to redirect_to(issues_group_path(group.to_param))
+      end
+    end
   end
 
   describe 'GET #merge_requests' do
@@ -74,6 +92,24 @@
         expect(assigns(:merge_requests)).to eq [merge_request_2, merge_request_1]
       end
     end
+
+    context 'when requesting the canonical path with different casing' do
+      it 'redirects to the correct casing' do
+        get :merge_requests, id: group.to_param.upcase
+
+        expect(response).to redirect_to(merge_requests_group_path(group.to_param))
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let(:redirect_route) { group.redirect_routes.create(path: 'old-path') }
+
+      it 'redirects to the canonical path' do
+        get :merge_requests, id: redirect_route.path
+
+        expect(response).to redirect_to(merge_requests_group_path(group.to_param))
+      end
+    end
   end
 
   describe 'DELETE #destroy' do
@@ -81,7 +117,7 @@
       it 'returns 404' do
         sign_in(create(:user))
 
-        delete :destroy, id: group.path
+        delete :destroy, id: group.to_param
 
         expect(response.status).to eq(404)
       end
@@ -94,15 +130,39 @@
 
       it 'schedules a group destroy' do
         Sidekiq::Testing.fake! do
-          expect { delete :destroy, id: group.path }.to change(GroupDestroyWorker.jobs, :size).by(1)
+          expect { delete :destroy, id: group.to_param }.to change(GroupDestroyWorker.jobs, :size).by(1)
         end
       end
 
       it 'redirects to the root path' do
-        delete :destroy, id: group.path
+        delete :destroy, id: group.to_param
 
         expect(response).to redirect_to(root_path)
       end
+
+      context 'when requesting the canonical path with different casing' do
+        it 'does not 404' do
+          delete :destroy, id: group.to_param.upcase
+
+          expect(response).to_not have_http_status(404)
+        end
+
+        it 'does not redirect to the correct casing' do
+          delete :destroy, id: group.to_param.upcase
+
+          expect(response).to_not redirect_to(group_path(group.to_param))
+        end
+      end
+
+      context 'when requesting a redirected path' do
+        let(:redirect_route) { group.redirect_routes.create(path: 'old-path') }
+
+        it 'returns not found' do
+          delete :destroy, id: redirect_route.path
+
+          expect(response).to have_http_status(404)
+        end
+      end
     end
   end
 
@@ -111,7 +171,7 @@
       sign_in(user)
     end
 
-    it 'updates the path succesfully' do
+    it 'updates the path successfully' do
       post :update, id: group.to_param, group: { path: 'new_path' }
 
       expect(response).to have_http_status(302)
@@ -125,5 +185,29 @@
       expect(assigns(:group).errors).not_to be_empty
       expect(assigns(:group).path).not_to eq('new_path')
     end
+
+    context 'when requesting the canonical path with different casing' do
+      it 'does not 404' do
+        post :update, id: group.to_param.upcase, group: { path: 'new_path' }
+
+        expect(response).to_not have_http_status(404)
+      end
+
+      it 'does not redirect to the correct casing' do
+        post :update, id: group.to_param.upcase, group: { path: 'new_path' }
+
+        expect(response).to_not redirect_to(group_path(group.to_param))
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let(:redirect_route) { group.redirect_routes.create(path: 'old-path') }
+
+      it 'returns not found' do
+        post :update, id: redirect_route.path, group: { path: 'new_path' }
+
+        expect(response).to have_http_status(404)
+      end
+    end
   end
 end
diff --git a/spec/controllers/projects_controller_spec.rb b/spec/controllers/projects_controller_spec.rb
index eafc2154568d1..1b0dd7c63696d 100644
--- a/spec/controllers/projects_controller_spec.rb
+++ b/spec/controllers/projects_controller_spec.rb
@@ -218,19 +218,32 @@
         expect(response).to redirect_to(namespace_project_path)
       end
     end
+
+    context 'when requesting a redirected path' do
+      let!(:redirect_route) { public_project.redirect_routes.create!(path: "foo/bar") }
+
+      it 'redirects to the canonical path' do
+        get :show, namespace_id: 'foo', id: 'bar'
+
+        expect(response).to redirect_to(public_project)
+      end
+    end
   end
 
   describe "#update" do
     render_views
 
     let(:admin) { create(:admin) }
+    let(:project) { create(:project, :repository) }
+    let(:new_path) { 'renamed_path' }
+    let(:project_params) { { path: new_path } }
+
+    before do
+      sign_in(admin)
+    end
 
     it "sets the repository to the right path after a rename" do
-      project = create(:project, :repository)
-      new_path = 'renamed_path'
-      project_params = { path: new_path }
       controller.instance_variable_set(:@project, project)
-      sign_in(admin)
 
       put :update,
           namespace_id: project.namespace,
@@ -241,6 +254,34 @@
       expect(assigns(:repository).path).to eq(project.repository.path)
       expect(response).to have_http_status(302)
     end
+
+    context 'when requesting the canonical path' do
+      it "is case-insensitive" do
+        controller.instance_variable_set(:@project, project)
+
+        put :update,
+            namespace_id: 'FOo',
+            id: 'baR',
+            project: project_params
+
+        expect(project.repository.path).to include(new_path)
+        expect(assigns(:repository).path).to eq(project.repository.path)
+        expect(response).to have_http_status(302)
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let!(:redirect_route) { project.redirect_routes.create!(path: "foo/bar") }
+
+      it 'returns not found' do
+        put :update,
+            namespace_id: 'foo',
+            id: 'bar',
+            project: project_params
+
+        expect(response).to have_http_status(404)
+      end
+    end
   end
 
   describe "#destroy" do
@@ -276,6 +317,31 @@
         expect(merge_request.reload.state).to eq('closed')
       end
     end
+
+    context 'when requesting the canonical path' do
+      it "is case-insensitive" do
+        controller.instance_variable_set(:@project, project)
+        sign_in(admin)
+
+        orig_id = project.id
+        delete :destroy, namespace_id: project.namespace, id: project.path.upcase
+
+        expect { Project.find(orig_id) }.to raise_error(ActiveRecord::RecordNotFound)
+        expect(response).to have_http_status(302)
+        expect(response).to redirect_to(dashboard_projects_path)
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let!(:redirect_route) { project.redirect_routes.create!(path: "foo/bar") }
+
+      it 'returns not found' do
+        sign_in(admin)
+        delete :destroy, namespace_id: 'foo', id: 'bar'
+
+        expect(response).to have_http_status(404)
+      end
+    end
   end
 
   describe 'PUT #new_issue_address' do
@@ -397,6 +463,16 @@
       expect(parsed_body["Tags"]).to include("v1.0.0")
       expect(parsed_body["Commits"]).to include("123456")
     end
+
+    context 'when requesting a redirected path' do
+      let!(:redirect_route) { public_project.redirect_routes.create!(path: "foo/bar") }
+
+      it 'redirects to the canonical path' do
+        get :refs, namespace_id: 'foo', id: 'bar'
+
+        expect(response).to redirect_to(refs_namespace_project_path(namespace_id: public_project.namespace, id: public_project))
+      end
+    end
   end
 
   describe 'POST #preview_markdown' do
diff --git a/spec/controllers/users_controller_spec.rb b/spec/controllers/users_controller_spec.rb
index bbe9aaf737fa8..73fd5b6f90ded 100644
--- a/spec/controllers/users_controller_spec.rb
+++ b/spec/controllers/users_controller_spec.rb
@@ -4,15 +4,6 @@
   let(:user) { create(:user) }
 
   describe 'GET #show' do
-    it 'is case-insensitive' do
-      user = create(:user, username: 'CamelCaseUser')
-      sign_in(user)
-
-      get :show, username: user.username.downcase
-
-      expect(response).to be_success
-    end
-
     context 'with rendered views' do
       render_views
 
@@ -61,6 +52,38 @@
         end
       end
     end
+
+    context 'when requesting the canonical path' do
+      let(:user) { create(:user, username: 'CamelCaseUser') }
+
+      before { sign_in(user) }
+
+      context 'with exactly matching casing' do
+        it 'responds with success' do
+          get :show, username: user.username
+
+          expect(response).to be_success
+        end
+      end
+
+      context 'with different casing' do
+        it 'redirects to the correct casing' do
+          get :show, username: user.username.downcase
+
+          expect(response).to redirect_to(user)
+        end
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let(:redirect_route) { user.namespace.redirect_routes.create(path: 'old-username') }
+
+      it 'redirects to the canonical path' do
+        get :show, username: redirect_route.path
+
+        expect(response).to redirect_to(user)
+      end
+    end
   end
 
   describe 'GET #calendar' do
@@ -88,11 +111,43 @@
         expect(assigns(:contributions_calendar).projects.count).to eq(2)
       end
     end
+
+    context 'when requesting the canonical path' do
+      let(:user) { create(:user, username: 'CamelCaseUser') }
+
+      before { sign_in(user) }
+
+      context 'with exactly matching casing' do
+        it 'responds with success' do
+          get :calendar, username: user.username
+
+          expect(response).to be_success
+        end
+      end
+
+      context 'with different casing' do
+        it 'redirects to the correct casing' do
+          get :calendar, username: user.username.downcase
+
+          expect(response).to redirect_to(user_calendar_path(user))
+        end
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let(:redirect_route) { user.namespace.redirect_routes.create(path: 'old-username') }
+
+      it 'redirects to the canonical path' do
+        get :calendar, username: redirect_route.path
+
+        expect(response).to redirect_to(user_calendar_path(user))
+      end
+    end
   end
 
   describe 'GET #calendar_activities' do
     let!(:project) { create(:empty_project) }
-    let!(:user) { create(:user) }
+    let(:user) { create(:user) }
 
     before do
       allow_any_instance_of(User).to receive(:contributed_projects_ids).and_return([project.id])
@@ -110,6 +165,36 @@
       get :calendar_activities, username: user.username
       expect(response).to render_template('calendar_activities')
     end
+
+    context 'when requesting the canonical path' do
+      let(:user) { create(:user, username: 'CamelCaseUser') }
+
+      context 'with exactly matching casing' do
+        it 'responds with success' do
+          get :calendar_activities, username: user.username
+
+          expect(response).to be_success
+        end
+      end
+
+      context 'with different casing' do
+        it 'redirects to the correct casing' do
+          get :calendar_activities, username: user.username.downcase
+
+          expect(response).to redirect_to(user_calendar_activities_path(user))
+        end
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let(:redirect_route) { user.namespace.redirect_routes.create(path: 'old-username') }
+
+      it 'redirects to the canonical path' do
+        get :calendar_activities, username: redirect_route.path
+
+        expect(response).to redirect_to(user_calendar_activities_path(user))
+      end
+    end
   end
 
   describe 'GET #snippets' do
@@ -132,5 +217,81 @@
         expect(JSON.parse(response.body)).to have_key('html')
       end
     end
+
+    context 'when requesting the canonical path' do
+      let(:user) { create(:user, username: 'CamelCaseUser') }
+
+      context 'with exactly matching casing' do
+        it 'responds with success' do
+          get :snippets, username: user.username
+
+          expect(response).to be_success
+        end
+      end
+
+      context 'with different casing' do
+        it 'redirects to the correct casing' do
+          get :snippets, username: user.username.downcase
+
+          expect(response).to redirect_to(user_snippets_path(user))
+        end
+      end
+    end
+
+    context 'when requesting a redirected path' do
+      let(:redirect_route) { user.namespace.redirect_routes.create(path: 'old-username') }
+
+      it 'redirects to the canonical path' do
+        get :snippets, username: redirect_route.path
+
+        expect(response).to redirect_to(user_snippets_path(user))
+      end
+    end
+  end
+
+  describe 'GET #exists' do
+    before do
+      sign_in(user)
+    end
+
+    context 'when user exists' do
+      it 'returns JSON indicating the user exists' do
+        get :exists, username: user.username
+
+        expected_json = { exists: true }.to_json
+        expect(response.body).to eq(expected_json)
+      end
+
+      context 'when the casing is different' do
+        let(:user) { create(:user, username: 'CamelCaseUser') }
+
+        it 'returns JSON indicating the user exists' do
+          get :exists, username: user.username.downcase
+
+          expected_json = { exists: true }.to_json
+          expect(response.body).to eq(expected_json)
+        end
+      end
+    end
+
+    context 'when the user does not exist' do
+      it 'returns JSON indicating the user does not exist' do
+        get :exists, username: 'foo'
+
+        expected_json = { exists: false }.to_json
+        expect(response.body).to eq(expected_json)
+      end
+
+      context 'when a user changed their username' do
+        let(:redirect_route) { user.namespace.redirect_routes.create(path: 'old-username') }
+
+        it 'returns JSON indicating a user by that username does not exist' do
+          get :exists, username: 'old-username'
+
+          expected_json = { exists: false }.to_json
+          expect(response.body).to eq(expected_json)
+        end
+      end
+    end
   end
 end
diff --git a/spec/features/groups/group_settings_spec.rb b/spec/features/groups/group_settings_spec.rb
index f18f2f2310f03..cc25db4ad602f 100644
--- a/spec/features/groups/group_settings_spec.rb
+++ b/spec/features/groups/group_settings_spec.rb
@@ -52,6 +52,9 @@
       given!(:project) { create(:project, group: group, path: 'project') }
       given(:old_project_full_path) { "/#{group.path}/#{project.path}" }
       given(:new_project_full_path) { "/#{new_group_path}/#{project.path}" }
+      
+      before(:context) { TestEnv.clean_test_path }
+      after(:example) { TestEnv.clean_test_path }
 
       scenario 'the project is accessible via the new path' do
         update_path(new_group_path)
diff --git a/spec/features/profiles/account_spec.rb b/spec/features/profiles/account_spec.rb
index b3902a6b37f7b..05a7587f8d44b 100644
--- a/spec/features/profiles/account_spec.rb
+++ b/spec/features/profiles/account_spec.rb
@@ -31,9 +31,8 @@
       given(:new_project_path) { "/#{new_username}/#{project.path}" }
       given(:old_project_path) { "/#{user.username}/#{project.path}" }
 
-      after do
-        TestEnv.clean_test_path
-      end
+      before(:context) { TestEnv.clean_test_path }
+      after(:example) { TestEnv.clean_test_path }
 
       scenario 'the project is accessible via the new path' do
         update_username(new_username)
diff --git a/spec/features/projects/features_visibility_spec.rb b/spec/features/projects/features_visibility_spec.rb
index b080a8d500e52..a6ca0f4131a0a 100644
--- a/spec/features/projects/features_visibility_spec.rb
+++ b/spec/features/projects/features_visibility_spec.rb
@@ -68,20 +68,23 @@
   end
 
   describe 'project features visibility pages' do
-    before do
-      @tools =
-        {
-          builds: namespace_project_pipelines_path(project.namespace, project),
-          issues: namespace_project_issues_path(project.namespace, project),
-          wiki: namespace_project_wiki_path(project.namespace, project, :home),
-          snippets: namespace_project_snippets_path(project.namespace, project),
-          merge_requests: namespace_project_merge_requests_path(project.namespace, project),
-        }
-    end
+    let(:tools) {
+      {
+        builds: namespace_project_pipelines_path(project.namespace, project),
+        issues: namespace_project_issues_path(project.namespace, project),
+        wiki: namespace_project_wiki_path(project.namespace, project, :home),
+        snippets: namespace_project_snippets_path(project.namespace, project),
+        merge_requests: namespace_project_merge_requests_path(project.namespace, project),
+      }
+    }
 
     context 'normal user' do
+      before do
+        login_as(member)
+      end
+
       it 'renders 200 if tool is enabled' do
-        @tools.each do |method_name, url|
+        tools.each do |method_name, url|
           project.project_feature.update_attribute("#{method_name}_access_level", ProjectFeature::ENABLED)
           visit url
           expect(page.status_code).to eq(200)
@@ -89,7 +92,7 @@
       end
 
       it 'renders 404 if feature is disabled' do
-        @tools.each do |method_name, url|
+        tools.each do |method_name, url|
           project.project_feature.update_attribute("#{method_name}_access_level", ProjectFeature::DISABLED)
           visit url
           expect(page.status_code).to eq(404)
@@ -99,21 +102,21 @@
       it 'renders 404 if feature is enabled only for team members' do
         project.team.truncate
 
-        @tools.each do |method_name, url|
+        tools.each do |method_name, url|
           project.project_feature.update_attribute("#{method_name}_access_level", ProjectFeature::PRIVATE)
           visit url
           expect(page.status_code).to eq(404)
         end
       end
 
-      it 'renders 200 if users is member of group' do
+      it 'renders 200 if user is member of group' do
         group = create(:group)
         project.group = group
         project.save
 
         group.add_owner(member)
 
-        @tools.each do |method_name, url|
+        tools.each do |method_name, url|
           project.project_feature.update_attribute("#{method_name}_access_level", ProjectFeature::PRIVATE)
           visit url
           expect(page.status_code).to eq(200)
@@ -128,7 +131,7 @@
       end
 
       it 'renders 404 if feature is disabled' do
-        @tools.each do |method_name, url|
+        tools.each do |method_name, url|
           project.project_feature.update_attribute("#{method_name}_access_level", ProjectFeature::DISABLED)
           visit url
           expect(page.status_code).to eq(404)
@@ -138,7 +141,7 @@
       it 'renders 200 if feature is enabled only for team members' do
         project.team.truncate
 
-        @tools.each do |method_name, url|
+        tools.each do |method_name, url|
           project.project_feature.update_attribute("#{method_name}_access_level", ProjectFeature::PRIVATE)
           visit url
           expect(page.status_code).to eq(200)
diff --git a/spec/features/projects/project_settings_spec.rb b/spec/features/projects/project_settings_spec.rb
index 364cc8ef0f29a..c0311e73ef5e7 100644
--- a/spec/features/projects/project_settings_spec.rb
+++ b/spec/features/projects/project_settings_spec.rb
@@ -58,6 +58,9 @@
       # Not using empty project because we need a repo to exist
       let(:project) { create(:project, namespace: user.namespace, name: 'gitlabhq') }
 
+      before(:context) { TestEnv.clean_test_path }
+      after(:example) { TestEnv.clean_test_path }
+
       specify 'the project is accessible via the new path' do
         rename_project(project, path: 'bar')
         new_path = namespace_project_path(project.namespace, 'bar')
@@ -92,9 +95,10 @@
     # Not using empty project because we need a repo to exist
     let(:project) { create(:project, namespace: user.namespace, name: 'gitlabhq') }
     let(:group) { create(:group) }
-    before do
-      group.add_owner(user)
-    end
+
+    before(:context) { TestEnv.clean_test_path }
+    before(:example) { group.add_owner(user) }
+    after(:example) { TestEnv.clean_test_path }
 
     specify 'the project is accessible via the new path' do
       transfer_project(project, group)
diff --git a/spec/lib/constraints/group_url_constrainer_spec.rb b/spec/lib/constraints/group_url_constrainer_spec.rb
index f95adf3a84b88..db680489a8daa 100644
--- a/spec/lib/constraints/group_url_constrainer_spec.rb
+++ b/spec/lib/constraints/group_url_constrainer_spec.rb
@@ -29,9 +29,37 @@
 
       it { expect(subject.matches?(request)).to be_falsey }
     end
+
+    context 'when the request matches a redirect route' do
+      context 'for a root group' do
+        let!(:redirect_route) { group.redirect_routes.create!(path: 'gitlabb') }
+
+        context 'and is a GET request' do
+          let(:request) { build_request(redirect_route.path) }
+
+          it { expect(subject.matches?(request)).to be_truthy }
+        end
+
+        context 'and is NOT a GET request' do
+          let(:request) { build_request(redirect_route.path, 'POST') }
+
+          it { expect(subject.matches?(request)).to be_falsey }
+        end
+      end
+
+      context 'for a nested group' do
+        let!(:nested_group) { create(:group, path: 'nested', parent: group) }
+        let!(:redirect_route) { nested_group.redirect_routes.create!(path: 'gitlabb/nested') }
+        let(:request) { build_request(redirect_route.path) }
+
+        it { expect(subject.matches?(request)).to be_truthy }
+      end
+    end
   end
 
-  def build_request(path)
-    double(:request, params: { id: path })
+  def build_request(path, method = 'GET')
+    double(:request,
+      'get?': (method == 'GET'),
+      params: { id: path })
   end
 end
diff --git a/spec/lib/constraints/project_url_constrainer_spec.rb b/spec/lib/constraints/project_url_constrainer_spec.rb
index 4f25ad8896050..b6884e37aa3d9 100644
--- a/spec/lib/constraints/project_url_constrainer_spec.rb
+++ b/spec/lib/constraints/project_url_constrainer_spec.rb
@@ -24,9 +24,26 @@
         it { expect(subject.matches?(request)).to be_falsey }
       end
     end
+
+    context 'when the request matches a redirect route' do
+      let(:old_project_path) { 'old_project_path' }
+      let!(:redirect_route) { project.redirect_routes.create!(path: "#{namespace.full_path}/#{old_project_path}") }
+
+      context 'and is a GET request' do
+        let(:request) { build_request(namespace.full_path, old_project_path) }
+        it { expect(subject.matches?(request)).to be_truthy }
+      end
+
+      context 'and is NOT a GET request' do
+        let(:request) { build_request(namespace.full_path, old_project_path, 'POST') }
+        it { expect(subject.matches?(request)).to be_falsey }
+      end
+    end
   end
 
-  def build_request(namespace, project)
-    double(:request, params: { namespace_id: namespace, id: project })
+  def build_request(namespace, project, method = 'GET')
+    double(:request,
+      'get?': (method == 'GET'),
+      params: { namespace_id: namespace, id: project })
   end
 end
diff --git a/spec/lib/constraints/user_url_constrainer_spec.rb b/spec/lib/constraints/user_url_constrainer_spec.rb
index 207b6fe6c9ec5..ed69b8309790d 100644
--- a/spec/lib/constraints/user_url_constrainer_spec.rb
+++ b/spec/lib/constraints/user_url_constrainer_spec.rb
@@ -15,9 +15,26 @@
 
       it { expect(subject.matches?(request)).to be_falsey }
     end
+
+    context 'when the request matches a redirect route' do
+      let(:old_project_path) { 'old_project_path' }
+      let!(:redirect_route) { user.namespace.redirect_routes.create!(path: 'foo') }
+
+      context 'and is a GET request' do
+        let(:request) { build_request(redirect_route.path) }
+        it { expect(subject.matches?(request)).to be_truthy }
+      end
+
+      context 'and is NOT a GET request' do
+        let(:request) { build_request(redirect_route.path, 'POST') }
+        it { expect(subject.matches?(request)).to be_falsey }
+      end
+    end
   end
 
-  def build_request(username)
-    double(:request, params: { username: username })
+  def build_request(username, method = 'GET')
+    double(:request,
+      'get?': (method == 'GET'),
+      params: { username: username })
   end
 end
diff --git a/spec/lib/gitlab/import_export/all_models.yml b/spec/lib/gitlab/import_export/all_models.yml
index 0abf89d060cec..f451fb5017b31 100644
--- a/spec/lib/gitlab/import_export/all_models.yml
+++ b/spec/lib/gitlab/import_export/all_models.yml
@@ -225,6 +225,7 @@ project:
 - authorized_users
 - project_authorizations
 - route
+- redirect_routes
 - statistics
 - container_repositories
 - uploads
diff --git a/spec/models/concerns/routable_spec.rb b/spec/models/concerns/routable_spec.rb
index 221647d7a4872..49a4132f7638c 100644
--- a/spec/models/concerns/routable_spec.rb
+++ b/spec/models/concerns/routable_spec.rb
@@ -9,6 +9,7 @@
 
   describe 'Associations' do
     it { is_expected.to have_one(:route).dependent(:destroy) }
+    it { is_expected.to have_many(:redirect_routes).dependent(:destroy) }
   end
 
   describe 'Callbacks' do
@@ -35,10 +36,53 @@
   describe '.find_by_full_path' do
     let!(:nested_group) { create(:group, parent: group) }
 
-    it { expect(described_class.find_by_full_path(group.to_param)).to eq(group) }
-    it { expect(described_class.find_by_full_path(group.to_param.upcase)).to eq(group) }
-    it { expect(described_class.find_by_full_path(nested_group.to_param)).to eq(nested_group) }
-    it { expect(described_class.find_by_full_path('unknown')).to eq(nil) }
+    context 'without any redirect routes' do
+      it { expect(described_class.find_by_full_path(group.to_param)).to eq(group) }
+      it { expect(described_class.find_by_full_path(group.to_param.upcase)).to eq(group) }
+      it { expect(described_class.find_by_full_path(nested_group.to_param)).to eq(nested_group) }
+      it { expect(described_class.find_by_full_path('unknown')).to eq(nil) }
+    end
+
+    context 'with redirect routes' do
+      let!(:group_redirect_route) { group.redirect_routes.create!(path: 'bar') }
+      let!(:nested_group_redirect_route) { nested_group.redirect_routes.create!(path: nested_group.path.sub('foo', 'bar')) }
+
+      context 'without follow_redirects option' do
+        context 'with the given path not matching any route' do
+          it { expect(described_class.find_by_full_path('unknown')).to eq(nil) }
+        end
+
+        context 'with the given path matching the canonical route' do
+          it { expect(described_class.find_by_full_path(group.to_param)).to eq(group) }
+          it { expect(described_class.find_by_full_path(group.to_param.upcase)).to eq(group) }
+          it { expect(described_class.find_by_full_path(nested_group.to_param)).to eq(nested_group) }
+        end
+
+        context 'with the given path matching a redirect route' do
+          it { expect(described_class.find_by_full_path(group_redirect_route.path)).to eq(nil) }
+          it { expect(described_class.find_by_full_path(group_redirect_route.path.upcase)).to eq(nil) }
+          it { expect(described_class.find_by_full_path(nested_group_redirect_route.path)).to eq(nil) }
+        end
+      end
+
+      context 'with follow_redirects option set to true' do
+        context 'with the given path not matching any route' do
+          it { expect(described_class.find_by_full_path('unknown', follow_redirects: true)).to eq(nil) }
+        end
+
+        context 'with the given path matching the canonical route' do
+          it { expect(described_class.find_by_full_path(group.to_param, follow_redirects: true)).to eq(group) }
+          it { expect(described_class.find_by_full_path(group.to_param.upcase, follow_redirects: true)).to eq(group) }
+          it { expect(described_class.find_by_full_path(nested_group.to_param, follow_redirects: true)).to eq(nested_group) }
+        end
+
+        context 'with the given path matching a redirect route' do
+          it { expect(described_class.find_by_full_path(group_redirect_route.path, follow_redirects: true)).to eq(group) }
+          it { expect(described_class.find_by_full_path(group_redirect_route.path.upcase, follow_redirects: true)).to eq(group) }
+          it { expect(described_class.find_by_full_path(nested_group_redirect_route.path, follow_redirects: true)).to eq(nested_group) }
+        end
+      end
+    end
   end
 
   describe '.where_full_path_in' do
diff --git a/spec/models/redirect_route_spec.rb b/spec/models/redirect_route_spec.rb
new file mode 100644
index 0000000000000..fb72d87d94d12
--- /dev/null
+++ b/spec/models/redirect_route_spec.rb
@@ -0,0 +1,16 @@
+require 'rails_helper'
+
+describe RedirectRoute, models: true do
+  let!(:group) { create(:group, path: 'git_lab', name: 'git_lab') }
+  let!(:redirect_route) { group.redirect_routes.create(path: 'gitlabb') }
+
+  describe 'relationships' do
+    it { is_expected.to belong_to(:source) }
+  end
+
+  describe 'validations' do
+    it { is_expected.to validate_presence_of(:source) }
+    it { is_expected.to validate_presence_of(:path) }
+    it { is_expected.to validate_uniqueness_of(:path) }
+  end
+end
diff --git a/spec/models/user_spec.rb b/spec/models/user_spec.rb
index 401eaac07a139..63e71f5ff2f8d 100644
--- a/spec/models/user_spec.rb
+++ b/spec/models/user_spec.rb
@@ -849,6 +849,65 @@
     end
   end
 
+  describe '.find_by_full_path' do
+    let!(:user) { create(:user) }
+
+    context 'with a route matching the given path' do
+      let!(:route) { user.namespace.route }
+
+      it 'returns the user' do
+        expect(User.find_by_full_path(route.path)).to eq(user)
+      end
+
+      it 'is case-insensitive' do
+        expect(User.find_by_full_path(route.path.upcase)).to eq(user)
+        expect(User.find_by_full_path(route.path.downcase)).to eq(user)
+      end
+    end
+
+    context 'with a redirect route matching the given path' do
+      let!(:redirect_route) { user.namespace.redirect_routes.create(path: 'foo') }
+
+      context 'without the follow_redirects option' do
+        it 'returns nil' do
+          expect(User.find_by_full_path(redirect_route.path)).to eq(nil)
+        end
+      end
+
+      context 'with the follow_redirects option set to true' do
+        it 'returns the user' do
+          expect(User.find_by_full_path(redirect_route.path, follow_redirects: true)).to eq(user)
+        end
+
+        it 'is case-insensitive' do
+          expect(User.find_by_full_path(redirect_route.path.upcase, follow_redirects: true)).to eq(user)
+          expect(User.find_by_full_path(redirect_route.path.downcase, follow_redirects: true)).to eq(user)
+        end
+      end
+    end
+
+    context 'without a route or a redirect route matching the given path' do
+      context 'without the follow_redirects option' do
+        it 'returns nil' do
+          expect(User.find_by_full_path('unknown')).to eq(nil)
+        end
+      end
+      context 'with the follow_redirects option set to true' do
+        it 'returns nil' do
+          expect(User.find_by_full_path('unknown', follow_redirects: true)).to eq(nil)
+        end
+      end
+    end
+
+    context 'with a group route matching the given path' do
+      let!(:group) { create(:group, path: 'group_path') }
+
+      it 'returns nil' do
+        expect(User.find_by_full_path('group_path')).to eq(nil)
+      end
+    end
+  end
+
   describe 'all_ssh_keys' do
     it { is_expected.to have_many(:keys).dependent(:destroy) }
 
diff --git a/spec/routing/project_routing_spec.rb b/spec/routing/project_routing_spec.rb
index 163df072cf634..50e96d56191f6 100644
--- a/spec/routing/project_routing_spec.rb
+++ b/spec/routing/project_routing_spec.rb
@@ -3,7 +3,7 @@
 describe 'project routing' do
   before do
     allow(Project).to receive(:find_by_full_path).and_return(false)
-    allow(Project).to receive(:find_by_full_path).with('gitlab/gitlabhq').and_return(true)
+    allow(Project).to receive(:find_by_full_path).with('gitlab/gitlabhq', any_args).and_return(true)
   end
 
   # Shared examples for a resource inside a Project
@@ -93,13 +93,13 @@
       end
 
       context 'name with dot' do
-        before { allow(Project).to receive(:find_by_full_path).with('gitlab/gitlabhq.keys').and_return(true) }
+        before { allow(Project).to receive(:find_by_full_path).with('gitlab/gitlabhq.keys', any_args).and_return(true) }
 
         it { expect(get('/gitlab/gitlabhq.keys')).to route_to('projects#show', namespace_id: 'gitlab', id: 'gitlabhq.keys') }
       end
 
       context 'with nested group' do
-        before { allow(Project).to receive(:find_by_full_path).with('gitlab/subgroup/gitlabhq').and_return(true) }
+        before { allow(Project).to receive(:find_by_full_path).with('gitlab/subgroup/gitlabhq', any_args).and_return(true) }
 
         it { expect(get('/gitlab/subgroup/gitlabhq')).to route_to('projects#show', namespace_id: 'gitlab/subgroup', id: 'gitlabhq') }
       end
-- 
GitLab


From a0368e91310a3b2c0e9e0b717f931a482eb0b90a Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Mon, 1 May 2017 16:48:05 -0700
Subject: [PATCH 324/363] Create redirect routes on path change

---
 app/models/route.rb       | 24 +++++++++++++++++-------
 spec/models/route_spec.rb | 34 +++++++++++++++++++++++++++++++++-
 2 files changed, 50 insertions(+), 8 deletions(-)

diff --git a/app/models/route.rb b/app/models/route.rb
index 4b3efab5c3cb8..df801714b5f4c 100644
--- a/app/models/route.rb
+++ b/app/models/route.rb
@@ -8,15 +8,17 @@ class Route < ActiveRecord::Base
     presence: true,
     uniqueness: { case_sensitive: false }
 
-  after_update :rename_descendants
+  after_update :create_redirect_if_path_changed
+  after_update :rename_direct_descendant_routes
 
   scope :inside_path, -> (path) { where('routes.path LIKE ?', "#{sanitize_sql_like(path)}/%") }
+  scope :direct_descendant_routes, -> (path) { where('routes.path LIKE ? AND routes.path NOT LIKE ?', "#{sanitize_sql_like(path)}/%", "#{sanitize_sql_like(path)}/%/%") }
 
-  def rename_descendants
+  def rename_direct_descendant_routes
     if path_changed? || name_changed?
-      descendants = self.class.inside_path(path_was)
+      direct_descendant_routes = self.class.direct_descendant_routes(path_was)
 
-      descendants.each do |route|
+      direct_descendant_routes.each do |route|
         attributes = {}
 
         if path_changed? && route.path.present?
@@ -27,10 +29,18 @@ def rename_descendants
           attributes[:name] = route.name.sub(name_was, name)
         end
 
-        # Note that update_columns skips validation and callbacks.
-        # We need this to avoid recursive call of rename_descendants method
-        route.update_columns(attributes) unless attributes.empty?
+        route.update(attributes) unless attributes.empty?
       end
     end
   end
+
+  def create_redirect_if_path_changed
+    if path_changed?
+      create_redirect(path_was)
+    end
+  end
+
+  def create_redirect(old_path)
+    source.redirect_routes.create(path: old_path)
+  end
 end
diff --git a/spec/models/route_spec.rb b/spec/models/route_spec.rb
index 171a51fcc5b7f..c95043469a81b 100644
--- a/spec/models/route_spec.rb
+++ b/spec/models/route_spec.rb
@@ -26,7 +26,20 @@
     end
   end
 
-  describe '#rename_descendants' do
+  describe '.direct_descendant_routes' do
+    let!(:nested_group) { create(:group, path: 'test', name: 'test', parent: group) }
+    let!(:deep_nested_group) { create(:group, path: 'foo', name: 'foo', parent: nested_group) }
+    let!(:another_group) { create(:group, path: 'other') }
+    let!(:similar_group) { create(:group, path: 'gitllab') }
+    let!(:another_group_nested) { create(:group, path: 'another', name: 'another', parent: similar_group) }
+
+    it 'returns correct routes' do
+      expect(Route.direct_descendant_routes('git_lab')).to match_array([nested_group.route])
+      expect(Route.direct_descendant_routes('git_lab/test')).to match_array([deep_nested_group.route])
+    end
+  end
+
+  describe '#rename_direct_descendant_routes' do
     let!(:nested_group) { create(:group, path: 'test', name: 'test', parent: group) }
     let!(:deep_nested_group) { create(:group, path: 'foo', name: 'foo', parent: nested_group) }
     let!(:similar_group) { create(:group, path: 'gitlab-org', name: 'gitlab-org') }
@@ -77,4 +90,23 @@
       end
     end
   end
+
+  describe '#create_redirect_if_path_changed' do
+    context 'if the path changed' do
+      it 'creates a RedirectRoute for the old path' do
+        route.path = 'new-path'
+        route.save!
+        redirect = route.source.redirect_routes.first
+        expect(redirect.path).to eq('git_lab')
+      end
+    end
+
+    context 'if the path did not change' do
+      it 'does not create a RedirectRoute' do
+        route.updated_at = Time.zone.now.utc
+        route.save!
+        expect(route.source.redirect_routes).to be_empty
+      end
+    end
+  end
 end
-- 
GitLab


From 72872ee2136436e48ce394268fc8bfb8a2118810 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Wed, 3 May 2017 10:14:30 -0700
Subject: [PATCH 325/363] Delete conflicting redirects

---
 app/models/redirect_route.rb       |   2 +
 app/models/route.rb                |  21 ++++--
 spec/models/redirect_route_spec.rb |  13 +++-
 spec/models/route_spec.rb          | 111 +++++++++++++++++++++++++----
 4 files changed, 127 insertions(+), 20 deletions(-)

diff --git a/app/models/redirect_route.rb b/app/models/redirect_route.rb
index e36ca988701f1..99812bcde5321 100644
--- a/app/models/redirect_route.rb
+++ b/app/models/redirect_route.rb
@@ -7,4 +7,6 @@ class RedirectRoute < ActiveRecord::Base
     length: { within: 1..255 },
     presence: true,
     uniqueness: { case_sensitive: false }
+
+  scope :matching_path_and_descendants, -> (path) { where('redirect_routes.path = ? OR redirect_routes.path LIKE ?', path, "#{sanitize_sql_like(path)}/%") }
 end
diff --git a/app/models/route.rb b/app/models/route.rb
index df801714b5f4c..e0d85ff7db735 100644
--- a/app/models/route.rb
+++ b/app/models/route.rb
@@ -8,7 +8,8 @@ class Route < ActiveRecord::Base
     presence: true,
     uniqueness: { case_sensitive: false }
 
-  after_update :create_redirect_if_path_changed
+  after_save :delete_conflicting_redirects
+  after_update :create_redirect_for_old_path
   after_update :rename_direct_descendant_routes
 
   scope :inside_path, -> (path) { where('routes.path LIKE ?', "#{sanitize_sql_like(path)}/%") }
@@ -34,13 +35,19 @@ def rename_direct_descendant_routes
     end
   end
 
-  def create_redirect_if_path_changed
-    if path_changed?
-      create_redirect(path_was)
-    end
+  def delete_conflicting_redirects
+    conflicting_redirects.delete_all
+  end
+
+  def conflicting_redirects
+    RedirectRoute.matching_path_and_descendants(path)
+  end
+
+  def create_redirect_for_old_path
+    create_redirect(path_was) if path_changed?
   end
 
-  def create_redirect(old_path)
-    source.redirect_routes.create(path: old_path)
+  def create_redirect(path)
+    RedirectRoute.create(source: source, path: path)
   end
 end
diff --git a/spec/models/redirect_route_spec.rb b/spec/models/redirect_route_spec.rb
index fb72d87d94d12..71827421dd720 100644
--- a/spec/models/redirect_route_spec.rb
+++ b/spec/models/redirect_route_spec.rb
@@ -1,7 +1,7 @@
 require 'rails_helper'
 
 describe RedirectRoute, models: true do
-  let!(:group) { create(:group, path: 'git_lab', name: 'git_lab') }
+  let(:group) { create(:group) }
   let!(:redirect_route) { group.redirect_routes.create(path: 'gitlabb') }
 
   describe 'relationships' do
@@ -13,4 +13,15 @@
     it { is_expected.to validate_presence_of(:path) }
     it { is_expected.to validate_uniqueness_of(:path) }
   end
+
+  describe '.matching_path_and_descendants' do
+    let!(:redirect2) { group.redirect_routes.create(path: 'gitlabb/test') }
+    let!(:redirect3) { group.redirect_routes.create(path: 'gitlabb/test/foo') }
+    let!(:redirect4) { group.redirect_routes.create(path: 'gitlabb/test/foo/bar') }
+    let!(:redirect5) { group.redirect_routes.create(path: 'gitlabb/test/baz') }
+
+    it 'returns correct routes' do
+      expect(RedirectRoute.matching_path_and_descendants('gitlabb/test')).to match_array([redirect2, redirect3, redirect4, redirect5])
+    end
+  end
 end
diff --git a/spec/models/route_spec.rb b/spec/models/route_spec.rb
index c95043469a81b..1aeddcef982b3 100644
--- a/spec/models/route_spec.rb
+++ b/spec/models/route_spec.rb
@@ -1,19 +1,43 @@
 require 'spec_helper'
 
 describe Route, models: true do
-  let!(:group) { create(:group, path: 'git_lab', name: 'git_lab') }
-  let!(:route) { group.route }
+  let(:group) { create(:group, path: 'git_lab', name: 'git_lab') }
+  let(:route) { group.route }
 
   describe 'relationships' do
     it { is_expected.to belong_to(:source) }
   end
 
   describe 'validations' do
+    before { route }
     it { is_expected.to validate_presence_of(:source) }
     it { is_expected.to validate_presence_of(:path) }
     it { is_expected.to validate_uniqueness_of(:path) }
   end
 
+  describe 'callbacks' do
+    context 'after update' do
+      it 'calls #create_redirect_for_old_path' do
+        expect(route).to receive(:create_redirect_for_old_path)
+        route.update_attributes(path: 'foo')
+      end
+
+      it 'calls #delete_conflicting_redirects' do
+        expect(route).to receive(:delete_conflicting_redirects)
+        route.update_attributes(path: 'foo')
+      end
+    end
+
+    context 'after create' do
+      it 'calls #delete_conflicting_redirects' do
+        route.destroy
+        new_route = Route.new(source: group, path: group.path)
+        expect(new_route).to receive(:delete_conflicting_redirects)
+        new_route.save!
+      end
+    end
+  end
+
   describe '.inside_path' do
     let!(:nested_group) { create(:group, path: 'test', name: 'test', parent: group) }
     let!(:deep_nested_group) { create(:group, path: 'foo', name: 'foo', parent: nested_group) }
@@ -50,7 +74,7 @@
       context 'when route name is set' do
         before { route.update_attributes(path: 'bar') }
 
-        it "updates children routes with new path" do
+        it 'updates children routes with new path' do
           expect(described_class.exists?(path: 'bar')).to be_truthy
           expect(described_class.exists?(path: 'bar/test')).to be_truthy
           expect(described_class.exists?(path: 'bar/test/foo')).to be_truthy
@@ -69,10 +93,24 @@
           expect(route.update_attributes(path: 'bar')).to be_truthy
         end
       end
+
+      context 'when conflicting redirects exist' do
+        let!(:conflicting_redirect1) { route.create_redirect('bar/test') }
+        let!(:conflicting_redirect2) { route.create_redirect('bar/test/foo') }
+        let!(:conflicting_redirect3) { route.create_redirect('gitlab-org') }
+
+        it 'deletes the conflicting redirects' do
+          route.update_attributes(path: 'bar')
+
+          expect(RedirectRoute.exists?(path: 'bar/test')).to be_falsey
+          expect(RedirectRoute.exists?(path: 'bar/test/foo')).to be_falsey
+          expect(RedirectRoute.exists?(path: 'gitlab-org')).to be_truthy
+        end
+      end
     end
 
     context 'name update' do
-      it "updates children routes with new path" do
+      it 'updates children routes with new path' do
         route.update_attributes(name: 'bar')
 
         expect(described_class.exists?(name: 'bar')).to be_truthy
@@ -91,21 +129,70 @@
     end
   end
 
-  describe '#create_redirect_if_path_changed' do
+  describe '#create_redirect_for_old_path' do
     context 'if the path changed' do
       it 'creates a RedirectRoute for the old path' do
+        redirect_scope = route.source.redirect_routes.where(path: 'git_lab')
+        expect(redirect_scope.exists?).to be_falsey
         route.path = 'new-path'
         route.save!
-        redirect = route.source.redirect_routes.first
-        expect(redirect.path).to eq('git_lab')
+        expect(redirect_scope.exists?).to be_truthy
       end
     end
+  end
 
-    context 'if the path did not change' do
-      it 'does not create a RedirectRoute' do
-        route.updated_at = Time.zone.now.utc
-        route.save!
-        expect(route.source.redirect_routes).to be_empty
+  describe '#create_redirect' do
+    it 'creates a RedirectRoute with the same source' do
+      redirect_route = route.create_redirect('foo')
+      expect(redirect_route).to be_a(RedirectRoute)
+      expect(redirect_route).to be_persisted
+      expect(redirect_route.source).to eq(route.source)
+      expect(redirect_route.path).to eq('foo')
+    end
+  end
+
+  describe '#delete_conflicting_redirects' do
+    context 'when a redirect route with the same path exists' do
+      let!(:redirect1) { route.create_redirect(route.path) }
+
+      it 'deletes the redirect' do
+        route.delete_conflicting_redirects
+        expect(route.conflicting_redirects).to be_empty
+      end
+
+      context 'when redirect routes with paths descending from the route path exists' do
+        let!(:redirect2) { route.create_redirect("#{route.path}/foo") }
+        let!(:redirect3) { route.create_redirect("#{route.path}/foo/bar") }
+        let!(:redirect4) { route.create_redirect("#{route.path}/baz/quz") }
+        let!(:other_redirect) { route.create_redirect("other") }
+
+        it 'deletes all redirects with paths that descend from the route path' do
+          route.delete_conflicting_redirects
+          expect(route.conflicting_redirects).to be_empty
+        end
+      end
+    end
+  end
+
+  describe '#conflicting_redirects' do
+    context 'when a redirect route with the same path exists' do
+      let!(:redirect1) { route.create_redirect(route.path) }
+
+      it 'returns the redirect route' do
+        expect(route.conflicting_redirects).to be_an(ActiveRecord::Relation)
+        expect(route.conflicting_redirects).to match_array([redirect1])
+      end
+
+      context 'when redirect routes with paths descending from the route path exists' do
+        let!(:redirect2) { route.create_redirect("#{route.path}/foo") }
+        let!(:redirect3) { route.create_redirect("#{route.path}/foo/bar") }
+        let!(:redirect4) { route.create_redirect("#{route.path}/baz/quz") }
+        let!(:other_redirect) { route.create_redirect("other") }
+
+        it 'returns the redirect routes' do
+          expect(route.conflicting_redirects).to be_an(ActiveRecord::Relation)
+          expect(route.conflicting_redirects).to match_array([redirect1, redirect2, redirect3, redirect4])
+        end
       end
     end
   end
-- 
GitLab


From ca5c762cf577838a98c0ba96ce20b5f1d5dafc91 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Wed, 3 May 2017 10:33:01 -0700
Subject: [PATCH 326/363] Refactor

---
 app/controllers/users_controller.rb |  8 +++-----
 app/models/route.rb                 | 10 ++++++----
 2 files changed, 9 insertions(+), 9 deletions(-)

diff --git a/app/controllers/users_controller.rb b/app/controllers/users_controller.rb
index 6b6b3b20a8ddf..d7c1241698af7 100644
--- a/app/controllers/users_controller.rb
+++ b/app/controllers/users_controller.rb
@@ -94,11 +94,9 @@ def exists
   private
 
   def authorize_read_user!
-    if can?(current_user, :read_user, user)
-      ensure_canonical_path(user.namespace, params[:username])
-    else
-      render_404
-    end
+    render_404 unless can?(current_user, :read_user, user)
+    
+    ensure_canonical_path(user.namespace, params[:username])
   end
 
   def user
diff --git a/app/models/route.rb b/app/models/route.rb
index e0d85ff7db735..accc423ae463a 100644
--- a/app/models/route.rb
+++ b/app/models/route.rb
@@ -43,11 +43,13 @@ def conflicting_redirects
     RedirectRoute.matching_path_and_descendants(path)
   end
 
-  def create_redirect_for_old_path
-    create_redirect(path_was) if path_changed?
-  end
-
   def create_redirect(path)
     RedirectRoute.create(source: source, path: path)
   end
+
+  private
+
+  def create_redirect_for_old_path
+    create_redirect(path_was) if path_changed?
+  end
 end
-- 
GitLab


From e8f2a7007a96d19cf7f7e7278bf21007fafa9f1c Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Wed, 3 May 2017 10:54:06 -0700
Subject: [PATCH 327/363] Fix or workaround spec failure
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Not sure why this fixes this error:

```
ActiveRecord::RecordNotUnique:
       PG::UniqueViolation: ERROR:  duplicate key value violates unique constraint "index_routes_on_source_type_and_source_id"
       DETAIL:  Key (source_type, source_id)=(Project, 1) already exists.
       : INSERT INTO "routes" ("source_type", "path", "name", "source_id", "created_at", "updated_at") VALUES ($1, $2, $3, $4, $5, $6) RETURNING "id"
```
---
 spec/features/projects/project_settings_spec.rb | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/spec/features/projects/project_settings_spec.rb b/spec/features/projects/project_settings_spec.rb
index c0311e73ef5e7..11dcab4d73742 100644
--- a/spec/features/projects/project_settings_spec.rb
+++ b/spec/features/projects/project_settings_spec.rb
@@ -93,8 +93,8 @@
 
   describe 'Transfer project section', js: true do
     # Not using empty project because we need a repo to exist
-    let(:project) { create(:project, namespace: user.namespace, name: 'gitlabhq') }
-    let(:group) { create(:group) }
+    let!(:project) { create(:project, namespace: user.namespace, name: 'gitlabhq') }
+    let!(:group) { create(:group) }
 
     before(:context) { TestEnv.clean_test_path }
     before(:example) { group.add_owner(user) }
-- 
GitLab


From 4da848ef28fb9ff247145670a107dff82c83b270 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Wed, 3 May 2017 11:49:17 -0700
Subject: [PATCH 328/363] Add index for source association and for path

---
 ...0503184421_add_index_to_redirect_routes.rb | 22 +++++++++++++++++++
 db/schema.rb                                  |  5 ++++-
 2 files changed, 26 insertions(+), 1 deletion(-)
 create mode 100644 db/migrate/20170503184421_add_index_to_redirect_routes.rb

diff --git a/db/migrate/20170503184421_add_index_to_redirect_routes.rb b/db/migrate/20170503184421_add_index_to_redirect_routes.rb
new file mode 100644
index 0000000000000..5991f6ab6a11a
--- /dev/null
+++ b/db/migrate/20170503184421_add_index_to_redirect_routes.rb
@@ -0,0 +1,22 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+# rubocop:disable RemoveIndex
+class AddIndexToRedirectRoutes < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  disable_ddl_transaction!
+
+  def up
+    add_concurrent_index(:redirect_routes, :path, unique: true)
+    add_concurrent_index(:redirect_routes, [:source_type, :source_id])
+  end
+
+  def down
+    remove_index(:redirect_routes, :path) if index_exists?(:redirect_routes, :path)
+    remove_index(:redirect_routes, [:source_type, :source_id]) if index_exists?(:redirect_routes, [:source_type, :source_id])
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 3a8970b2235e8..45c3b9a4c917c 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -1048,6 +1048,9 @@
     t.datetime "updated_at", null: false
   end
 
+  add_index "redirect_routes", ["path"], name: "index_redirect_routes_on_path", unique: true, using: :btree
+  add_index "redirect_routes", ["source_type", "source_id"], name: "index_redirect_routes_on_source_type_and_source_id", using: :btree
+
   create_table "releases", force: :cascade do |t|
     t.string "tag"
     t.text "description"
@@ -1423,4 +1426,4 @@
   add_foreign_key "timelogs", "merge_requests", name: "fk_timelogs_merge_requests_merge_request_id", on_delete: :cascade
   add_foreign_key "trending_projects", "projects", on_delete: :cascade
   add_foreign_key "u2f_registrations", "users"
-end
\ No newline at end of file
+end
-- 
GitLab


From 03144e1c0f2ba41a7570850b69d5029bc2619fd2 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Wed, 3 May 2017 12:00:52 -0700
Subject: [PATCH 329/363] Index redirect_routes path for LIKE

---
 ...032_index_redirect_routes_path_for_like.rb | 29 +++++++++++++++++++
 db/schema.rb                                  |  1 +
 lib/tasks/migrate/setup_postgresql.rake       |  1 +
 3 files changed, 31 insertions(+)
 create mode 100644 db/migrate/20170503185032_index_redirect_routes_path_for_like.rb

diff --git a/db/migrate/20170503185032_index_redirect_routes_path_for_like.rb b/db/migrate/20170503185032_index_redirect_routes_path_for_like.rb
new file mode 100644
index 0000000000000..5b8b6c828be8e
--- /dev/null
+++ b/db/migrate/20170503185032_index_redirect_routes_path_for_like.rb
@@ -0,0 +1,29 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class IndexRedirectRoutesPathForLike < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  INDEX_NAME = 'index_redirect_routes_on_path_text_pattern_ops'
+
+  disable_ddl_transaction!
+
+  def up
+    return unless Gitlab::Database.postgresql?
+
+    unless index_exists?(:redirect_routes, :path, name: INDEX_NAME)
+      execute("CREATE INDEX CONCURRENTLY #{INDEX_NAME} ON redirect_routes (path varchar_pattern_ops);")
+    end
+  end
+
+  def down
+    return unless Gitlab::Database.postgresql?
+
+    if index_exists?(:redirect_routes, :path, name: INDEX_NAME)
+      execute("DROP INDEX CONCURRENTLY #{INDEX_NAME};")
+    end
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 45c3b9a4c917c..8d3f4b4bd3616 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -1049,6 +1049,7 @@
   end
 
   add_index "redirect_routes", ["path"], name: "index_redirect_routes_on_path", unique: true, using: :btree
+  add_index "redirect_routes", ["path"], name: "index_redirect_routes_on_path_text_pattern_ops", using: :btree, opclasses: {"path"=>"varchar_pattern_ops"}
   add_index "redirect_routes", ["source_type", "source_id"], name: "index_redirect_routes_on_source_type_and_source_id", using: :btree
 
   create_table "releases", force: :cascade do |t|
diff --git a/lib/tasks/migrate/setup_postgresql.rake b/lib/tasks/migrate/setup_postgresql.rake
index 8938bc515f5ae..1e00b47303d5f 100644
--- a/lib/tasks/migrate/setup_postgresql.rake
+++ b/lib/tasks/migrate/setup_postgresql.rake
@@ -11,4 +11,5 @@ task setup_postgresql: :environment do
   AddUsersLowerUsernameEmailIndexes.new.up
   AddLowerPathIndexToRoutes.new.up
   IndexRoutesPathForLike.new.up
+  IndexRedirectRoutesPathForLike.new.up
 end
-- 
GitLab


From fc061c2ecd2e292383017c703220bfb22d0d6dce Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Wed, 3 May 2017 12:01:44 -0700
Subject: [PATCH 330/363] Fix Rubocop failures

---
 db/migrate/20170427215854_create_redirect_routes.rb | 1 -
 spec/controllers/groups_controller_spec.rb          | 8 ++++----
 spec/features/projects/features_visibility_spec.rb  | 4 ++--
 3 files changed, 6 insertions(+), 7 deletions(-)

diff --git a/db/migrate/20170427215854_create_redirect_routes.rb b/db/migrate/20170427215854_create_redirect_routes.rb
index 57b79a0267fd3..2bf086b3e30fc 100644
--- a/db/migrate/20170427215854_create_redirect_routes.rb
+++ b/db/migrate/20170427215854_create_redirect_routes.rb
@@ -1,5 +1,4 @@
 class CreateRedirectRoutes < ActiveRecord::Migration
-
   # Set this constant to true if this migration requires downtime.
   DOWNTIME = false
 
diff --git a/spec/controllers/groups_controller_spec.rb b/spec/controllers/groups_controller_spec.rb
index 3398878f33058..d86bf1a3d0e49 100644
--- a/spec/controllers/groups_controller_spec.rb
+++ b/spec/controllers/groups_controller_spec.rb
@@ -144,13 +144,13 @@
         it 'does not 404' do
           delete :destroy, id: group.to_param.upcase
 
-          expect(response).to_not have_http_status(404)
+          expect(response).not_to have_http_status(404)
         end
 
         it 'does not redirect to the correct casing' do
           delete :destroy, id: group.to_param.upcase
 
-          expect(response).to_not redirect_to(group_path(group.to_param))
+          expect(response).not_to redirect_to(group_path(group.to_param))
         end
       end
 
@@ -190,13 +190,13 @@
       it 'does not 404' do
         post :update, id: group.to_param.upcase, group: { path: 'new_path' }
 
-        expect(response).to_not have_http_status(404)
+        expect(response).not_to have_http_status(404)
       end
 
       it 'does not redirect to the correct casing' do
         post :update, id: group.to_param.upcase, group: { path: 'new_path' }
 
-        expect(response).to_not redirect_to(group_path(group.to_param))
+        expect(response).not_to redirect_to(group_path(group.to_param))
       end
     end
 
diff --git a/spec/features/projects/features_visibility_spec.rb b/spec/features/projects/features_visibility_spec.rb
index a6ca0f4131a0a..e1781cf320ac5 100644
--- a/spec/features/projects/features_visibility_spec.rb
+++ b/spec/features/projects/features_visibility_spec.rb
@@ -68,7 +68,7 @@
   end
 
   describe 'project features visibility pages' do
-    let(:tools) {
+    let(:tools) do
       {
         builds: namespace_project_pipelines_path(project.namespace, project),
         issues: namespace_project_issues_path(project.namespace, project),
@@ -76,7 +76,7 @@
         snippets: namespace_project_snippets_path(project.namespace, project),
         merge_requests: namespace_project_merge_requests_path(project.namespace, project),
       }
-    }
+    end
 
     context 'normal user' do
       before do
-- 
GitLab


From 0c866f4a575d8127efbf3eafda83d8ccfbd97817 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Wed, 3 May 2017 15:26:44 -0700
Subject: [PATCH 331/363] Resolve discussions

---
 app/controllers/users_controller.rb           | 18 ++++++++-------
 app/models/route.rb                           | 22 +++++++++----------
 app/models/user.rb                            |  2 +-
 ...0503184421_add_index_to_redirect_routes.rb |  5 ++---
 spec/controllers/users_controller_spec.rb     | 18 +++++++++++++++
 5 files changed, 42 insertions(+), 23 deletions(-)

diff --git a/app/controllers/users_controller.rb b/app/controllers/users_controller.rb
index d7c1241698af7..67783866c3fe4 100644
--- a/app/controllers/users_controller.rb
+++ b/app/controllers/users_controller.rb
@@ -3,7 +3,6 @@ class UsersController < ApplicationController
 
   skip_before_action :authenticate_user!
   before_action :user, except: [:exists]
-  before_action :authorize_read_user!, except: [:exists]
 
   def show
     respond_to do |format|
@@ -93,14 +92,17 @@ def exists
 
   private
 
-  def authorize_read_user!
-    render_404 unless can?(current_user, :read_user, user)
-    
-    ensure_canonical_path(user.namespace, params[:username])
-  end
-
   def user
-    @user ||= User.find_by_full_path(params[:username], follow_redirects: true)
+    return @user if @user
+
+    @user = User.find_by_full_path(params[:username], follow_redirects: true)
+
+    return render_404 unless @user
+    return render_404 unless can?(current_user, :read_user, @user)
+
+    ensure_canonical_path(@user.namespace, params[:username])
+
+    @user
   end
 
   def contributed_projects
diff --git a/app/models/route.rb b/app/models/route.rb
index accc423ae463a..3d798ce937ba1 100644
--- a/app/models/route.rb
+++ b/app/models/route.rb
@@ -16,22 +16,22 @@ class Route < ActiveRecord::Base
   scope :direct_descendant_routes, -> (path) { where('routes.path LIKE ? AND routes.path NOT LIKE ?', "#{sanitize_sql_like(path)}/%", "#{sanitize_sql_like(path)}/%/%") }
 
   def rename_direct_descendant_routes
-    if path_changed? || name_changed?
-      direct_descendant_routes = self.class.direct_descendant_routes(path_was)
+    return if !path_changed? && !name_changed?
 
-      direct_descendant_routes.each do |route|
-        attributes = {}
+    direct_descendant_routes = self.class.direct_descendant_routes(path_was)
 
-        if path_changed? && route.path.present?
-          attributes[:path] = route.path.sub(path_was, path)
-        end
+    direct_descendant_routes.each do |route|
+      attributes = {}
 
-        if name_changed? && name_was.present? && route.name.present?
-          attributes[:name] = route.name.sub(name_was, name)
-        end
+      if path_changed? && route.path.present?
+        attributes[:path] = route.path.sub(path_was, path)
+      end
 
-        route.update(attributes) unless attributes.empty?
+      if name_changed? && name_was.present? && route.name.present?
+        attributes[:name] = route.name.sub(name_was, name)
       end
+
+      route.update(attributes) unless attributes.empty?
     end
   end
 
diff --git a/app/models/user.rb b/app/models/user.rb
index c6354b989ade2..7da92d03427fc 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -335,7 +335,7 @@ def find_by_ssh_key_id(key_id)
 
     def find_by_full_path(path, follow_redirects: false)
       namespace = Namespace.find_by_full_path(path, follow_redirects: follow_redirects)
-      namespace.owner if namespace && namespace.owner
+      namespace&.owner
     end
 
     def reference_prefix
diff --git a/db/migrate/20170503184421_add_index_to_redirect_routes.rb b/db/migrate/20170503184421_add_index_to_redirect_routes.rb
index 5991f6ab6a11a..9062cf19a7398 100644
--- a/db/migrate/20170503184421_add_index_to_redirect_routes.rb
+++ b/db/migrate/20170503184421_add_index_to_redirect_routes.rb
@@ -1,7 +1,6 @@
 # See http://doc.gitlab.com/ce/development/migration_style_guide.html
 # for more information on how to write migrations for GitLab.
 
-# rubocop:disable RemoveIndex
 class AddIndexToRedirectRoutes < ActiveRecord::Migration
   include Gitlab::Database::MigrationHelpers
 
@@ -16,7 +15,7 @@ def up
   end
 
   def down
-    remove_index(:redirect_routes, :path) if index_exists?(:redirect_routes, :path)
-    remove_index(:redirect_routes, [:source_type, :source_id]) if index_exists?(:redirect_routes, [:source_type, :source_id])
+    remove_concurrent_index(:redirect_routes, :path) if index_exists?(:redirect_routes, :path)
+    remove_concurrent_index(:redirect_routes, [:source_type, :source_id]) if index_exists?(:redirect_routes, [:source_type, :source_id])
   end
 end
diff --git a/spec/controllers/users_controller_spec.rb b/spec/controllers/users_controller_spec.rb
index 73fd5b6f90ded..73f448d69ed05 100644
--- a/spec/controllers/users_controller_spec.rb
+++ b/spec/controllers/users_controller_spec.rb
@@ -84,6 +84,24 @@
         expect(response).to redirect_to(user)
       end
     end
+
+    context 'when a user by that username does not exist' do
+      context 'when logged out' do
+        it 'renders 404 (does not redirect to login)' do
+          get :show, username: 'nonexistent'
+          expect(response).to have_http_status(404)
+        end
+      end
+
+      context 'when logged in' do
+        before { sign_in(user) }
+
+        it 'renders 404' do
+          get :show, username: 'nonexistent'
+          expect(response).to have_http_status(404)
+        end
+      end
+    end
   end
 
   describe 'GET #calendar' do
-- 
GitLab


From e4bcc90d95fa3b78544cb9ddd6019a5f914c1628 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Thu, 4 May 2017 11:12:19 -0700
Subject: [PATCH 332/363] =?UTF-8?q?Add=20=E2=80=9Cproject=20moved=E2=80=9D?=
 =?UTF-8?q?=20flash=20message=20on=20redirect?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

---
 app/controllers/concerns/routable_actions.rb | 1 +
 spec/controllers/groups_controller_spec.rb   | 2 ++
 spec/controllers/projects_controller_spec.rb | 2 ++
 spec/controllers/users_controller_spec.rb    | 4 ++++
 4 files changed, 9 insertions(+)

diff --git a/app/controllers/concerns/routable_actions.rb b/app/controllers/concerns/routable_actions.rb
index 6f16377a156f6..1714bc25e524c 100644
--- a/app/controllers/concerns/routable_actions.rb
+++ b/app/controllers/concerns/routable_actions.rb
@@ -5,6 +5,7 @@ def ensure_canonical_path(routable, requested_path)
     return unless request.get?
 
     if routable.full_path != requested_path
+      flash[:notice] = 'This project has moved to this location. Please update your links and bookmarks.'
       redirect_to request.original_url.sub(requested_path, routable.full_path)
     end
   end
diff --git a/spec/controllers/groups_controller_spec.rb b/spec/controllers/groups_controller_spec.rb
index d86bf1a3d0e49..b5fd611747a3d 100644
--- a/spec/controllers/groups_controller_spec.rb
+++ b/spec/controllers/groups_controller_spec.rb
@@ -65,6 +65,7 @@
         get :issues, id: redirect_route.path
 
         expect(response).to redirect_to(issues_group_path(group.to_param))
+        expect(controller).to set_flash[:notice].to(/moved/)
       end
     end
   end
@@ -108,6 +109,7 @@
         get :merge_requests, id: redirect_route.path
 
         expect(response).to redirect_to(merge_requests_group_path(group.to_param))
+        expect(controller).to set_flash[:notice].to(/moved/)
       end
     end
   end
diff --git a/spec/controllers/projects_controller_spec.rb b/spec/controllers/projects_controller_spec.rb
index 1b0dd7c63696d..5e3e943c124e1 100644
--- a/spec/controllers/projects_controller_spec.rb
+++ b/spec/controllers/projects_controller_spec.rb
@@ -226,6 +226,7 @@
         get :show, namespace_id: 'foo', id: 'bar'
 
         expect(response).to redirect_to(public_project)
+        expect(controller).to set_flash[:notice].to(/moved/)
       end
     end
   end
@@ -471,6 +472,7 @@
         get :refs, namespace_id: 'foo', id: 'bar'
 
         expect(response).to redirect_to(refs_namespace_project_path(namespace_id: public_project.namespace, id: public_project))
+        expect(controller).to set_flash[:notice].to(/moved/)
       end
     end
   end
diff --git a/spec/controllers/users_controller_spec.rb b/spec/controllers/users_controller_spec.rb
index 73f448d69ed05..5e8caa89cb7ad 100644
--- a/spec/controllers/users_controller_spec.rb
+++ b/spec/controllers/users_controller_spec.rb
@@ -82,6 +82,7 @@
         get :show, username: redirect_route.path
 
         expect(response).to redirect_to(user)
+        expect(controller).to set_flash[:notice].to(/moved/)
       end
     end
 
@@ -159,6 +160,7 @@
         get :calendar, username: redirect_route.path
 
         expect(response).to redirect_to(user_calendar_path(user))
+        expect(controller).to set_flash[:notice].to(/moved/)
       end
     end
   end
@@ -211,6 +213,7 @@
         get :calendar_activities, username: redirect_route.path
 
         expect(response).to redirect_to(user_calendar_activities_path(user))
+        expect(controller).to set_flash[:notice].to(/moved/)
       end
     end
   end
@@ -263,6 +266,7 @@
         get :snippets, username: redirect_route.path
 
         expect(response).to redirect_to(user_snippets_path(user))
+        expect(controller).to set_flash[:notice].to(/moved/)
       end
     end
   end
-- 
GitLab


From 9e48f02ea802814e4df1f1de5ed509942dca7581 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Thu, 4 May 2017 14:20:13 -0700
Subject: [PATCH 333/363] Dry up routable lookups. Fixes #30317

Note: This changes the behavior of user lookups (see the spec change) so it acts the same way as groups and projects. Unauthenticated clients attempting to access a user page will be redirected to login whether the user exists and is publicly restricted, or does not exist at all.
---
 app/controllers/concerns/routable_actions.rb  | 28 ++++++++++-
 .../groups/application_controller.rb          | 17 ++-----
 .../projects/application_controller.rb        | 49 +++++++++----------
 app/controllers/users_controller.rb           | 11 +----
 spec/controllers/users_controller_spec.rb     |  8 +--
 5 files changed, 58 insertions(+), 55 deletions(-)

diff --git a/app/controllers/concerns/routable_actions.rb b/app/controllers/concerns/routable_actions.rb
index 1714bc25e524c..ba236fa5459a2 100644
--- a/app/controllers/concerns/routable_actions.rb
+++ b/app/controllers/concerns/routable_actions.rb
@@ -1,12 +1,36 @@
 module RoutableActions
   extend ActiveSupport::Concern
 
+  def find_routable!(routable_klass, requested_full_path, extra_authorization_method: nil)
+    routable = routable_klass.find_by_full_path(requested_full_path, follow_redirects: request.get?)
+
+    if authorized?(routable_klass, routable, extra_authorization_method)
+      ensure_canonical_path(routable, requested_full_path)
+      routable
+    else
+      route_not_found
+      nil
+    end
+  end
+
+  def authorized?(routable_klass, routable, extra_authorization_method)
+    action = :"read_#{routable_klass.to_s.underscore}"
+    return false unless can?(current_user, action, routable)
+
+    if extra_authorization_method
+      send(extra_authorization_method, routable)
+    else
+      true
+    end
+  end
+
   def ensure_canonical_path(routable, requested_path)
     return unless request.get?
 
-    if routable.full_path != requested_path
+    canonical_path = routable.try(:full_path) || routable.namespace.full_path
+    if canonical_path != requested_path
       flash[:notice] = 'This project has moved to this location. Please update your links and bookmarks.'
-      redirect_to request.original_url.sub(requested_path, routable.full_path)
+      redirect_to request.original_url.sub(requested_path, canonical_path)
     end
   end
 end
diff --git a/app/controllers/groups/application_controller.rb b/app/controllers/groups/application_controller.rb
index 209d8b1a08afe..2157a56dea240 100644
--- a/app/controllers/groups/application_controller.rb
+++ b/app/controllers/groups/application_controller.rb
@@ -9,20 +9,11 @@ class Groups::ApplicationController < ApplicationController
   private
 
   def group
-    unless @group
-      given_path = params[:group_id] || params[:id]
-      @group = Group.find_by_full_path(given_path, follow_redirects: request.get?)
-
-      if @group && can?(current_user, :read_group, @group)
-        ensure_canonical_path(@group, given_path)
-      else
-        @group = nil
-
-        route_not_found
-      end
-    end
+    @group ||= find_routable!(Group, requested_full_path)
+  end
 
-    @group
+  def requested_full_path
+    params[:group_id] || params[:id]
   end
 
   def group_projects
diff --git a/app/controllers/projects/application_controller.rb b/app/controllers/projects/application_controller.rb
index dbdf68776f1d7..2301e1cca779b 100644
--- a/app/controllers/projects/application_controller.rb
+++ b/app/controllers/projects/application_controller.rb
@@ -2,6 +2,7 @@ class Projects::ApplicationController < ApplicationController
   include RoutableActions
 
   skip_before_action :authenticate_user!
+  before_action :redirect_git_extension
   before_action :project
   before_action :repository
   layout 'project'
@@ -10,34 +11,30 @@ class Projects::ApplicationController < ApplicationController
 
   private
 
-  def project
-    unless @project
-      namespace = params[:namespace_id]
-      id = params[:project_id] || params[:id]
-
-      # Redirect from
-      #   localhost/group/project.git
-      # to
-      #   localhost/group/project
-      #
-      if params[:format] == 'git'
-        redirect_to request.original_url.gsub(/\.git\/?\Z/, '')
-        return
-      end
-
-      project_path = "#{namespace}/#{id}"
-      @project = Project.find_by_full_path(project_path, follow_redirects: request.get?)
-
-      if can?(current_user, :read_project, @project) && !@project.pending_delete?
-        ensure_canonical_path(@project, project_path)
-      else
-        @project = nil
-
-        route_not_found
-      end
+  def redirect_git_extension
+    # Redirect from
+    #   localhost/group/project.git
+    # to
+    #   localhost/group/project
+    #
+    if params[:format] == 'git'
+      redirect_to request.original_url.gsub(/\.git\/?\Z/, '')
+      return
     end
+  end
+
+  def project
+    @project ||= find_routable!(Project, requested_full_path, extra_authorization_method: :project_not_being_deleted?)
+  end
+
+  def requested_full_path
+    namespace = params[:namespace_id]
+    id = params[:project_id] || params[:id]
+    "#{namespace}/#{id}"
+  end
 
-    @project
+  def project_not_being_deleted?(project)
+    !project.pending_delete?
   end
 
   def repository
diff --git a/app/controllers/users_controller.rb b/app/controllers/users_controller.rb
index 67783866c3fe4..ca89ed221c677 100644
--- a/app/controllers/users_controller.rb
+++ b/app/controllers/users_controller.rb
@@ -93,16 +93,7 @@ def exists
   private
 
   def user
-    return @user if @user
-
-    @user = User.find_by_full_path(params[:username], follow_redirects: true)
-
-    return render_404 unless @user
-    return render_404 unless can?(current_user, :read_user, @user)
-
-    ensure_canonical_path(@user.namespace, params[:username])
-
-    @user
+    @user ||= find_routable!(User, params[:username])
   end
 
   def contributed_projects
diff --git a/spec/controllers/users_controller_spec.rb b/spec/controllers/users_controller_spec.rb
index 5e8caa89cb7ad..9b6b9358a4022 100644
--- a/spec/controllers/users_controller_spec.rb
+++ b/spec/controllers/users_controller_spec.rb
@@ -36,9 +36,9 @@
       end
 
       context 'when logged out' do
-        it 'renders 404' do
+        it 'redirects to login page' do
           get :show, username: user.username
-          expect(response).to have_http_status(404)
+          expect(response).to redirect_to new_user_session_path
         end
       end
 
@@ -88,9 +88,9 @@
 
     context 'when a user by that username does not exist' do
       context 'when logged out' do
-        it 'renders 404 (does not redirect to login)' do
+        it 'redirects to login page' do
           get :show, username: 'nonexistent'
-          expect(response).to have_http_status(404)
+          expect(response).to redirect_to new_user_session_path
         end
       end
 
-- 
GitLab


From f05469f99b8c52c4dab7ac9160b47676c87124f9 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Thu, 4 May 2017 17:06:01 -0700
Subject: [PATCH 334/363] Resolve discussions

---
 app/controllers/concerns/routable_actions.rb  | 16 +++++++++-------
 .../groups/application_controller.rb          |  6 +-----
 .../projects/application_controller.rb        | 19 ++++++-------------
 app/models/user.rb                            |  4 ++++
 .../17361-redirect-renamed-paths.yml          |  4 ++++
 spec/controllers/groups_controller_spec.rb    |  2 ++
 spec/controllers/projects_controller_spec.rb  |  1 +
 spec/controllers/users_controller_spec.rb     |  4 ++++
 8 files changed, 31 insertions(+), 25 deletions(-)
 create mode 100644 changelogs/unreleased/17361-redirect-renamed-paths.yml

diff --git a/app/controllers/concerns/routable_actions.rb b/app/controllers/concerns/routable_actions.rb
index ba236fa5459a2..d4ab67824441b 100644
--- a/app/controllers/concerns/routable_actions.rb
+++ b/app/controllers/concerns/routable_actions.rb
@@ -1,10 +1,10 @@
 module RoutableActions
   extend ActiveSupport::Concern
 
-  def find_routable!(routable_klass, requested_full_path, extra_authorization_method: nil)
+  def find_routable!(routable_klass, requested_full_path, extra_authorization_proc: nil)
     routable = routable_klass.find_by_full_path(requested_full_path, follow_redirects: request.get?)
 
-    if authorized?(routable_klass, routable, extra_authorization_method)
+    if routable_authorized?(routable_klass, routable, extra_authorization_proc)
       ensure_canonical_path(routable, requested_full_path)
       routable
     else
@@ -13,12 +13,12 @@ def find_routable!(routable_klass, requested_full_path, extra_authorization_meth
     end
   end
 
-  def authorized?(routable_klass, routable, extra_authorization_method)
+  def routable_authorized?(routable_klass, routable, extra_authorization_proc)
     action = :"read_#{routable_klass.to_s.underscore}"
     return false unless can?(current_user, action, routable)
 
-    if extra_authorization_method
-      send(extra_authorization_method, routable)
+    if extra_authorization_proc
+      extra_authorization_proc.call(routable)
     else
       true
     end
@@ -27,9 +27,11 @@ def authorized?(routable_klass, routable, extra_authorization_method)
   def ensure_canonical_path(routable, requested_path)
     return unless request.get?
 
-    canonical_path = routable.try(:full_path) || routable.namespace.full_path
+    canonical_path = routable.full_path
     if canonical_path != requested_path
-      flash[:notice] = 'This project has moved to this location. Please update your links and bookmarks.'
+      if canonical_path.casecmp(requested_path) != 0
+        flash[:notice] = "Project '#{requested_path}' was moved to '#{canonical_path}'. Please update any links and bookmarks that may still have the old path."
+      end
       redirect_to request.original_url.sub(requested_path, canonical_path)
     end
   end
diff --git a/app/controllers/groups/application_controller.rb b/app/controllers/groups/application_controller.rb
index 2157a56dea240..afffb813b440b 100644
--- a/app/controllers/groups/application_controller.rb
+++ b/app/controllers/groups/application_controller.rb
@@ -9,11 +9,7 @@ class Groups::ApplicationController < ApplicationController
   private
 
   def group
-    @group ||= find_routable!(Group, requested_full_path)
-  end
-
-  def requested_full_path
-    params[:group_id] || params[:id]
+    @group ||= find_routable!(Group, params[:group_id] || params[:id])
   end
 
   def group_projects
diff --git a/app/controllers/projects/application_controller.rb b/app/controllers/projects/application_controller.rb
index 2301e1cca779b..25232fc9457c6 100644
--- a/app/controllers/projects/application_controller.rb
+++ b/app/controllers/projects/application_controller.rb
@@ -17,24 +17,17 @@ def redirect_git_extension
     # to
     #   localhost/group/project
     #
-    if params[:format] == 'git'
-      redirect_to request.original_url.gsub(/\.git\/?\Z/, '')
-      return
-    end
+    redirect_to url_for(params.merge(format: nil)) if params[:format] == 'git'
   end
 
   def project
-    @project ||= find_routable!(Project, requested_full_path, extra_authorization_method: :project_not_being_deleted?)
-  end
-
-  def requested_full_path
-    namespace = params[:namespace_id]
-    id = params[:project_id] || params[:id]
-    "#{namespace}/#{id}"
+    @project ||= find_routable!(Project,
+      File.join(params[:namespace_id], params[:project_id] || params[:id]),
+      extra_authorization_proc: project_not_being_deleted?)
   end
 
-  def project_not_being_deleted?(project)
-    !project.pending_delete?
+  def project_not_being_deleted?
+    ->(project) { !project.pending_delete? }
   end
 
   def repository
diff --git a/app/models/user.rb b/app/models/user.rb
index 7da92d03427fc..dd2c8f1b6ef1b 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -360,6 +360,10 @@ def ghost
     end
   end
 
+  def full_path
+    username
+  end
+
   def self.internal_attributes
     [:ghost]
   end
diff --git a/changelogs/unreleased/17361-redirect-renamed-paths.yml b/changelogs/unreleased/17361-redirect-renamed-paths.yml
new file mode 100644
index 0000000000000..7a33c9fb3ec47
--- /dev/null
+++ b/changelogs/unreleased/17361-redirect-renamed-paths.yml
@@ -0,0 +1,4 @@
+---
+title: Redirect old links after renaming a user/group/project.
+merge_request: 10370
+author:
diff --git a/spec/controllers/groups_controller_spec.rb b/spec/controllers/groups_controller_spec.rb
index b5fd611747a3d..073b87a1cb483 100644
--- a/spec/controllers/groups_controller_spec.rb
+++ b/spec/controllers/groups_controller_spec.rb
@@ -55,6 +55,7 @@
         get :issues, id: group.to_param.upcase
 
         expect(response).to redirect_to(issues_group_path(group.to_param))
+        expect(controller).not_to set_flash[:notice]
       end
     end
 
@@ -99,6 +100,7 @@
         get :merge_requests, id: group.to_param.upcase
 
         expect(response).to redirect_to(merge_requests_group_path(group.to_param))
+        expect(controller).not_to set_flash[:notice]
       end
     end
 
diff --git a/spec/controllers/projects_controller_spec.rb b/spec/controllers/projects_controller_spec.rb
index 5e3e943c124e1..e46ef447df252 100644
--- a/spec/controllers/projects_controller_spec.rb
+++ b/spec/controllers/projects_controller_spec.rb
@@ -185,6 +185,7 @@
 
           expect(assigns(:project)).to eq(public_project)
           expect(response).to redirect_to("/#{public_project.full_path}")
+          expect(controller).not_to set_flash[:notice]
         end
       end
     end
diff --git a/spec/controllers/users_controller_spec.rb b/spec/controllers/users_controller_spec.rb
index 9b6b9358a4022..74c5aa44ba96d 100644
--- a/spec/controllers/users_controller_spec.rb
+++ b/spec/controllers/users_controller_spec.rb
@@ -71,6 +71,7 @@
           get :show, username: user.username.downcase
 
           expect(response).to redirect_to(user)
+          expect(controller).not_to set_flash[:notice]
         end
       end
     end
@@ -149,6 +150,7 @@
           get :calendar, username: user.username.downcase
 
           expect(response).to redirect_to(user_calendar_path(user))
+          expect(controller).not_to set_flash[:notice]
         end
       end
     end
@@ -202,6 +204,7 @@
           get :calendar_activities, username: user.username.downcase
 
           expect(response).to redirect_to(user_calendar_activities_path(user))
+          expect(controller).not_to set_flash[:notice]
         end
       end
     end
@@ -255,6 +258,7 @@
           get :snippets, username: user.username.downcase
 
           expect(response).to redirect_to(user_snippets_path(user))
+          expect(controller).not_to set_flash[:notice]
         end
       end
     end
-- 
GitLab


From e1c245af51e294c84552cff8021342e7ae493b8a Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Fri, 5 May 2017 10:48:01 -0700
Subject: [PATCH 335/363] Resolve discussions

---
 app/controllers/projects/application_controller.rb | 11 +++++------
 app/models/route.rb                                |  2 +-
 2 files changed, 6 insertions(+), 7 deletions(-)

diff --git a/app/controllers/projects/application_controller.rb b/app/controllers/projects/application_controller.rb
index 25232fc9457c6..b4b0dfc3eb890 100644
--- a/app/controllers/projects/application_controller.rb
+++ b/app/controllers/projects/application_controller.rb
@@ -21,13 +21,12 @@ def redirect_git_extension
   end
 
   def project
-    @project ||= find_routable!(Project,
-      File.join(params[:namespace_id], params[:project_id] || params[:id]),
-      extra_authorization_proc: project_not_being_deleted?)
-  end
+    return @project if @project
+
+    path = File.join(params[:namespace_id], params[:project_id] || params[:id])
+    auth_proc = ->(project) { !project.pending_delete? }
 
-  def project_not_being_deleted?
-    ->(project) { !project.pending_delete? }
+    @project = find_routable!(Project, path, extra_authorization_proc: auth_proc)
   end
 
   def repository
diff --git a/app/models/route.rb b/app/models/route.rb
index 3d798ce937ba1..b34cce9077ac1 100644
--- a/app/models/route.rb
+++ b/app/models/route.rb
@@ -16,7 +16,7 @@ class Route < ActiveRecord::Base
   scope :direct_descendant_routes, -> (path) { where('routes.path LIKE ? AND routes.path NOT LIKE ?', "#{sanitize_sql_like(path)}/%", "#{sanitize_sql_like(path)}/%/%") }
 
   def rename_direct_descendant_routes
-    return if !path_changed? && !name_changed?
+    return unless path_changed? || name_changed?
 
     direct_descendant_routes = self.class.direct_descendant_routes(path_was)
 
-- 
GitLab


From f1d48c25a2fa53995ed9fa491a72193bfc83be32 Mon Sep 17 00:00:00 2001
From: Clement Ho <ClemMakesApps@gmail.com>
Date: Fri, 5 May 2017 14:45:14 -0500
Subject: [PATCH 336/363] Fix failing spec and eslint

---
 .../user_mock_data_helper.js}                 |  2 +-
 spec/javascripts/sidebar/assignees_spec.js    | 12 +++++------
 spec/javascripts/sidebar/mock_data.js         |  2 +-
 .../sidebar/sidebar_mediator_spec.js          | 10 ++++++----
 .../sidebar/sidebar_service_spec.js           | 20 +++++++++++--------
 .../javascripts/sidebar/sidebar_store_spec.js |  2 +-
 6 files changed, 27 insertions(+), 21 deletions(-)
 rename spec/javascripts/{sidebar/user_mock_data.js => helpers/user_mock_data_helper.js} (75%)

diff --git a/spec/javascripts/sidebar/user_mock_data.js b/spec/javascripts/helpers/user_mock_data_helper.js
similarity index 75%
rename from spec/javascripts/sidebar/user_mock_data.js
rename to spec/javascripts/helpers/user_mock_data_helper.js
index 9e7b834623878..a9783ea065cf7 100644
--- a/spec/javascripts/sidebar/user_mock_data.js
+++ b/spec/javascripts/helpers/user_mock_data_helper.js
@@ -4,7 +4,7 @@ export default {
     for (let i = 0; i < numberUsers; i = i += 1) {
       users.push(
         {
-          avatarUrl: 'http://gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+          avatar: 'http://gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
           id: (i + 1),
           name: `GitLab User ${i}`,
           username: `gitlab${i}`,
diff --git a/spec/javascripts/sidebar/assignees_spec.js b/spec/javascripts/sidebar/assignees_spec.js
index cf47940ef1c18..c9453a2118972 100644
--- a/spec/javascripts/sidebar/assignees_spec.js
+++ b/spec/javascripts/sidebar/assignees_spec.js
@@ -1,7 +1,7 @@
 import Vue from 'vue';
 import Assignee from '~/sidebar/components/assignees/assignees';
 import UsersMock from './mock_data';
-import UsersMockHelper from '../test_helpers/user_mock_data';
+import UsersMockHelper from '../helpers/user_mock_data_helper';
 
 describe('Assignee component', () => {
   let component;
@@ -86,7 +86,7 @@ describe('Assignee component', () => {
       const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
       const assignee = collapsed.children[0];
       expect(collapsed.childElementCount).toEqual(1);
-      expect(assignee.querySelector('.avatar').getAttribute('src')).toEqual(UsersMock.user.avatarUrl);
+      expect(assignee.querySelector('.avatar').getAttribute('src')).toEqual(UsersMock.user.avatar);
       expect(assignee.querySelector('.avatar').getAttribute('alt')).toEqual(`${UsersMock.user.name}'s avatar`);
       expect(assignee.querySelector('.author').innerText.trim()).toEqual(UsersMock.user.name);
     });
@@ -104,7 +104,7 @@ describe('Assignee component', () => {
 
       expect(component.$el.querySelector('.author_link')).not.toBeNull();
       // The image
-      expect(component.$el.querySelector('.author_link img').getAttribute('src')).toEqual(UsersMock.user.avatarUrl);
+      expect(component.$el.querySelector('.author_link img').getAttribute('src')).toEqual(UsersMock.user.avatar);
       // Author name
       expect(component.$el.querySelector('.author_link .author').innerText.trim()).toEqual(UsersMock.user.name);
       // Username
@@ -141,12 +141,12 @@ describe('Assignee component', () => {
       expect(collapsed.childElementCount).toEqual(2);
 
       const first = collapsed.children[0];
-      expect(first.querySelector('.avatar').getAttribute('src')).toEqual(users[0].avatarUrl);
+      expect(first.querySelector('.avatar').getAttribute('src')).toEqual(users[0].avatar);
       expect(first.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[0].name}'s avatar`);
       expect(first.querySelector('.author').innerText.trim()).toEqual(users[0].name);
 
       const second = collapsed.children[1];
-      expect(second.querySelector('.avatar').getAttribute('src')).toEqual(users[1].avatarUrl);
+      expect(second.querySelector('.avatar').getAttribute('src')).toEqual(users[1].avatar);
       expect(second.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[1].name}'s avatar`);
       expect(second.querySelector('.author').innerText.trim()).toEqual(users[1].name);
     });
@@ -165,7 +165,7 @@ describe('Assignee component', () => {
       expect(collapsed.childElementCount).toEqual(2);
 
       const first = collapsed.children[0];
-      expect(first.querySelector('.avatar').getAttribute('src')).toEqual(users[0].avatarUrl);
+      expect(first.querySelector('.avatar').getAttribute('src')).toEqual(users[0].avatar);
       expect(first.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[0].name}'s avatar`);
       expect(first.querySelector('.author').innerText.trim()).toEqual(users[0].name);
 
diff --git a/spec/javascripts/sidebar/mock_data.js b/spec/javascripts/sidebar/mock_data.js
index 0599a63912c71..9fc8667ecc9a7 100644
--- a/spec/javascripts/sidebar/mock_data.js
+++ b/spec/javascripts/sidebar/mock_data.js
@@ -93,7 +93,7 @@ export default {
     human_total_time_spent: null,
   },
   user: {
-    avatarUrl: 'http://gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+    avatar: 'http://gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
     id: 1,
     name: 'Administrator',
     username: 'root',
diff --git a/spec/javascripts/sidebar/sidebar_mediator_spec.js b/spec/javascripts/sidebar/sidebar_mediator_spec.js
index 9bfca0c099103..2b00fa173346b 100644
--- a/spec/javascripts/sidebar/sidebar_mediator_spec.js
+++ b/spec/javascripts/sidebar/sidebar_mediator_spec.js
@@ -24,10 +24,12 @@ describe('Sidebar mediator', () => {
   });
 
   it('saves assignees', (done) => {
-    this.mediator.saveAssignees('issue[assignee_ids]').then((resp) => {
-      expect(resp.status).toEqual(200);
-      done();
-    });
+    this.mediator.saveAssignees('issue[assignee_ids]')
+      .then((resp) => {
+        expect(resp.status).toEqual(200);
+        done();
+      })
+      .catch(() => {});
   });
 
   it('fetches the data', () => {
diff --git a/spec/javascripts/sidebar/sidebar_service_spec.js b/spec/javascripts/sidebar/sidebar_service_spec.js
index ed7be76dbcc8d..d41162096a68f 100644
--- a/spec/javascripts/sidebar/sidebar_service_spec.js
+++ b/spec/javascripts/sidebar/sidebar_service_spec.js
@@ -13,16 +13,20 @@ describe('Sidebar service', () => {
   });
 
   it('gets the data', (done) => {
-    this.service.get().then((resp) => {
-      expect(resp).toBeDefined();
-      done();
-    });
+    this.service.get()
+      .then((resp) => {
+        expect(resp).toBeDefined();
+        done();
+      })
+      .catch(() => {});
   });
 
   it('updates the data', (done) => {
-    this.service.update('issue[assignee_ids]', [1]).then((resp) => {
-      expect(resp).toBeDefined();
-      done();
-    });
+    this.service.update('issue[assignee_ids]', [1])
+      .then((resp) => {
+        expect(resp).toBeDefined();
+        done();
+      })
+      .catch(() => {});
   });
 });
diff --git a/spec/javascripts/sidebar/sidebar_store_spec.js b/spec/javascripts/sidebar/sidebar_store_spec.js
index 6c0f2b6a793f8..29facf483b54e 100644
--- a/spec/javascripts/sidebar/sidebar_store_spec.js
+++ b/spec/javascripts/sidebar/sidebar_store_spec.js
@@ -1,6 +1,6 @@
 import SidebarStore from '~/sidebar/stores/sidebar_store';
 import Mock from './mock_data';
-import UsersMockHelper from '../test_helpers/user_mock_data';
+import UsersMockHelper from '../helpers/user_mock_data_helper';
 
 describe('Sidebar store', () => {
   const assignee = {
-- 
GitLab


From b0ee22609a89572d6e3f98eebccf9fb2335dd939 Mon Sep 17 00:00:00 2001
From: Michael Kozono <mkozono@gmail.com>
Date: Fri, 5 May 2017 14:31:33 -0700
Subject: [PATCH 337/363] Reduce risk of deadlocks
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

We’ve seen a deadlock in CI here https://gitlab.com/mkozono/gitlab-ce/builds/15644492#down-build-trace. This commit should not fix that particular failure, but perhaps it will avoid others.

* Don’t call delete_conflicting_redirects after update if the path wasn’t changed
* Rename descendants without using recursion again, so we can run delete_conflicting_redirects exactly once.
---
 app/models/route.rb       | 24 +++++++++++++++++-------
 spec/models/route_spec.rb | 15 +--------------
 2 files changed, 18 insertions(+), 21 deletions(-)

diff --git a/app/models/route.rb b/app/models/route.rb
index b34cce9077ac1..12a7fa3d01bfb 100644
--- a/app/models/route.rb
+++ b/app/models/route.rb
@@ -8,19 +8,19 @@ class Route < ActiveRecord::Base
     presence: true,
     uniqueness: { case_sensitive: false }
 
-  after_save :delete_conflicting_redirects
+  after_create :delete_conflicting_redirects
+  after_update :delete_conflicting_redirects, if: :path_changed?
   after_update :create_redirect_for_old_path
-  after_update :rename_direct_descendant_routes
+  after_update :rename_descendants
 
   scope :inside_path, -> (path) { where('routes.path LIKE ?', "#{sanitize_sql_like(path)}/%") }
-  scope :direct_descendant_routes, -> (path) { where('routes.path LIKE ? AND routes.path NOT LIKE ?', "#{sanitize_sql_like(path)}/%", "#{sanitize_sql_like(path)}/%/%") }
 
-  def rename_direct_descendant_routes
+  def rename_descendants
     return unless path_changed? || name_changed?
 
-    direct_descendant_routes = self.class.direct_descendant_routes(path_was)
+    descendant_routes = self.class.inside_path(path_was)
 
-    direct_descendant_routes.each do |route|
+    descendant_routes.each do |route|
       attributes = {}
 
       if path_changed? && route.path.present?
@@ -31,7 +31,17 @@ def rename_direct_descendant_routes
         attributes[:name] = route.name.sub(name_was, name)
       end
 
-      route.update(attributes) unless attributes.empty?
+      if attributes.present?
+        old_path = route.path
+
+        # Callbacks must be run manually
+        route.update_columns(attributes)
+
+        # We are not calling route.delete_conflicting_redirects here, in hopes
+        # of avoiding deadlocks. The parent (self, in this method) already
+        # called it, which deletes conflicts for all descendants.
+        route.create_redirect(old_path) if attributes[:path]
+      end
     end
   end
 
diff --git a/spec/models/route_spec.rb b/spec/models/route_spec.rb
index 1aeddcef982b3..c1fe1b06c52b8 100644
--- a/spec/models/route_spec.rb
+++ b/spec/models/route_spec.rb
@@ -50,20 +50,7 @@
     end
   end
 
-  describe '.direct_descendant_routes' do
-    let!(:nested_group) { create(:group, path: 'test', name: 'test', parent: group) }
-    let!(:deep_nested_group) { create(:group, path: 'foo', name: 'foo', parent: nested_group) }
-    let!(:another_group) { create(:group, path: 'other') }
-    let!(:similar_group) { create(:group, path: 'gitllab') }
-    let!(:another_group_nested) { create(:group, path: 'another', name: 'another', parent: similar_group) }
-
-    it 'returns correct routes' do
-      expect(Route.direct_descendant_routes('git_lab')).to match_array([nested_group.route])
-      expect(Route.direct_descendant_routes('git_lab/test')).to match_array([deep_nested_group.route])
-    end
-  end
-
-  describe '#rename_direct_descendant_routes' do
+  describe '#rename_descendants' do
     let!(:nested_group) { create(:group, path: 'test', name: 'test', parent: group) }
     let!(:deep_nested_group) { create(:group, path: 'foo', name: 'foo', parent: nested_group) }
     let!(:similar_group) { create(:group, path: 'gitlab-org', name: 'gitlab-org') }
-- 
GitLab


From b801f414aa6516ed6cd816220adff2f844175b91 Mon Sep 17 00:00:00 2001
From: Kushal Pandya <kushalspandya@gmail.com>
Date: Fri, 5 May 2017 21:41:55 +0000
Subject: [PATCH 338/363] Fix Karma failures for jQuery deferreds

---
 app/assets/javascripts/notes.js |   2 +-
 spec/javascripts/notes_spec.js  | 217 +++++++++++++-------------------
 2 files changed, 89 insertions(+), 130 deletions(-)

diff --git a/app/assets/javascripts/notes.js b/app/assets/javascripts/notes.js
index 72709f68070d9..55391ebc089a2 100644
--- a/app/assets/javascripts/notes.js
+++ b/app/assets/javascripts/notes.js
@@ -1356,7 +1356,7 @@ const normalizeNewlines = function(str) {
 
       // Show updated comment content temporarily
       $noteBodyText.html(formContent);
-      $editingNote.removeClass('is-editing').addClass('being-posted fade-in-half');
+      $editingNote.removeClass('is-editing fade-in-full').addClass('being-posted fade-in-half');
       $editingNote.find('.note-headline-meta a').html('<i class="fa fa-spinner fa-spin" aria-label="Comment is being updated" aria-hidden="true"></i>');
 
       /* eslint-disable promise/catch-or-return */
diff --git a/spec/javascripts/notes_spec.js b/spec/javascripts/notes_spec.js
index 7bffa90ab145e..cfd599f793e73 100644
--- a/spec/javascripts/notes_spec.js
+++ b/spec/javascripts/notes_spec.js
@@ -29,7 +29,7 @@ import '~/notes';
         $('.js-comment-button').on('click', function(e) {
           e.preventDefault();
         });
-        this.notes = new Notes();
+        this.notes = new Notes('', []);
       });
 
       it('modifies the Markdown field', function() {
@@ -51,7 +51,7 @@ import '~/notes';
       var textarea = '.js-note-text';
 
       beforeEach(function() {
-        this.notes = new Notes();
+        this.notes = new Notes('', []);
 
         this.autoSizeSpy = spyOnEvent($(textarea), 'autosize:update');
         spyOn(this.notes, 'renderNote').and.stub();
@@ -273,9 +273,92 @@ import '~/notes';
       });
     });
 
+    describe('postComment & updateComment', () => {
+      const sampleComment = 'foo';
+      const updatedComment = 'bar';
+      const note = {
+        id: 1234,
+        html: `<li class="note note-row-1234 timeline-entry" id="note_1234">
+                <div class="note-text">${sampleComment}</div>
+               </li>`,
+        note: sampleComment,
+        valid: true
+      };
+      let $form;
+      let $notesContainer;
+
+      beforeEach(() => {
+        this.notes = new Notes('', []);
+        window.gon.current_username = 'root';
+        window.gon.current_user_fullname = 'Administrator';
+        $form = $('form.js-main-target-form');
+        $notesContainer = $('ul.main-notes-list');
+        $form.find('textarea.js-note-text').val(sampleComment);
+      });
+
+      it('should show placeholder note while new comment is being posted', () => {
+        $('.js-comment-button').click();
+        expect($notesContainer.find('.note.being-posted').length > 0).toEqual(true);
+      });
+
+      it('should remove placeholder note when new comment is done posting', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.resolve(note);
+        expect($notesContainer.find('.note.being-posted').length).toEqual(0);
+      });
+
+      it('should show actual note element when new comment is done posting', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.resolve(note);
+        expect($notesContainer.find(`#note_${note.id}`).length > 0).toEqual(true);
+      });
+
+      it('should reset Form when new comment is done posting', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.resolve(note);
+        expect($form.find('textarea.js-note-text').val()).toEqual('');
+      });
+
+      it('should show flash error message when new comment failed to be posted', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.reject();
+        expect($notesContainer.parent().find('.flash-container .flash-text').is(':visible')).toEqual(true);
+      });
+
+      it('should show flash error message when comment failed to be updated', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.resolve(note);
+        const $noteEl = $notesContainer.find(`#note_${note.id}`);
+        $noteEl.find('.js-note-edit').click();
+        $noteEl.find('textarea.js-note-text').val(updatedComment);
+        $noteEl.find('.js-comment-save-button').click();
+
+        deferred.reject();
+        const $updatedNoteEl = $notesContainer.find(`#note_${note.id}`);
+        expect($updatedNoteEl.hasClass('.being-posted')).toEqual(false); // Remove being-posted visuals
+        expect($updatedNoteEl.find('.note-text').text().trim()).toEqual(sampleComment); // See if comment reverted back to original
+        expect($('.flash-container').is(':visible')).toEqual(true); // Flash error message shown
+      });
+    });
+
     describe('getFormData', () => {
       it('should return form metadata object from form reference', () => {
-        this.notes = new Notes();
+        this.notes = new Notes('', []);
 
         const $form = $('form');
         const sampleComment = 'foobar';
@@ -290,7 +373,7 @@ import '~/notes';
 
     describe('hasSlashCommands', () => {
       beforeEach(() => {
-        this.notes = new Notes();
+        this.notes = new Notes('', []);
       });
 
       it('should return true when comment has slash commands', () => {
@@ -327,7 +410,7 @@ import '~/notes';
       const currentUserFullname = 'Administrator';
 
       beforeEach(() => {
-        this.notes = new Notes();
+        this.notes = new Notes('', []);
       });
 
       it('should return constructed placeholder element for regular note based on form contents', () => {
@@ -364,129 +447,5 @@ import '~/notes';
         expect($tempNote.find('.timeline-content').hasClass('discussion')).toBeTruthy();
       });
     });
-
-    describe('postComment & updateComment', () => {
-      const sampleComment = 'foo';
-      const updatedComment = 'bar';
-      const note = {
-        id: 1234,
-        html: `<li class="note note-row-1234 timeline-entry" id="note_1234">
-                <div class="note-text">${sampleComment}</div>
-               </li>`,
-        note: sampleComment,
-        valid: true
-      };
-      let $form;
-      let $notesContainer;
-
-      beforeEach(() => {
-        this.notes = new Notes();
-        window.gon.current_username = 'root';
-        window.gon.current_user_fullname = 'Administrator';
-        $form = $('form');
-        $notesContainer = $('ul.main-notes-list');
-        $form.find('textarea.js-note-text').val(sampleComment);
-        $('.js-comment-button').click();
-      });
-
-      it('should show placeholder note while new comment is being posted', () => {
-        expect($notesContainer.find('.note.being-posted').length > 0).toEqual(true);
-      });
-
-      it('should remove placeholder note when new comment is done posting', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.success(note);
-          expect($notesContainer.find('.note.being-posted').length).toEqual(0);
-        });
-      });
-
-      it('should show actual note element when new comment is done posting', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.success(note);
-          expect($notesContainer.find(`#${note.id}`).length > 0).toEqual(true);
-        });
-      });
-
-      it('should reset Form when new comment is done posting', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.success(note);
-          expect($form.find('textarea.js-note-text')).toEqual('');
-        });
-      });
-
-      it('should trigger ajax:success event on Form when new comment is done posting', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.success(note);
-          spyOn($form, 'trigger');
-          expect($form.trigger).toHaveBeenCalledWith('ajax:success', [note]);
-        });
-      });
-
-      it('should show flash error message when new comment failed to be posted', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.error();
-          expect($notesContainer.parent().find('.flash-container .flash-text').is(':visible')).toEqual(true);
-        });
-      });
-
-      it('should refill form textarea with original comment content when new comment failed to be posted', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.error();
-          expect($form.find('textarea.js-note-text')).toEqual(sampleComment);
-        });
-      });
-
-      it('should show updated comment as _actively being posted_ while comment being updated', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.success(note);
-          const $noteEl = $notesContainer.find(`#note_${note.id}`);
-          $noteEl.find('.js-note-edit').click();
-          $noteEl.find('textarea.js-note-text').val(updatedComment);
-          $noteEl.find('.js-comment-save-button').click();
-          expect($noteEl.hasClass('.being-posted')).toEqual(true);
-          expect($noteEl.find('.note-text').text()).toEqual(updatedComment);
-        });
-      });
-
-      it('should show updated comment when comment update is done posting', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.success(note);
-          const $noteEl = $notesContainer.find(`#note_${note.id}`);
-          $noteEl.find('.js-note-edit').click();
-          $noteEl.find('textarea.js-note-text').val(updatedComment);
-          $noteEl.find('.js-comment-save-button').click();
-
-          spyOn($, 'ajax').and.callFake((updateOptions) => {
-            const updatedNote = Object.assign({}, note);
-            updatedNote.note = updatedComment;
-            updatedNote.html = `<li class="note note-row-1234 timeline-entry" id="note_1234">
-                                  <div class="note-text">${updatedComment}</div>
-                                </li>`;
-            updateOptions.success(updatedNote);
-            const $updatedNoteEl = $notesContainer.find(`#note_${updatedNote.id}`);
-            expect($updatedNoteEl.hasClass('.being-posted')).toEqual(false); // Remove being-posted visuals
-            expect($updatedNoteEl.find('note-text').text().trim()).toEqual(updatedComment); // Verify if comment text updated
-          });
-        });
-      });
-
-      it('should show flash error message when comment failed to be updated', () => {
-        spyOn($, 'ajax').and.callFake((options) => {
-          options.success(note);
-          const $noteEl = $notesContainer.find(`#note_${note.id}`);
-          $noteEl.find('.js-note-edit').click();
-          $noteEl.find('textarea.js-note-text').val(updatedComment);
-          $noteEl.find('.js-comment-save-button').click();
-
-          spyOn($, 'ajax').and.callFake((updateOptions) => {
-            updateOptions.error();
-            const $updatedNoteEl = $notesContainer.find(`#note_${note.id}`);
-            expect($updatedNoteEl.hasClass('.being-posted')).toEqual(false); // Remove being-posted visuals
-            expect($updatedNoteEl.find('note-text').text().trim()).toEqual(sampleComment); // See if comment reverted back to original
-            expect($notesContainer.parent().find('.flash-container .flash-text').is(':visible')).toEqual(true); // Flash error message shown
-          });
-        });
-      });
-    });
   });
 }).call(window);
-- 
GitLab


From b51f2a60800829ee7585f10451a41bc4db0359a9 Mon Sep 17 00:00:00 2001
From: Winnie Hellmann <winnie@gitlab.com>
Date: Fri, 5 May 2017 22:47:32 +0000
Subject: [PATCH 339/363] Colorize labels in issue search field

---
 .../filtered_search_visual_tokens.js          |  40 +++++-
 .../javascripts/lib/utils/ajax_cache.js       |  32 +++++
 app/assets/stylesheets/framework/filters.scss |  14 +-
 .../stylesheets/framework/variables.scss      |   2 +
 .../dashboard/labels_controller.rb            |   2 +-
 app/controllers/groups/labels_controller.rb   |   2 +-
 app/controllers/projects/labels_controller.rb |   2 +-
 app/serializers/label_entity.rb               |   1 +
 app/serializers/label_serializer.rb           |   7 +
 .../unreleased/winh-visual-token-labels.yml   |   4 +
 .../filtered_search_visual_tokens_spec.js     | 101 ++++++++++++++
 spec/javascripts/fixtures/labels.rb           |  56 ++++++++
 spec/javascripts/lib/utils/ajax_cache_spec.js | 129 ++++++++++++++++++
 spec/serializers/label_serializer_spec.rb     |  46 +++++++
 14 files changed, 431 insertions(+), 7 deletions(-)
 create mode 100644 app/assets/javascripts/lib/utils/ajax_cache.js
 create mode 100644 app/serializers/label_serializer.rb
 create mode 100644 changelogs/unreleased/winh-visual-token-labels.yml
 create mode 100644 spec/javascripts/fixtures/labels.rb
 create mode 100644 spec/javascripts/lib/utils/ajax_cache_spec.js
 create mode 100644 spec/serializers/label_serializer_spec.rb

diff --git a/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js b/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
index 59ce0587e1e40..f3003b86493ea 100644
--- a/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
+++ b/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
@@ -1,3 +1,5 @@
+import AjaxCache from '~/lib/utils/ajax_cache';
+import '~/flash'; /* global Flash */
 import FilteredSearchContainer from './container';
 
 class FilteredSearchVisualTokens {
@@ -48,6 +50,40 @@ class FilteredSearchVisualTokens {
     `;
   }
 
+  static updateLabelTokenColor(tokenValueContainer, tokenValue) {
+    const filteredSearchInput = FilteredSearchContainer.container.querySelector('.filtered-search');
+    const baseEndpoint = filteredSearchInput.dataset.baseEndpoint;
+    const labelsEndpoint = `${baseEndpoint}/labels.json`;
+
+    return AjaxCache.retrieve(labelsEndpoint)
+    .then((labels) => {
+      const matchingLabel = (labels || []).find(label => `~${gl.DropdownUtils.getEscapedText(label.title)}` === tokenValue);
+
+      if (!matchingLabel) {
+        return;
+      }
+
+      const tokenValueStyle = tokenValueContainer.style;
+      tokenValueStyle.backgroundColor = matchingLabel.color;
+      tokenValueStyle.color = matchingLabel.text_color;
+
+      if (matchingLabel.text_color === '#FFFFFF') {
+        const removeToken = tokenValueContainer.querySelector('.remove-token');
+        removeToken.classList.add('inverted');
+      }
+    })
+    .catch(() => new Flash('An error occurred while fetching label colors.'));
+  }
+
+  static renderVisualTokenValue(parentElement, tokenName, tokenValue) {
+    const tokenValueContainer = parentElement.querySelector('.value-container');
+    tokenValueContainer.querySelector('.value').innerText = tokenValue;
+
+    if (tokenName.toLowerCase() === 'label') {
+      FilteredSearchVisualTokens.updateLabelTokenColor(tokenValueContainer, tokenValue);
+    }
+  }
+
   static addVisualTokenElement(name, value, isSearchTerm) {
     const li = document.createElement('li');
     li.classList.add('js-visual-token');
@@ -55,7 +91,7 @@ class FilteredSearchVisualTokens {
 
     if (value) {
       li.innerHTML = FilteredSearchVisualTokens.createVisualTokenElementHTML();
-      li.querySelector('.value').innerText = value;
+      FilteredSearchVisualTokens.renderVisualTokenValue(li, name, value);
     } else {
       li.innerHTML = '<div class="name"></div>';
     }
@@ -74,7 +110,7 @@ class FilteredSearchVisualTokens {
       const name = FilteredSearchVisualTokens.getLastTokenPartial();
       lastVisualToken.innerHTML = FilteredSearchVisualTokens.createVisualTokenElementHTML();
       lastVisualToken.querySelector('.name').innerText = name;
-      lastVisualToken.querySelector('.value').innerText = value;
+      FilteredSearchVisualTokens.renderVisualTokenValue(lastVisualToken, name, value);
     }
   }
 
diff --git a/app/assets/javascripts/lib/utils/ajax_cache.js b/app/assets/javascripts/lib/utils/ajax_cache.js
new file mode 100644
index 0000000000000..d99eefb5089b2
--- /dev/null
+++ b/app/assets/javascripts/lib/utils/ajax_cache.js
@@ -0,0 +1,32 @@
+const AjaxCache = {
+  internalStorage: { },
+  get(endpoint) {
+    return this.internalStorage[endpoint];
+  },
+  hasData(endpoint) {
+    return Object.prototype.hasOwnProperty.call(this.internalStorage, endpoint);
+  },
+  purge(endpoint) {
+    delete this.internalStorage[endpoint];
+  },
+  retrieve(endpoint) {
+    if (AjaxCache.hasData(endpoint)) {
+      return Promise.resolve(AjaxCache.get(endpoint));
+    }
+
+    return new Promise((resolve, reject) => {
+      $.ajax(endpoint) // eslint-disable-line promise/catch-or-return
+      .then(data => resolve(data),
+        (jqXHR, textStatus, errorThrown) => {
+          const error = new Error(`${endpoint}: ${errorThrown}`);
+          error.textStatus = textStatus;
+          reject(error);
+        },
+      );
+    })
+    .then((data) => { this.internalStorage[endpoint] = data; })
+    .then(() => AjaxCache.get(endpoint));
+  },
+};
+
+export default AjaxCache;
diff --git a/app/assets/stylesheets/framework/filters.scss b/app/assets/stylesheets/framework/filters.scss
index 0692f65043bcf..e624d0d951eb1 100644
--- a/app/assets/stylesheets/framework/filters.scss
+++ b/app/assets/stylesheets/framework/filters.scss
@@ -114,11 +114,21 @@
     padding-right: 8px;
 
     .fa-close {
-      color: $gl-text-color-disabled;
+      color: $gl-text-color-secondary;
     }
 
     &:hover .fa-close {
-      color: $gl-text-color-secondary;
+      color: $gl-text-color;
+    }
+
+    &.inverted {
+      .fa-close {
+        color: $gl-text-color-secondary-inverted;
+      }
+
+      &:hover .fa-close {
+        color: $gl-text-color-inverted;
+      }
     }
   }
 
diff --git a/app/assets/stylesheets/framework/variables.scss b/app/assets/stylesheets/framework/variables.scss
index 49741c963df50..08bcb58261317 100644
--- a/app/assets/stylesheets/framework/variables.scss
+++ b/app/assets/stylesheets/framework/variables.scss
@@ -101,6 +101,8 @@ $gl-font-size: 14px;
 $gl-text-color: rgba(0, 0, 0, .85);
 $gl-text-color-secondary: rgba(0, 0, 0, .55);
 $gl-text-color-disabled: rgba(0, 0, 0, .35);
+$gl-text-color-inverted: rgba(255, 255, 255, 1.0);
+$gl-text-color-secondary-inverted: rgba(255, 255, 255, .85);
 $gl-text-green: $green-600;
 $gl-text-red: $red-500;
 $gl-text-orange: $orange-600;
diff --git a/app/controllers/dashboard/labels_controller.rb b/app/controllers/dashboard/labels_controller.rb
index d5031da867af6..dd1d46a68c71c 100644
--- a/app/controllers/dashboard/labels_controller.rb
+++ b/app/controllers/dashboard/labels_controller.rb
@@ -3,7 +3,7 @@ def index
     labels = LabelsFinder.new(current_user).execute
 
     respond_to do |format|
-      format.json { render json: labels.as_json(only: [:id, :title, :color]) }
+      format.json { render json: LabelSerializer.new.represent_appearance(labels) }
     end
   end
 end
diff --git a/app/controllers/groups/labels_controller.rb b/app/controllers/groups/labels_controller.rb
index facb25525b5ef..3fa0516fb0ce4 100644
--- a/app/controllers/groups/labels_controller.rb
+++ b/app/controllers/groups/labels_controller.rb
@@ -15,7 +15,7 @@ def index
 
       format.json do
         available_labels = LabelsFinder.new(current_user, group_id: @group.id).execute
-        render json: available_labels.as_json(only: [:id, :title, :color])
+        render json: LabelSerializer.new.represent_appearance(available_labels)
       end
     end
   end
diff --git a/app/controllers/projects/labels_controller.rb b/app/controllers/projects/labels_controller.rb
index 2f55ba4e700b0..71bfb7163da14 100644
--- a/app/controllers/projects/labels_controller.rb
+++ b/app/controllers/projects/labels_controller.rb
@@ -19,7 +19,7 @@ def index
     respond_to do |format|
       format.html
       format.json do
-        render json: @available_labels.as_json(only: [:id, :title, :color])
+        render json: LabelSerializer.new.represent_appearance(@available_labels)
       end
     end
   end
diff --git a/app/serializers/label_entity.rb b/app/serializers/label_entity.rb
index 304fd9de08f8e..ad565654342ba 100644
--- a/app/serializers/label_entity.rb
+++ b/app/serializers/label_entity.rb
@@ -6,6 +6,7 @@ class LabelEntity < Grape::Entity
   expose :group_id
   expose :project_id
   expose :template
+  expose :text_color
   expose :created_at
   expose :updated_at
 end
diff --git a/app/serializers/label_serializer.rb b/app/serializers/label_serializer.rb
new file mode 100644
index 0000000000000..ad6ba8c46c99e
--- /dev/null
+++ b/app/serializers/label_serializer.rb
@@ -0,0 +1,7 @@
+class LabelSerializer < BaseSerializer
+  entity LabelEntity
+
+  def represent_appearance(resource)
+    represent(resource, { only: [:id, :title, :color, :text_color] })
+  end
+end
diff --git a/changelogs/unreleased/winh-visual-token-labels.yml b/changelogs/unreleased/winh-visual-token-labels.yml
new file mode 100644
index 0000000000000..d4952e910b4c5
--- /dev/null
+++ b/changelogs/unreleased/winh-visual-token-labels.yml
@@ -0,0 +1,4 @@
+---
+title: Colorize labels in search field
+merge_request: 11047
+author:
diff --git a/spec/javascripts/filtered_search/filtered_search_visual_tokens_spec.js b/spec/javascripts/filtered_search/filtered_search_visual_tokens_spec.js
index d75b90612815e..8b750561eb7b5 100644
--- a/spec/javascripts/filtered_search/filtered_search_visual_tokens_spec.js
+++ b/spec/javascripts/filtered_search/filtered_search_visual_tokens_spec.js
@@ -1,3 +1,5 @@
+import AjaxCache from '~/lib/utils/ajax_cache';
+
 require('~/filtered_search/filtered_search_visual_tokens');
 const FilteredSearchSpecHelper = require('../helpers/filtered_search_spec_helper');
 
@@ -611,4 +613,103 @@ describe('Filtered Search Visual Tokens', () => {
       expect(token.querySelector('.value').innerText).toEqual('~bug');
     });
   });
+
+  describe('renderVisualTokenValue', () => {
+    let searchTokens;
+
+    beforeEach(() => {
+      tokensContainer.innerHTML = FilteredSearchSpecHelper.createTokensContainerHTML(`
+        ${FilteredSearchSpecHelper.createFilterVisualTokenHTML('label', 'none')}
+        ${FilteredSearchSpecHelper.createSearchVisualTokenHTML('search')}
+        ${FilteredSearchSpecHelper.createFilterVisualTokenHTML('milestone', 'upcoming')}
+      `);
+
+      searchTokens = document.querySelectorAll('.filtered-search-token');
+    });
+
+    it('renders a token value element', () => {
+      spyOn(gl.FilteredSearchVisualTokens, 'updateLabelTokenColor');
+      const updateLabelTokenColorSpy = gl.FilteredSearchVisualTokens.updateLabelTokenColor;
+
+      expect(searchTokens.length).toBe(2);
+      Array.prototype.forEach.call(searchTokens, (token) => {
+        updateLabelTokenColorSpy.calls.reset();
+
+        const tokenName = token.querySelector('.name').innerText;
+        const tokenValue = 'new value';
+        gl.FilteredSearchVisualTokens.renderVisualTokenValue(token, tokenName, tokenValue);
+
+        const tokenValueElement = token.querySelector('.value');
+        expect(tokenValueElement.innerText).toBe(tokenValue);
+
+        if (tokenName.toLowerCase() === 'label') {
+          const tokenValueContainer = token.querySelector('.value-container');
+          expect(updateLabelTokenColorSpy.calls.count()).toBe(1);
+          const expectedArgs = [tokenValueContainer, tokenValue];
+          expect(updateLabelTokenColorSpy.calls.argsFor(0)).toEqual(expectedArgs);
+        } else {
+          expect(updateLabelTokenColorSpy.calls.count()).toBe(0);
+        }
+      });
+    });
+  });
+
+  describe('updateLabelTokenColor', () => {
+    const jsonFixtureName = 'labels/project_labels.json';
+    const dummyEndpoint = '/dummy/endpoint';
+
+    preloadFixtures(jsonFixtureName);
+    const labelData = getJSONFixture(jsonFixtureName);
+    const findLabel = tokenValue => labelData.find(
+      label => tokenValue === `~${gl.DropdownUtils.getEscapedText(label.title)}`,
+    );
+
+    const bugLabelToken = FilteredSearchSpecHelper.createFilterVisualToken('label', '~bug');
+    const missingLabelToken = FilteredSearchSpecHelper.createFilterVisualToken('label', '~doesnotexist');
+    const spaceLabelToken = FilteredSearchSpecHelper.createFilterVisualToken('label', '~"some space"');
+
+    const parseColor = (color) => {
+      const dummyElement = document.createElement('div');
+      dummyElement.style.color = color;
+      return dummyElement.style.color;
+    };
+
+    beforeEach(() => {
+      tokensContainer.innerHTML = FilteredSearchSpecHelper.createTokensContainerHTML(`
+        ${bugLabelToken.outerHTML}
+        ${missingLabelToken.outerHTML}
+        ${spaceLabelToken.outerHTML}
+      `);
+
+      const filteredSearchInput = document.querySelector('.filtered-search');
+      filteredSearchInput.dataset.baseEndpoint = dummyEndpoint;
+
+      AjaxCache.internalStorage = { };
+      AjaxCache.internalStorage[`${dummyEndpoint}/labels.json`] = labelData;
+    });
+
+    const testCase = (token, done) => {
+      const tokenValueContainer = token.querySelector('.value-container');
+      const tokenValue = token.querySelector('.value').innerText;
+      const label = findLabel(tokenValue);
+
+      gl.FilteredSearchVisualTokens.updateLabelTokenColor(tokenValueContainer, tokenValue)
+      .then(() => {
+        if (label) {
+          expect(tokenValueContainer.getAttribute('style')).not.toBe(null);
+          expect(tokenValueContainer.style.backgroundColor).toBe(parseColor(label.color));
+          expect(tokenValueContainer.style.color).toBe(parseColor(label.text_color));
+        } else {
+          expect(token).toBe(missingLabelToken);
+          expect(tokenValueContainer.getAttribute('style')).toBe(null);
+        }
+      })
+      .then(done)
+      .catch(fail);
+    };
+
+    it('updates the color of a label token', done => testCase(bugLabelToken, done));
+    it('updates the color of a label token with spaces', done => testCase(spaceLabelToken, done));
+    it('does not change color of a missing label', done => testCase(missingLabelToken, done));
+  });
 });
diff --git a/spec/javascripts/fixtures/labels.rb b/spec/javascripts/fixtures/labels.rb
new file mode 100644
index 0000000000000..2e4811b64a4b4
--- /dev/null
+++ b/spec/javascripts/fixtures/labels.rb
@@ -0,0 +1,56 @@
+require 'spec_helper'
+
+describe 'Labels (JavaScript fixtures)' do
+  include JavaScriptFixturesHelpers
+
+  let(:admin) { create(:admin) }
+  let(:group) { create(:group, name: 'frontend-fixtures-group' )}
+  let(:project) { create(:project_empty_repo, namespace: group, path: 'labels-project') }
+
+  let!(:project_label_bug) { create(:label, project: project, title: 'bug', color: '#FF0000') }
+  let!(:project_label_enhancement) { create(:label, project: project, title: 'enhancement', color: '#00FF00') }
+  let!(:project_label_feature) { create(:label, project: project, title: 'feature', color: '#0000FF') }
+
+  let!(:group_label_roses) { create(:group_label, group: group, title: 'roses', color: '#FF0000') }
+  let!(:groub_label_space) { create(:group_label, group: group, title: 'some space', color: '#FFFFFF') }
+  let!(:groub_label_violets) { create(:group_label, group: group, title: 'violets', color: '#0000FF') }
+
+  before(:all) do
+    clean_frontend_fixtures('labels/')
+  end
+
+  describe Groups::LabelsController, '(JavaScript fixtures)', type: :controller do
+    render_views
+
+    before(:each) do
+      sign_in(admin)
+    end
+
+    it 'labels/group_labels.json' do |example|
+      get :index,
+        group_id: group,
+        format: 'json'
+
+      expect(response).to be_success
+      store_frontend_fixture(response, example.description)
+    end
+  end
+
+  describe Projects::LabelsController, '(JavaScript fixtures)', type: :controller do
+    render_views
+
+    before(:each) do
+      sign_in(admin)
+    end
+
+    it 'labels/project_labels.json' do |example|
+      get :index,
+        namespace_id: group,
+        project_id: project,
+        format: 'json'
+
+      expect(response).to be_success
+      store_frontend_fixture(response, example.description)
+    end
+  end
+end
diff --git a/spec/javascripts/lib/utils/ajax_cache_spec.js b/spec/javascripts/lib/utils/ajax_cache_spec.js
new file mode 100644
index 0000000000000..7b466a11b9240
--- /dev/null
+++ b/spec/javascripts/lib/utils/ajax_cache_spec.js
@@ -0,0 +1,129 @@
+import AjaxCache from '~/lib/utils/ajax_cache';
+
+describe('AjaxCache', () => {
+  const dummyEndpoint = '/AjaxCache/dummyEndpoint';
+  const dummyResponse = {
+    important: 'dummy data',
+  };
+  let ajaxSpy = (url) => {
+    expect(url).toBe(dummyEndpoint);
+    const deferred = $.Deferred();
+    deferred.resolve(dummyResponse);
+    return deferred.promise();
+  };
+
+  beforeEach(() => {
+    AjaxCache.internalStorage = { };
+    spyOn(jQuery, 'ajax').and.callFake(url => ajaxSpy(url));
+  });
+
+  describe('#get', () => {
+    it('returns undefined if cache is empty', () => {
+      const data = AjaxCache.get(dummyEndpoint);
+
+      expect(data).toBe(undefined);
+    });
+
+    it('returns undefined if cache contains no matching data', () => {
+      AjaxCache.internalStorage['not matching'] = dummyResponse;
+
+      const data = AjaxCache.get(dummyEndpoint);
+
+      expect(data).toBe(undefined);
+    });
+
+    it('returns matching data', () => {
+      AjaxCache.internalStorage[dummyEndpoint] = dummyResponse;
+
+      const data = AjaxCache.get(dummyEndpoint);
+
+      expect(data).toBe(dummyResponse);
+    });
+  });
+
+  describe('#hasData', () => {
+    it('returns false if cache is empty', () => {
+      expect(AjaxCache.hasData(dummyEndpoint)).toBe(false);
+    });
+
+    it('returns false if cache contains no matching data', () => {
+      AjaxCache.internalStorage['not matching'] = dummyResponse;
+
+      expect(AjaxCache.hasData(dummyEndpoint)).toBe(false);
+    });
+
+    it('returns true if data is available', () => {
+      AjaxCache.internalStorage[dummyEndpoint] = dummyResponse;
+
+      expect(AjaxCache.hasData(dummyEndpoint)).toBe(true);
+    });
+  });
+
+  describe('#purge', () => {
+    it('does nothing if cache is empty', () => {
+      AjaxCache.purge(dummyEndpoint);
+
+      expect(AjaxCache.internalStorage).toEqual({ });
+    });
+
+    it('does nothing if cache contains no matching data', () => {
+      AjaxCache.internalStorage['not matching'] = dummyResponse;
+
+      AjaxCache.purge(dummyEndpoint);
+
+      expect(AjaxCache.internalStorage['not matching']).toBe(dummyResponse);
+    });
+
+    it('removes matching data', () => {
+      AjaxCache.internalStorage[dummyEndpoint] = dummyResponse;
+
+      AjaxCache.purge(dummyEndpoint);
+
+      expect(AjaxCache.internalStorage).toEqual({ });
+    });
+  });
+
+  describe('#retrieve', () => {
+    it('stores and returns data from Ajax call if cache is empty', (done) => {
+      AjaxCache.retrieve(dummyEndpoint)
+      .then((data) => {
+        expect(data).toBe(dummyResponse);
+        expect(AjaxCache.internalStorage[dummyEndpoint]).toBe(dummyResponse);
+      })
+      .then(done)
+      .catch(fail);
+    });
+
+    it('returns undefined if Ajax call fails and cache is empty', (done) => {
+      const dummyStatusText = 'exploded';
+      const dummyErrorMessage = 'server exploded';
+      ajaxSpy = (url) => {
+        expect(url).toBe(dummyEndpoint);
+        const deferred = $.Deferred();
+        deferred.reject(null, dummyStatusText, dummyErrorMessage);
+        return deferred.promise();
+      };
+
+      AjaxCache.retrieve(dummyEndpoint)
+      .then(data => fail(`Received unexpected data: ${JSON.stringify(data)}`))
+      .catch((error) => {
+        expect(error.message).toBe(`${dummyEndpoint}: ${dummyErrorMessage}`);
+        expect(error.textStatus).toBe(dummyStatusText);
+        done();
+      })
+      .catch(fail);
+    });
+
+    it('makes no Ajax call if matching data exists', (done) => {
+      AjaxCache.internalStorage[dummyEndpoint] = dummyResponse;
+      ajaxSpy = () => fail(new Error('expected no Ajax call!'));
+
+      AjaxCache.retrieve(dummyEndpoint)
+      .then((data) => {
+        expect(data).toBe(dummyResponse);
+      })
+      .then(done)
+      .catch(fail);
+    });
+  });
+});
diff --git a/spec/serializers/label_serializer_spec.rb b/spec/serializers/label_serializer_spec.rb
new file mode 100644
index 0000000000000..c58c7da1f9ed5
--- /dev/null
+++ b/spec/serializers/label_serializer_spec.rb
@@ -0,0 +1,46 @@
+require 'spec_helper'
+
+describe LabelSerializer do
+  let(:user) { create(:user) }
+
+  let(:serializer) do
+    described_class.new(user: user)
+  end
+
+  subject { serializer.represent(resource) }
+
+  describe '#represent' do
+    context 'when a single object is being serialized' do
+      let(:resource) { create(:label) }
+
+      it 'serializes the label object' do
+        expect(subject[:id]).to eq resource.id
+      end
+    end
+
+    context 'when multiple objects are being serialized' do
+      let(:num_labels) { 2 }
+      let(:resource) { create_list(:label, num_labels) }
+
+      it 'serializes the array of labels' do
+        expect(subject.size).to eq(num_labels)
+      end
+    end
+  end
+
+  describe '#represent_appearance' do
+    context 'when represents only appearance' do
+      let(:resource) { create(:label) }
+
+      subject { serializer.represent_appearance(resource) }
+
+      it 'serializes only attributes used for appearance' do
+        expect(subject.keys).to eq([:id, :title, :color, :text_color])
+        expect(subject[:id]).to eq(resource.id)
+        expect(subject[:title]).to eq(resource.title)
+        expect(subject[:color]).to eq(resource.color)
+        expect(subject[:text_color]).to eq(resource.text_color)
+      end
+    end
+  end
+end
-- 
GitLab


From 61d0ac55d47b92668ce95b1e800e879f95327a48 Mon Sep 17 00:00:00 2001
From: Ruben Davila <rdavila84@gmail.com>
Date: Fri, 5 May 2017 20:50:48 -0500
Subject: [PATCH 340/363] Use an absolute path for locale path in FastGettext
 config

Unicorn was unable to start due to this bad config on Omnibus.
---
 config/initializers/fast_gettext.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/config/initializers/fast_gettext.rb b/config/initializers/fast_gettext.rb
index bea85b43b9547..a69fe0c902e80 100644
--- a/config/initializers/fast_gettext.rb
+++ b/config/initializers/fast_gettext.rb
@@ -1,4 +1,4 @@
-FastGettext.add_text_domain 'gitlab', path: 'locale', type: :po
+FastGettext.add_text_domain 'gitlab', path: File.join(Rails.root, 'locale'), type: :po
 FastGettext.default_text_domain = 'gitlab'
 FastGettext.default_available_locales = Gitlab::I18n.available_locales
 
-- 
GitLab


From b69bb08bf5a653695a2f4fb8b224924b42bc42c3 Mon Sep 17 00:00:00 2001
From: Mike Greiling <mike@pixelcog.com>
Date: Fri, 5 May 2017 22:41:53 -0500
Subject: [PATCH 341/363] add tooltips to user contrib graph key

---
 app/assets/javascripts/users/calendar.js | 26 ++++++++++++++++--------
 1 file changed, 17 insertions(+), 9 deletions(-)

diff --git a/app/assets/javascripts/users/calendar.js b/app/assets/javascripts/users/calendar.js
index 754d448564fef..32ffa2f0ac08c 100644
--- a/app/assets/javascripts/users/calendar.js
+++ b/app/assets/javascripts/users/calendar.js
@@ -168,15 +168,23 @@ import d3 from 'd3';
     };
 
     Calendar.prototype.renderKey = function() {
-      var keyColors;
-      keyColors = ['#ededed', this.colorKey(0), this.colorKey(1), this.colorKey(2), this.colorKey(3)];
-      return this.svg.append('g').attr('transform', "translate(18, " + (this.daySizeWithSpace * 8 + 16) + ")").selectAll('rect').data(keyColors).enter().append('rect').attr('width', this.daySize).attr('height', this.daySize).attr('x', (function(_this) {
-        return function(color, i) {
-          return _this.daySizeWithSpace * i;
-        };
-      })(this)).attr('y', 0).attr('fill', function(color) {
-        return color;
-      });
+      const keyValues = ['no contributions', '1-9 contributions', '10-19 contributions', '20-29 contributions', '30+ contributions'];
+      const keyColors = ['#ededed', this.colorKey(0), this.colorKey(1), this.colorKey(2), this.colorKey(3)];
+
+      this.svg.append('g')
+        .attr('transform', `translate(18, ${this.daySizeWithSpace * 8 + 16})`)
+        .selectAll('rect')
+          .data(keyColors)
+          .enter()
+          .append('rect')
+            .attr('width', this.daySize)
+            .attr('height', this.daySize)
+            .attr('x', (color, i) => this.daySizeWithSpace * i)
+            .attr('y', 0)
+            .attr('fill', color => color)
+            .attr('class', 'js-tooltip')
+            .attr('title', (color, i) => keyValues[i])
+            .attr('data-container', 'body');
     };
 
     Calendar.prototype.initColor = function() {
-- 
GitLab


From 22bd020028a9e7cadafce01705dcd189cf14f945 Mon Sep 17 00:00:00 2001
From: Mike Greiling <mike@pixelcog.com>
Date: Fri, 5 May 2017 22:43:11 -0500
Subject: [PATCH 342/363] add CHANGELOG.md entry for !11138

---
 .../unreleased/23751-add-contribution-graph-key-tooltips.yml  | 4 ++++
 1 file changed, 4 insertions(+)
 create mode 100644 changelogs/unreleased/23751-add-contribution-graph-key-tooltips.yml

diff --git a/changelogs/unreleased/23751-add-contribution-graph-key-tooltips.yml b/changelogs/unreleased/23751-add-contribution-graph-key-tooltips.yml
new file mode 100644
index 0000000000000..7c4c6fb46a057
--- /dev/null
+++ b/changelogs/unreleased/23751-add-contribution-graph-key-tooltips.yml
@@ -0,0 +1,4 @@
+---
+title: Add tooltips to user contribution graph key
+merge_request: 11138
+author:
-- 
GitLab


From da0d8e0491d993e0b5c7905c360b71fa930612d5 Mon Sep 17 00:00:00 2001
From: Sean McGivern <sean@gitlab.com>
Date: Sat, 6 May 2017 09:45:25 +0100
Subject: [PATCH 343/363] Fix `Routable.find_by_full_path` on MySQL

---
 app/models/concerns/routable.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/models/concerns/routable.rb b/app/models/concerns/routable.rb
index e351dbb45ddb8..c4463abdfe6bf 100644
--- a/app/models/concerns/routable.rb
+++ b/app/models/concerns/routable.rb
@@ -49,7 +49,7 @@ def find_by_full_path(path, follow_redirects: false)
         if Gitlab::Database.postgresql?
           joins(:redirect_routes).find_by("LOWER(redirect_routes.path) = LOWER(?)", path)
         else
-          joins(:redirect_routes).find_by(path: path)
+          joins(:redirect_routes).find_by(redirect_routes: { path: path })
         end
       end
     end
-- 
GitLab


From c17e6a6c68b0412b3433632802b852db474a7b30 Mon Sep 17 00:00:00 2001
From: Zeger-Jan van de Weg <zegerjan@gitlab.com>
Date: Sat, 6 May 2017 16:45:46 +0000
Subject: [PATCH 344/363] Real time pipeline show action

---
 .../projects/pipelines_controller.rb          |  22 +-
 app/models/ci/group.rb                        |  40 ++
 app/models/ci/stage.rb                        |   8 +
 app/serializers/job_group_entity.rb           |  16 +
 app/serializers/stage_entity.rb               |   8 +-
 app/serializers/status_entity.rb              |   7 +
 .../unreleased/zj-real-time-pipelines.yml     |   4 +
 lib/gitlab/ci/status/group/common.rb          |  21 ++
 lib/gitlab/ci/status/group/factory.rb         |  13 +
 lib/gitlab/etag_caching/router.rb             |   6 +-
 .../projects/pipelines_controller_spec.rb     |  31 ++
 spec/fixtures/api/schemas/pipeline.json       | 354 ++++++++++++++++++
 .../lib/gitlab/ci/status/group/common_spec.rb |  20 +
 .../gitlab/ci/status/group/factory_spec.rb    |  13 +
 spec/models/ci/group_spec.rb                  |  44 +++
 spec/models/ci/stage_spec.rb                  |  33 +-
 spec/serializers/stage_entity_spec.rb         |   8 +
 17 files changed, 637 insertions(+), 11 deletions(-)
 create mode 100644 app/models/ci/group.rb
 create mode 100644 app/serializers/job_group_entity.rb
 create mode 100644 changelogs/unreleased/zj-real-time-pipelines.yml
 create mode 100644 lib/gitlab/ci/status/group/common.rb
 create mode 100644 lib/gitlab/ci/status/group/factory.rb
 create mode 100644 spec/fixtures/api/schemas/pipeline.json
 create mode 100644 spec/lib/gitlab/ci/status/group/common_spec.rb
 create mode 100644 spec/lib/gitlab/ci/status/group/factory_spec.rb
 create mode 100644 spec/models/ci/group_spec.rb

diff --git a/app/controllers/projects/pipelines_controller.rb b/app/controllers/projects/pipelines_controller.rb
index f9adedcb074b8..5cb2e42820151 100644
--- a/app/controllers/projects/pipelines_controller.rb
+++ b/app/controllers/projects/pipelines_controller.rb
@@ -8,6 +8,8 @@ class Projects::PipelinesController < Projects::ApplicationController
 
   wrap_parameters Ci::Pipeline
 
+  POLLING_INTERVAL = 10_000
+
   def index
     @scope = params[:scope]
     @pipelines = PipelinesFinder
@@ -31,7 +33,7 @@ def index
     respond_to do |format|
       format.html
       format.json do
-        Gitlab::PollingInterval.set_header(response, interval: 10_000)
+        Gitlab::PollingInterval.set_header(response, interval: POLLING_INTERVAL)
 
         render json: {
           pipelines: PipelineSerializer
@@ -57,15 +59,25 @@ def create
     @pipeline = Ci::CreatePipelineService
       .new(project, current_user, create_params)
       .execute(ignore_skip_ci: true, save_on_errors: false)
-    unless @pipeline.persisted?
+
+    if @pipeline.persisted?
+      redirect_to namespace_project_pipeline_path(project.namespace, project, @pipeline)
+    else
       render 'new'
-      return
     end
-
-    redirect_to namespace_project_pipeline_path(project.namespace, project, @pipeline)
   end
 
   def show
+    respond_to do |format|
+      format.html
+      format.json do
+        Gitlab::PollingInterval.set_header(response, interval: POLLING_INTERVAL)
+
+        render json: PipelineSerializer
+          .new(project: @project, user: @current_user)
+          .represent(@pipeline, grouped: true)
+      end
+    end
   end
 
   def builds
diff --git a/app/models/ci/group.rb b/app/models/ci/group.rb
new file mode 100644
index 0000000000000..87898b086c6ad
--- /dev/null
+++ b/app/models/ci/group.rb
@@ -0,0 +1,40 @@
+module Ci
+  ##
+  # This domain model is a representation of a group of jobs that are related
+  # to each other, like `rspec 0 1`, `rspec 0 2`.
+  #
+  # It is not persisted in the database.
+  #
+  class Group
+    include StaticModel
+
+    attr_reader :stage, :name, :jobs
+
+    delegate :size, to: :jobs
+
+    def initialize(stage, name:, jobs:)
+      @stage = stage
+      @name = name
+      @jobs = jobs
+    end
+
+    def status
+      @status ||= commit_statuses.status
+    end
+
+    def detailed_status(current_user)
+      if jobs.one?
+        jobs.first.detailed_status(current_user)
+      else
+        Gitlab::Ci::Status::Group::Factory
+          .new(self, current_user).fabricate!
+      end
+    end
+
+    private
+
+    def commit_statuses
+      @commit_statuses ||= CommitStatus.where(id: jobs.map(&:id))
+    end
+  end
+end
diff --git a/app/models/ci/stage.rb b/app/models/ci/stage.rb
index e7d6b17d445ef..9bda3186c3040 100644
--- a/app/models/ci/stage.rb
+++ b/app/models/ci/stage.rb
@@ -15,6 +15,14 @@ def initialize(pipeline, name:, status: nil, warnings: nil)
       @warnings = warnings
     end
 
+    def groups
+      @groups ||= statuses.ordered.latest
+        .sort_by(&:sortable_name).group_by(&:group_name)
+        .map do |group_name, grouped_statuses|
+          Ci::Group.new(self, name: group_name, jobs: grouped_statuses)
+        end
+    end
+
     def to_param
       name
     end
diff --git a/app/serializers/job_group_entity.rb b/app/serializers/job_group_entity.rb
new file mode 100644
index 0000000000000..a4d3737429c4d
--- /dev/null
+++ b/app/serializers/job_group_entity.rb
@@ -0,0 +1,16 @@
+class JobGroupEntity < Grape::Entity
+  include RequestAwareEntity
+
+  expose :name
+  expose :size
+  expose :detailed_status, as: :status, with: StatusEntity
+  expose :jobs, with: BuildEntity
+
+  private
+
+  alias_method :group, :object
+
+  def detailed_status
+    group.detailed_status(request.user)
+  end
+end
diff --git a/app/serializers/stage_entity.rb b/app/serializers/stage_entity.rb
index 7a047bdc71238..97ced8730edf7 100644
--- a/app/serializers/stage_entity.rb
+++ b/app/serializers/stage_entity.rb
@@ -7,9 +7,11 @@ class StageEntity < Grape::Entity
     "#{stage.name}: #{detailed_status.label}"
   end
 
-  expose :detailed_status,
-    as: :status,
-    with: StatusEntity
+  expose :groups,
+    if: -> (_, opts) { opts[:grouped] },
+    with: JobGroupEntity
+
+  expose :detailed_status, as: :status, with: StatusEntity
 
   expose :path do |stage|
     namespace_project_pipeline_path(
diff --git a/app/serializers/status_entity.rb b/app/serializers/status_entity.rb
index 188c3747f184f..3e40ecf1c1c09 100644
--- a/app/serializers/status_entity.rb
+++ b/app/serializers/status_entity.rb
@@ -12,4 +12,11 @@ class StatusEntity < Grape::Entity
 
     ActionController::Base.helpers.image_path(File.join(dir, "#{status.favicon}.ico"))
   end
+
+  expose :action, if: -> (status, _) { status.has_action? } do
+    expose :action_icon, as: :icon
+    expose :action_title, as: :title
+    expose :action_path, as: :path
+    expose :action_method, as: :method
+  end
 end
diff --git a/changelogs/unreleased/zj-real-time-pipelines.yml b/changelogs/unreleased/zj-real-time-pipelines.yml
new file mode 100644
index 0000000000000..eec22e6746705
--- /dev/null
+++ b/changelogs/unreleased/zj-real-time-pipelines.yml
@@ -0,0 +1,4 @@
+---
+title: Pipeline view updates in near real time
+merge_request: 10777
+author:
diff --git a/lib/gitlab/ci/status/group/common.rb b/lib/gitlab/ci/status/group/common.rb
new file mode 100644
index 0000000000000..cfd4329a92335
--- /dev/null
+++ b/lib/gitlab/ci/status/group/common.rb
@@ -0,0 +1,21 @@
+module Gitlab
+  module Ci
+    module Status
+      module Group
+        module Common
+          def has_details?
+            false
+          end
+
+          def details_path
+            nil
+          end
+
+          def has_action?
+            false
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/lib/gitlab/ci/status/group/factory.rb b/lib/gitlab/ci/status/group/factory.rb
new file mode 100644
index 0000000000000..d118116cfc362
--- /dev/null
+++ b/lib/gitlab/ci/status/group/factory.rb
@@ -0,0 +1,13 @@
+module Gitlab
+  module Ci
+    module Status
+      module Group
+        class Factory < Status::Factory
+          def self.common_helpers
+            Status::Group::Common
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/lib/gitlab/etag_caching/router.rb b/lib/gitlab/etag_caching/router.rb
index aac210f19e838..692c909d83821 100644
--- a/lib/gitlab/etag_caching/router.rb
+++ b/lib/gitlab/etag_caching/router.rb
@@ -36,7 +36,11 @@ class Router
         Gitlab::EtagCaching::Router::Route.new(
           %r(^(?!.*(#{RESERVED_WORDS_REGEX})).*/pipelines\.json\z),
           'project_pipelines'
-        )
+        ),
+        Gitlab::EtagCaching::Router::Route.new(
+          %r(^(?!.*(#{RESERVED_WORDS})).*/pipelines/\d+\.json\z),
+          'project_pipeline'
+        ),
       ].freeze
 
       def self.match(env)
diff --git a/spec/controllers/projects/pipelines_controller_spec.rb b/spec/controllers/projects/pipelines_controller_spec.rb
index 1b47d163c0b3d..fb4a4721a5879 100644
--- a/spec/controllers/projects/pipelines_controller_spec.rb
+++ b/spec/controllers/projects/pipelines_controller_spec.rb
@@ -1,6 +1,8 @@
 require 'spec_helper'
 
 describe Projects::PipelinesController do
+  include ApiHelpers
+
   let(:user) { create(:user) }
   let(:project) { create(:empty_project, :public) }
 
@@ -24,6 +26,7 @@
 
     it 'returns JSON with serialized pipelines' do
       expect(response).to have_http_status(:ok)
+      expect(response).to match_response_schema('pipeline')
 
       expect(json_response).to include('pipelines')
       expect(json_response['pipelines'].count).to eq 4
@@ -34,6 +37,34 @@
     end
   end
 
+  describe 'GET show JSON' do
+    let!(:pipeline) { create(:ci_pipeline_with_one_job, project: project) }
+
+    it 'returns the pipeline' do
+      get_pipeline_json
+
+      expect(response).to have_http_status(:ok)
+      expect(json_response).not_to be_an(Array)
+      expect(json_response['id']).to be(pipeline.id)
+      expect(json_response['details']).to have_key 'stages'
+    end
+
+    context 'when the pipeline has multiple jobs' do
+      it 'does not perform N + 1 queries' do
+        control_count = ActiveRecord::QueryRecorder.new { get_pipeline_json }.count
+
+        create(:ci_build, pipeline: pipeline)
+
+        # The plus 2 is needed to group and sort
+        expect { get_pipeline_json }.not_to exceed_query_limit(control_count + 2)
+      end
+    end
+
+    def get_pipeline_json
+      get :show, namespace_id: project.namespace, project_id: project, id: pipeline, format: :json
+    end
+  end
+
   describe 'GET stages.json' do
     let(:pipeline) { create(:ci_pipeline, project: project) }
 
diff --git a/spec/fixtures/api/schemas/pipeline.json b/spec/fixtures/api/schemas/pipeline.json
new file mode 100644
index 0000000000000..55511d17b5e6d
--- /dev/null
+++ b/spec/fixtures/api/schemas/pipeline.json
@@ -0,0 +1,354 @@
+{
+    "$schema": "http://json-schema.org/draft-04/schema#",
+    "definitions": {},
+    "id": "http://example.com/example.json",
+    "properties": {
+        "commit": {
+            "id": "/properties/commit",
+            "properties": {
+                "author": {
+                    "id": "/properties/commit/properties/author",
+                    "type": "null"
+                },
+                "author_email": {
+                    "id": "/properties/commit/properties/author_email",
+                    "type": "string"
+                },
+                "author_gravatar_url": {
+                    "id": "/properties/commit/properties/author_gravatar_url",
+                    "type": "string"
+                },
+                "author_name": {
+                    "id": "/properties/commit/properties/author_name",
+                    "type": "string"
+                },
+                "authored_date": {
+                    "id": "/properties/commit/properties/authored_date",
+                    "type": "string"
+                },
+                "commit_path": {
+                    "id": "/properties/commit/properties/commit_path",
+                    "type": "string"
+                },
+                "commit_url": {
+                    "id": "/properties/commit/properties/commit_url",
+                    "type": "string"
+                },
+                "committed_date": {
+                    "id": "/properties/commit/properties/committed_date",
+                    "type": "string"
+                },
+                "committer_email": {
+                    "id": "/properties/commit/properties/committer_email",
+                    "type": "string"
+                },
+                "committer_name": {
+                    "id": "/properties/commit/properties/committer_name",
+                    "type": "string"
+                },
+                "created_at": {
+                    "id": "/properties/commit/properties/created_at",
+                    "type": "string"
+                },
+                "id": {
+                    "id": "/properties/commit/properties/id",
+                    "type": "string"
+                },
+                "message": {
+                    "id": "/properties/commit/properties/message",
+                    "type": "string"
+                },
+                "parent_ids": {
+                    "id": "/properties/commit/properties/parent_ids",
+                    "items": {
+                        "id": "/properties/commit/properties/parent_ids/items",
+                        "type": "string"
+                    },
+                    "type": "array"
+                },
+                "short_id": {
+                    "id": "/properties/commit/properties/short_id",
+                    "type": "string"
+                },
+                "title": {
+                    "id": "/properties/commit/properties/title",
+                    "type": "string"
+                }
+            },
+            "type": "object"
+        },
+        "created_at": {
+            "id": "/properties/created_at",
+            "type": "string"
+        },
+        "details": {
+            "id": "/properties/details",
+            "properties": {
+                "artifacts": {
+                    "id": "/properties/details/properties/artifacts",
+                    "items": {},
+                    "type": "array"
+                },
+                "duration": {
+                    "id": "/properties/details/properties/duration",
+                    "type": "integer"
+                },
+                "finished_at": {
+                    "id": "/properties/details/properties/finished_at",
+                    "type": "string"
+                },
+                "manual_actions": {
+                    "id": "/properties/details/properties/manual_actions",
+                    "items": {},
+                    "type": "array"
+                },
+                "stages": {
+                    "id": "/properties/details/properties/stages",
+                    "items": {
+                        "id": "/properties/details/properties/stages/items",
+                        "properties": {
+                            "dropdown_path": {
+                                "id": "/properties/details/properties/stages/items/properties/dropdown_path",
+                                "type": "string"
+                            },
+                            "groups": {
+                                "id": "/properties/details/properties/stages/items/properties/groups",
+                                "items": {
+                                    "id": "/properties/details/properties/stages/items/properties/groups/items",
+                                    "properties": {
+                                        "name": {
+                                            "id": "/properties/details/properties/stages/items/properties/groups/items/properties/name",
+                                            "type": "string"
+                                        },
+                                        "size": {
+                                            "id": "/properties/details/properties/stages/items/properties/groups/items/properties/size",
+                                            "type": "integer"
+                                        },
+                                        "status": {
+                                            "id": "/properties/details/properties/stages/items/properties/groups/items/properties/status",
+                                            "properties": {
+                                                "details_path": {
+                                                    "id": "/properties/details/properties/stages/items/properties/groups/items/properties/status/properties/details_path",
+                                                    "type": "null"
+                                                },
+                                                "favicon": {
+                                                    "id": "/properties/details/properties/stages/items/properties/groups/items/properties/status/properties/favicon",
+                                                    "type": "string"
+                                                },
+                                                "group": {
+                                                    "id": "/properties/details/properties/stages/items/properties/groups/items/properties/status/properties/group",
+                                                    "type": "string"
+                                                },
+                                                "has_details": {
+                                                    "id": "/properties/details/properties/stages/items/properties/groups/items/properties/status/properties/has_details",
+                                                    "type": "boolean"
+                                                },
+                                                "icon": {
+                                                    "id": "/properties/details/properties/stages/items/properties/groups/items/properties/status/properties/icon",
+                                                    "type": "string"
+                                                },
+                                                "label": {
+                                                    "id": "/properties/details/properties/stages/items/properties/groups/items/properties/status/properties/label",
+                                                    "type": "string"
+                                                },
+                                                "text": {
+                                                    "id": "/properties/details/properties/stages/items/properties/groups/items/properties/status/properties/text",
+                                                    "type": "string"
+                                                }
+                                            },
+                                            "type": "object"
+                                        }
+                                    },
+                                    "type": "object"
+                                },
+                                "type": "array"
+                            },
+                            "name": {
+                                "id": "/properties/details/properties/stages/items/properties/name",
+                                "type": "string"
+                            },
+                            "path": {
+                                "id": "/properties/details/properties/stages/items/properties/path",
+                                "type": "string"
+                            },
+                            "status": {
+                                "id": "/properties/details/properties/stages/items/properties/status",
+                                "properties": {
+                                    "details_path": {
+                                        "id": "/properties/details/properties/stages/items/properties/status/properties/details_path",
+                                        "type": "string"
+                                    },
+                                    "favicon": {
+                                        "id": "/properties/details/properties/stages/items/properties/status/properties/favicon",
+                                        "type": "string"
+                                    },
+                                    "group": {
+                                        "id": "/properties/details/properties/stages/items/properties/status/properties/group",
+                                        "type": "string"
+                                    },
+                                    "has_details": {
+                                        "id": "/properties/details/properties/stages/items/properties/status/properties/has_details",
+                                        "type": "boolean"
+                                    },
+                                    "icon": {
+                                        "id": "/properties/details/properties/stages/items/properties/status/properties/icon",
+                                        "type": "string"
+                                    },
+                                    "label": {
+                                        "id": "/properties/details/properties/stages/items/properties/status/properties/label",
+                                        "type": "string"
+                                    },
+                                    "text": {
+                                        "id": "/properties/details/properties/stages/items/properties/status/properties/text",
+                                        "type": "string"
+                                    }
+                                },
+                                "type": "object"
+                            },
+                            "title": {
+                                "id": "/properties/details/properties/stages/items/properties/title",
+                                "type": "string"
+                            }
+                        },
+                        "type": "object"
+                    },
+                    "type": "array"
+                },
+                "status": {
+                    "id": "/properties/details/properties/status",
+                    "properties": {
+                        "details_path": {
+                            "id": "/properties/details/properties/status/properties/details_path",
+                            "type": "string"
+                        },
+                        "favicon": {
+                            "id": "/properties/details/properties/status/properties/favicon",
+                            "type": "string"
+                        },
+                        "group": {
+                            "id": "/properties/details/properties/status/properties/group",
+                            "type": "string"
+                        },
+                        "has_details": {
+                            "id": "/properties/details/properties/status/properties/has_details",
+                            "type": "boolean"
+                        },
+                        "icon": {
+                            "id": "/properties/details/properties/status/properties/icon",
+                            "type": "string"
+                        },
+                        "label": {
+                            "id": "/properties/details/properties/status/properties/label",
+                            "type": "string"
+                        },
+                        "text": {
+                            "id": "/properties/details/properties/status/properties/text",
+                            "type": "string"
+                        }
+                    },
+                    "type": "object"
+                }
+            },
+            "type": "object"
+        },
+        "flags": {
+            "id": "/properties/flags",
+            "properties": {
+                "cancelable": {
+                    "id": "/properties/flags/properties/cancelable",
+                    "type": "boolean"
+                },
+                "latest": {
+                    "id": "/properties/flags/properties/latest",
+                    "type": "boolean"
+                },
+                "retryable": {
+                    "id": "/properties/flags/properties/retryable",
+                    "type": "boolean"
+                },
+                "stuck": {
+                    "id": "/properties/flags/properties/stuck",
+                    "type": "boolean"
+                },
+                "triggered": {
+                    "id": "/properties/flags/properties/triggered",
+                    "type": "boolean"
+                },
+                "yaml_errors": {
+                    "id": "/properties/flags/properties/yaml_errors",
+                    "type": "boolean"
+                }
+            },
+            "type": "object"
+        },
+        "id": {
+            "id": "/properties/id",
+            "type": "integer"
+        },
+        "path": {
+            "id": "/properties/path",
+            "type": "string"
+        },
+        "ref": {
+            "id": "/properties/ref",
+            "properties": {
+                "branch": {
+                    "id": "/properties/ref/properties/branch",
+                    "type": "boolean"
+                },
+                "name": {
+                    "id": "/properties/ref/properties/name",
+                    "type": "string"
+                },
+                "path": {
+                    "id": "/properties/ref/properties/path",
+                    "type": "string"
+                },
+                "tag": {
+                    "id": "/properties/ref/properties/tag",
+                    "type": "boolean"
+                }
+            },
+            "type": "object"
+        },
+        "retry_path": {
+            "id": "/properties/retry_path",
+            "type": "string"
+        },
+        "updated_at": {
+            "id": "/properties/updated_at",
+            "type": "string"
+        },
+        "user": {
+            "id": "/properties/user",
+            "properties": {
+                "avatar_url": {
+                    "id": "/properties/user/properties/avatar_url",
+                    "type": "string"
+                },
+                "id": {
+                    "id": "/properties/user/properties/id",
+                    "type": "integer"
+                },
+                "name": {
+                    "id": "/properties/user/properties/name",
+                    "type": "string"
+                },
+                "state": {
+                    "id": "/properties/user/properties/state",
+                    "type": "string"
+                },
+                "username": {
+                    "id": "/properties/user/properties/username",
+                    "type": "string"
+                },
+                "web_url": {
+                    "id": "/properties/user/properties/web_url",
+                    "type": "string"
+                }
+            },
+            "type": "object"
+        }
+    },
+    "type": "object"
+}
diff --git a/spec/lib/gitlab/ci/status/group/common_spec.rb b/spec/lib/gitlab/ci/status/group/common_spec.rb
new file mode 100644
index 0000000000000..c0ca05881f5e2
--- /dev/null
+++ b/spec/lib/gitlab/ci/status/group/common_spec.rb
@@ -0,0 +1,20 @@
+require 'spec_helper'
+
+describe Gitlab::Ci::Status::Group::Common do
+  subject do
+    Gitlab::Ci::Status::Core.new(double, double)
+      .extend(described_class)
+  end
+
+  it 'does not have action' do
+    expect(subject).not_to have_action
+  end
+
+  it 'has details' do
+    expect(subject).not_to have_details
+  end
+
+  it 'has no details_path' do
+    expect(subject.details_path).to be_falsy
+  end
+end
diff --git a/spec/lib/gitlab/ci/status/group/factory_spec.rb b/spec/lib/gitlab/ci/status/group/factory_spec.rb
new file mode 100644
index 0000000000000..0cd8312393801
--- /dev/null
+++ b/spec/lib/gitlab/ci/status/group/factory_spec.rb
@@ -0,0 +1,13 @@
+require 'spec_helper'
+
+describe Gitlab::Ci::Status::Group::Factory do
+  it 'inherits from the core factory' do
+    expect(described_class)
+      .to be < Gitlab::Ci::Status::Factory
+  end
+
+  it 'exposes group helpers' do
+    expect(described_class.common_helpers)
+      .to eq Gitlab::Ci::Status::Group::Common
+  end
+end
diff --git a/spec/models/ci/group_spec.rb b/spec/models/ci/group_spec.rb
new file mode 100644
index 0000000000000..62e1509308985
--- /dev/null
+++ b/spec/models/ci/group_spec.rb
@@ -0,0 +1,44 @@
+require 'spec_helper'
+
+describe Ci::Group, models: true do
+  subject do
+    described_class.new('test', name: 'rspec', jobs: jobs)
+  end
+
+  let!(:jobs) { build_list(:ci_build, 1, :success) }
+
+  it { is_expected.to include_module(StaticModel) }
+
+  it { is_expected.to respond_to(:stage) }
+  it { is_expected.to respond_to(:name) }
+  it { is_expected.to respond_to(:jobs) }
+  it { is_expected.to respond_to(:status) }
+
+  describe '#size' do
+    it 'returns the number of statuses in the group' do
+      expect(subject.size).to eq(1)
+    end
+  end
+
+  describe '#detailed_status' do
+    context 'when there is only one item in the group' do
+      it 'calls the status from the object itself' do
+        expect(jobs.first).to receive(:detailed_status)
+
+        expect(subject.detailed_status(double(:user)))
+      end
+    end
+
+    context 'when there are more than one commit status in the group' do
+      let(:jobs) do
+        [create(:ci_build, :failed),
+         create(:ci_build, :success)]
+      end
+
+      it 'fabricates a new detailed status object' do
+        expect(subject.detailed_status(double(:user)))
+          .to be_a(Gitlab::Ci::Status::Failed)
+      end
+    end
+  end
+end
diff --git a/spec/models/ci/stage_spec.rb b/spec/models/ci/stage_spec.rb
index c38faf32f7df3..372b662fab2eb 100644
--- a/spec/models/ci/stage_spec.rb
+++ b/spec/models/ci/stage_spec.rb
@@ -28,6 +28,35 @@
     end
   end
 
+  describe '#groups' do
+    before do
+      create_job(:ci_build, name: 'rspec 0 2')
+      create_job(:ci_build, name: 'rspec 0 1')
+      create_job(:ci_build, name: 'spinach 0 1')
+      create_job(:commit_status, name: 'aaaaa')
+    end
+
+    it 'returns an array of three groups' do
+      expect(stage.groups).to be_a Array
+      expect(stage.groups).to all(be_a Ci::Group)
+      expect(stage.groups.size).to eq 3
+    end
+
+    it 'returns groups with correctly ordered statuses' do
+      expect(stage.groups.first.jobs.map(&:name))
+        .to eq ['aaaaa']
+      expect(stage.groups.second.jobs.map(&:name))
+        .to eq ['rspec 0 1', 'rspec 0 2']
+      expect(stage.groups.third.jobs.map(&:name))
+        .to eq ['spinach 0 1']
+    end
+
+    it 'returns groups with correct names' do
+      expect(stage.groups.map(&:name))
+        .to eq %w[aaaaa rspec spinach]
+    end
+  end
+
   describe '#statuses_count' do
     before do
       create_job(:ci_build)
@@ -223,7 +252,7 @@
     end
   end
 
-  def create_job(type, status: 'success', stage: stage_name)
-    create(type, pipeline: pipeline, stage: stage, status: status)
+  def create_job(type, status: 'success', stage: stage_name, **opts)
+    create(type, pipeline: pipeline, stage: stage, status: status, **opts)
   end
 end
diff --git a/spec/serializers/stage_entity_spec.rb b/spec/serializers/stage_entity_spec.rb
index 4ab40d08432a7..0412b2d774181 100644
--- a/spec/serializers/stage_entity_spec.rb
+++ b/spec/serializers/stage_entity_spec.rb
@@ -47,5 +47,13 @@
     it 'contains stage title' do
       expect(subject[:title]).to eq 'test: passed'
     end
+
+    context 'when the jobs should be grouped' do
+      let(:entity) { described_class.new(stage, request: request, grouped: true) }
+
+      it 'exposes the group key' do
+        expect(subject).to include :groups
+      end
+    end
   end
 end
-- 
GitLab


From aa440eb1c0947d2dc551c61abbd9d271b9002050 Mon Sep 17 00:00:00 2001
From: Kamil Trzcinski <ayufan@ayufan.eu>
Date: Sat, 6 May 2017 19:02:06 +0200
Subject: [PATCH 345/363] Single commit squash of all changes for
 https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10878

It's needed due to https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10777 being merged with squash.
---
 app/assets/javascripts/ci_status_icons.js     |  34 -----
 app/assets/javascripts/dispatcher.js          |   3 +-
 .../lib/utils/bootstrap_linked_tabs.js        | 116 ++++++++--------
 app/assets/javascripts/pipelines.js           |  46 ++-----
 .../components/graph/action_component.vue     |  59 ++++++++
 .../graph/dropdown_action_component.vue       |  56 ++++++++
 .../graph/dropdown_job_component.vue          |  86 ++++++++++++
 .../components/graph/graph_component.vue      |  92 +++++++++++++
 .../components/graph/job_component.vue        | 124 +++++++++++++++++
 .../components/graph/job_name_component.vue   |  37 +++++
 .../graph/stage_column_component.vue          |  64 +++++++++
 .../pipelines/components/stage.vue            |   6 +-
 .../javascripts/pipelines/graph_bundle.js     |  10 ++
 .../pipelines/services/pipeline_service.js    |  14 ++
 .../pipelines/stores/pipeline_store.js        |  11 ++
 .../javascripts/vue_shared/ci_action_icons.js |  22 +++
 .../javascripts/vue_shared/ci_status_icons.js |  55 ++++++++
 .../vue_shared/components/ci_icon.vue         |  29 ++++
 .../javascripts/vue_shared/mixins/tooltip.js  |   9 ++
 app/assets/stylesheets/pages/pipelines.scss   |  69 +++++-----
 app/views/ci/status/_graph_badge.html.haml    |  20 ---
 app/views/projects/pipelines/_graph.html.haml |   4 -
 .../projects/pipelines/_with_tabs.html.haml   |   7 +-
 app/views/projects/stage/_graph.html.haml     |  19 ---
 .../projects/stage/_in_stage_group.html.haml  |  14 --
 .../25226-realtime-pipelines-fe.yml           |   4 +
 config/webpack.config.js                      |   2 +
 .../javascripts/bootstrap_linked_tabs_spec.js |   8 +-
 spec/javascripts/ci_status_icon_spec.js       |  44 ------
 spec/javascripts/fixtures/graph.html.haml     |   1 +
 .../pipelines/graph/action_component_spec.js  |  40 ++++++
 .../graph/dropdown_action_component_spec.js   |  30 ++++
 .../pipelines/graph/graph_component_spec.js   |  83 +++++++++++
 .../pipelines/graph/job_component_spec.js     | 117 ++++++++++++++++
 .../graph/job_name_component_spec.js          |  27 ++++
 .../graph/stage_column_component_spec.js      |  42 ++++++
 spec/javascripts/pipelines_spec.js            |  32 ++---
 .../vue_shared/ci_action_icons_spec.js        |  22 +++
 .../vue_shared/ci_status_icon_spec.js         |  27 ++++
 .../vue_shared/components/ci_icon_spec.js     | 130 ++++++++++++++++++
 40 files changed, 1321 insertions(+), 294 deletions(-)
 delete mode 100644 app/assets/javascripts/ci_status_icons.js
 create mode 100644 app/assets/javascripts/pipelines/components/graph/action_component.vue
 create mode 100644 app/assets/javascripts/pipelines/components/graph/dropdown_action_component.vue
 create mode 100644 app/assets/javascripts/pipelines/components/graph/dropdown_job_component.vue
 create mode 100644 app/assets/javascripts/pipelines/components/graph/graph_component.vue
 create mode 100644 app/assets/javascripts/pipelines/components/graph/job_component.vue
 create mode 100644 app/assets/javascripts/pipelines/components/graph/job_name_component.vue
 create mode 100644 app/assets/javascripts/pipelines/components/graph/stage_column_component.vue
 create mode 100644 app/assets/javascripts/pipelines/graph_bundle.js
 create mode 100644 app/assets/javascripts/pipelines/services/pipeline_service.js
 create mode 100644 app/assets/javascripts/pipelines/stores/pipeline_store.js
 create mode 100644 app/assets/javascripts/vue_shared/ci_action_icons.js
 create mode 100644 app/assets/javascripts/vue_shared/ci_status_icons.js
 create mode 100644 app/assets/javascripts/vue_shared/components/ci_icon.vue
 create mode 100644 app/assets/javascripts/vue_shared/mixins/tooltip.js
 delete mode 100644 app/views/ci/status/_graph_badge.html.haml
 delete mode 100644 app/views/projects/pipelines/_graph.html.haml
 delete mode 100644 app/views/projects/stage/_graph.html.haml
 delete mode 100644 app/views/projects/stage/_in_stage_group.html.haml
 create mode 100644 changelogs/unreleased/25226-realtime-pipelines-fe.yml
 delete mode 100644 spec/javascripts/ci_status_icon_spec.js
 create mode 100644 spec/javascripts/fixtures/graph.html.haml
 create mode 100644 spec/javascripts/pipelines/graph/action_component_spec.js
 create mode 100644 spec/javascripts/pipelines/graph/dropdown_action_component_spec.js
 create mode 100644 spec/javascripts/pipelines/graph/graph_component_spec.js
 create mode 100644 spec/javascripts/pipelines/graph/job_component_spec.js
 create mode 100644 spec/javascripts/pipelines/graph/job_name_component_spec.js
 create mode 100644 spec/javascripts/pipelines/graph/stage_column_component_spec.js
 create mode 100644 spec/javascripts/vue_shared/ci_action_icons_spec.js
 create mode 100644 spec/javascripts/vue_shared/ci_status_icon_spec.js
 create mode 100644 spec/javascripts/vue_shared/components/ci_icon_spec.js

diff --git a/app/assets/javascripts/ci_status_icons.js b/app/assets/javascripts/ci_status_icons.js
deleted file mode 100644
index f16616873b2ec..0000000000000
--- a/app/assets/javascripts/ci_status_icons.js
+++ /dev/null
@@ -1,34 +0,0 @@
-import CANCELED_SVG from 'icons/_icon_status_canceled_borderless.svg';
-import CREATED_SVG from 'icons/_icon_status_created_borderless.svg';
-import FAILED_SVG from 'icons/_icon_status_failed_borderless.svg';
-import MANUAL_SVG from 'icons/_icon_status_manual_borderless.svg';
-import PENDING_SVG from 'icons/_icon_status_pending_borderless.svg';
-import RUNNING_SVG from 'icons/_icon_status_running_borderless.svg';
-import SKIPPED_SVG from 'icons/_icon_status_skipped_borderless.svg';
-import SUCCESS_SVG from 'icons/_icon_status_success_borderless.svg';
-import WARNING_SVG from 'icons/_icon_status_warning_borderless.svg';
-
-const StatusIconEntityMap = {
-  icon_status_canceled: CANCELED_SVG,
-  icon_status_created: CREATED_SVG,
-  icon_status_failed: FAILED_SVG,
-  icon_status_manual: MANUAL_SVG,
-  icon_status_pending: PENDING_SVG,
-  icon_status_running: RUNNING_SVG,
-  icon_status_skipped: SKIPPED_SVG,
-  icon_status_success: SUCCESS_SVG,
-  icon_status_warning: WARNING_SVG,
-};
-
-export {
-  CANCELED_SVG,
-  CREATED_SVG,
-  FAILED_SVG,
-  MANUAL_SVG,
-  PENDING_SVG,
-  RUNNING_SVG,
-  SKIPPED_SVG,
-  SUCCESS_SVG,
-  WARNING_SVG,
-  StatusIconEntityMap as default,
-};
diff --git a/app/assets/javascripts/dispatcher.js b/app/assets/javascripts/dispatcher.js
index b16ff2a022128..d27d89cf91db7 100644
--- a/app/assets/javascripts/dispatcher.js
+++ b/app/assets/javascripts/dispatcher.js
@@ -49,6 +49,7 @@ import BlobForkSuggestion from './blob/blob_fork_suggestion';
 import UserCallout from './user_callout';
 import { ProtectedTagCreate, ProtectedTagEditList } from './protected_tags';
 import ShortcutsWiki from './shortcuts_wiki';
+import Pipelines from './pipelines';
 import BlobViewer from './blob/viewer/index';
 import AutoWidthDropdownSelect from './issuable/auto_width_dropdown_select';
 
@@ -257,7 +258,7 @@ const ShortcutsBlob = require('./shortcuts_blob');
           const { controllerAction } = document.querySelector('.js-pipeline-container').dataset;
           const pipelineStatusUrl = `${document.querySelector('.js-pipeline-tab-link a').getAttribute('href')}/status.json`;
 
-          new gl.Pipelines({
+          new Pipelines({
             initTabs: true,
             pipelineStatusUrl,
             tabsOptions: {
diff --git a/app/assets/javascripts/lib/utils/bootstrap_linked_tabs.js b/app/assets/javascripts/lib/utils/bootstrap_linked_tabs.js
index 2955bda1a3626..0bf2ba6acc268 100644
--- a/app/assets/javascripts/lib/utils/bootstrap_linked_tabs.js
+++ b/app/assets/javascripts/lib/utils/bootstrap_linked_tabs.js
@@ -31,82 +31,78 @@
  *
  * ### How to use
  *
- *  new window.gl.LinkedTabs({
+ *  new LinkedTabs({
  *    action: "#{controller.action_name}",
  *    defaultAction: 'tab1',
  *    parentEl: '.tab-links'
  *  });
  */
 
-(() => {
-  window.gl = window.gl || {};
+export default class LinkedTabs {
+  /**
+   * Binds the events and activates de default tab.
+   *
+   * @param  {Object} options
+   */
+  constructor(options = {}) {
+    this.options = options;
 
-  window.gl.LinkedTabs = class LinkedTabs {
-    /**
-     * Binds the events and activates de default tab.
-     *
-     * @param  {Object} options
-     */
-    constructor(options) {
-      this.options = options || {};
+    this.defaultAction = this.options.defaultAction;
+    this.action = this.options.action || this.defaultAction;
 
-      this.defaultAction = this.options.defaultAction;
-      this.action = this.options.action || this.defaultAction;
-
-      if (this.action === 'show') {
-        this.action = this.defaultAction;
-      }
+    if (this.action === 'show') {
+      this.action = this.defaultAction;
+    }
 
-      this.currentLocation = window.location;
+    this.currentLocation = window.location;
 
-      const tabSelector = `${this.options.parentEl} a[data-toggle="tab"]`;
+    const tabSelector = `${this.options.parentEl} a[data-toggle="tab"]`;
 
-      // since this is a custom event we need jQuery :(
-      $(document)
-        .off('shown.bs.tab', tabSelector)
-        .on('shown.bs.tab', tabSelector, e => this.tabShown(e));
+    // since this is a custom event we need jQuery :(
+    $(document)
+      .off('shown.bs.tab', tabSelector)
+      .on('shown.bs.tab', tabSelector, e => this.tabShown(e));
 
-      this.activateTab(this.action);
-    }
+    this.activateTab(this.action);
+  }
 
-    /**
-     * Handles the `shown.bs.tab` event to set the currect url action.
-     *
-     * @param  {type} evt
-     * @return {Function}
-     */
-    tabShown(evt) {
-      const source = evt.target.getAttribute('href');
+  /**
+   * Handles the `shown.bs.tab` event to set the currect url action.
+   *
+   * @param  {type} evt
+   * @return {Function}
+   */
+  tabShown(evt) {
+    const source = evt.target.getAttribute('href');
 
-      return this.setCurrentAction(source);
-    }
+    return this.setCurrentAction(source);
+  }
 
-    /**
-     * Updates the URL with the path that matched the given action.
-     *
-     * @param  {String} source
-     * @return {String}
-     */
-    setCurrentAction(source) {
-      const copySource = source;
+  /**
+   * Updates the URL with the path that matched the given action.
+   *
+   * @param  {String} source
+   * @return {String}
+   */
+  setCurrentAction(source) {
+    const copySource = source;
 
-      copySource.replace(/\/+$/, '');
+    copySource.replace(/\/+$/, '');
 
-      const newState = `${copySource}${this.currentLocation.search}${this.currentLocation.hash}`;
+    const newState = `${copySource}${this.currentLocation.search}${this.currentLocation.hash}`;
 
-      history.replaceState({
-        url: newState,
-      }, document.title, newState);
-      return newState;
-    }
+    history.replaceState({
+      url: newState,
+    }, document.title, newState);
+    return newState;
+  }
 
-    /**
-     * Given the current action activates the correct tab.
-     * http://getbootstrap.com/javascript/#tab-show
-     * Note: Will trigger `shown.bs.tab`
-     */
-    activateTab() {
-      return $(`${this.options.parentEl} a[data-action='${this.action}']`).tab('show');
-    }
-  };
-})();
+  /**
+   * Given the current action activates the correct tab.
+   * http://getbootstrap.com/javascript/#tab-show
+   * Note: Will trigger `shown.bs.tab`
+   */
+  activateTab() {
+    return $(`${this.options.parentEl} a[data-action='${this.action}']`).tab('show');
+  }
+}
diff --git a/app/assets/javascripts/pipelines.js b/app/assets/javascripts/pipelines.js
index 4252b6158877e..26a36ad54d1fe 100644
--- a/app/assets/javascripts/pipelines.js
+++ b/app/assets/javascripts/pipelines.js
@@ -1,42 +1,14 @@
-/* eslint-disable no-new, guard-for-in, no-restricted-syntax, no-continue, no-param-reassign, max-len */
+import LinkedTabs from './lib/utils/bootstrap_linked_tabs';
 
-require('./lib/utils/bootstrap_linked_tabs');
-
-((global) => {
-  class Pipelines {
-    constructor(options = {}) {
-      if (options.initTabs && options.tabsOptions) {
-        new global.LinkedTabs(options.tabsOptions);
-      }
-
-      if (options.pipelineStatusUrl) {
-        gl.utils.setCiStatusFavicon(options.pipelineStatusUrl);
-      }
-
-      this.addMarginToBuildColumns();
+export default class Pipelines {
+  constructor(options = {}) {
+    if (options.initTabs && options.tabsOptions) {
+      // eslint-disable-next-line no-new
+      new LinkedTabs(options.tabsOptions);
     }
 
-    addMarginToBuildColumns() {
-      this.pipelineGraph = document.querySelector('.js-pipeline-graph');
-
-      const secondChildBuildNodes = this.pipelineGraph.querySelectorAll('.build:nth-child(2)');
-
-      for (const buildNodeIndex in secondChildBuildNodes) {
-        const buildNode = secondChildBuildNodes[buildNodeIndex];
-        const firstChildBuildNode = buildNode.previousElementSibling;
-        if (!firstChildBuildNode || !firstChildBuildNode.matches('.build')) continue;
-        const multiBuildColumn = buildNode.closest('.stage-column');
-        const previousColumn = multiBuildColumn.previousElementSibling;
-        if (!previousColumn || !previousColumn.matches('.stage-column')) continue;
-        multiBuildColumn.classList.add('left-margin');
-        firstChildBuildNode.classList.add('left-connector');
-        const columnBuilds = previousColumn.querySelectorAll('.build');
-        if (columnBuilds.length === 1) previousColumn.classList.add('no-margin');
-      }
-
-      this.pipelineGraph.classList.remove('hidden');
+    if (options.pipelineStatusUrl) {
+      gl.utils.setCiStatusFavicon(options.pipelineStatusUrl);
     }
   }
-
-  global.Pipelines = Pipelines;
-})(window.gl || (window.gl = {}));
+}
diff --git a/app/assets/javascripts/pipelines/components/graph/action_component.vue b/app/assets/javascripts/pipelines/components/graph/action_component.vue
new file mode 100644
index 0000000000000..14e485791ea5f
--- /dev/null
+++ b/app/assets/javascripts/pipelines/components/graph/action_component.vue
@@ -0,0 +1,59 @@
+<script>
+  import getActionIcon from '../../../vue_shared/ci_action_icons';
+  import tooltipMixin from '../../../vue_shared/mixins/tooltip';
+
+  /**
+   * Renders either a cancel, retry or play icon pointing to the given path.
+   * TODO: Remove UJS from here and use an async request instead.
+   */
+  export default {
+    props: {
+      tooltipText: {
+        type: String,
+        required: true,
+      },
+
+      link: {
+        type: String,
+        required: true,
+      },
+
+      actionMethod: {
+        type: String,
+        required: true,
+      },
+
+      actionIcon: {
+        type: String,
+        required: true,
+      },
+    },
+
+    mixins: [
+      tooltipMixin,
+    ],
+
+    computed: {
+      actionIconSvg() {
+        return getActionIcon(this.actionIcon);
+      },
+    },
+  };
+</script>
+<template>
+  <a
+    :data-method="actionMethod"
+    :title="tooltipText"
+    :href="link"
+    ref="tooltip"
+    class="ci-action-icon-container"
+    data-toggle="tooltip"
+    data-container="body">
+
+    <i
+      class="ci-action-icon-wrapper"
+      v-html="actionIconSvg"
+      aria-hidden="true"
+      />
+  </a>
+</template>
diff --git a/app/assets/javascripts/pipelines/components/graph/dropdown_action_component.vue b/app/assets/javascripts/pipelines/components/graph/dropdown_action_component.vue
new file mode 100644
index 0000000000000..19cafff4e1c6e
--- /dev/null
+++ b/app/assets/javascripts/pipelines/components/graph/dropdown_action_component.vue
@@ -0,0 +1,56 @@
+<script>
+  import getActionIcon from '../../../vue_shared/ci_action_icons';
+  import tooltipMixin from '../../../vue_shared/mixins/tooltip';
+
+  /**
+   * Renders either a cancel, retry or play icon pointing to the given path.
+   * TODO: Remove UJS from here and use an async request instead.
+   */
+  export default {
+    props: {
+      tooltipText: {
+        type: String,
+        required: true,
+      },
+
+      link: {
+        type: String,
+        required: true,
+      },
+
+      actionMethod: {
+        type: String,
+        required: true,
+      },
+
+      actionIcon: {
+        type: String,
+        required: true,
+      },
+    },
+
+    mixins: [
+      tooltipMixin,
+    ],
+
+    computed: {
+      actionIconSvg() {
+        return getActionIcon(this.actionIcon);
+      },
+    },
+  };
+</script>
+<template>
+  <a
+    :data-method="actionMethod"
+    :title="tooltipText"
+    :href="link"
+    ref="tooltip"
+    rel="nofollow"
+    class="ci-action-icon-wrapper js-ci-status-icon"
+    data-toggle="tooltip"
+    data-container="body"
+    v-html="actionIconSvg"
+    aria-label="Job's action">
+  </a>
+</template>
diff --git a/app/assets/javascripts/pipelines/components/graph/dropdown_job_component.vue b/app/assets/javascripts/pipelines/components/graph/dropdown_job_component.vue
new file mode 100644
index 0000000000000..d597af8dfb5f8
--- /dev/null
+++ b/app/assets/javascripts/pipelines/components/graph/dropdown_job_component.vue
@@ -0,0 +1,86 @@
+<script>
+  import jobNameComponent from './job_name_component.vue';
+  import jobComponent from './job_component.vue';
+  import tooltipMixin from '../../../vue_shared/mixins/tooltip';
+
+  /**
+   * Renders the dropdown for the pipeline graph.
+   *
+   * The following object should be provided as `job`:
+   *
+   * {
+   *   "id": 4256,
+   *   "name": "test",
+   *   "status": {
+   *     "icon": "icon_status_success",
+   *     "text": "passed",
+   *     "label": "passed",
+   *     "group": "success",
+   *     "details_path": "/root/ci-mock/builds/4256",
+   *     "action": {
+   *       "icon": "icon_action_retry",
+   *       "title": "Retry",
+   *       "path": "/root/ci-mock/builds/4256/retry",
+   *       "method": "post"
+   *     }
+   *   }
+   * }
+   */
+  export default {
+    props: {
+      job: {
+        type: Object,
+        required: true,
+      },
+    },
+
+    mixins: [
+      tooltipMixin,
+    ],
+
+    components: {
+      jobComponent,
+      jobNameComponent,
+    },
+
+    computed: {
+      tooltipText() {
+        return `${this.job.name} - ${this.job.status.label}`;
+      },
+    },
+  };
+</script>
+<template>
+  <div>
+    <button
+      type="button"
+      data-toggle="dropdown"
+      data-container="body"
+      class="dropdown-menu-toggle build-content"
+      :title="tooltipText"
+      ref="tooltip">
+
+      <job-name-component
+        :name="job.name"
+        :status="job.status" />
+
+      <span class="dropdown-counter-badge">
+        {{job.size}}
+      </span>
+    </button>
+
+    <ul class="dropdown-menu big-pipeline-graph-dropdown-menu js-grouped-pipeline-dropdown">
+      <li class="scrollable-menu">
+        <ul>
+          <li v-for="item in job.jobs">
+            <job-component
+              :job="item"
+              :is-dropdown="true"
+              css-class-job-name="mini-pipeline-graph-dropdown-item"
+              />
+          </li>
+        </ul>
+      </li>
+    </ul>
+  </div>
+</template>
diff --git a/app/assets/javascripts/pipelines/components/graph/graph_component.vue b/app/assets/javascripts/pipelines/components/graph/graph_component.vue
new file mode 100644
index 0000000000000..a84161ef5e796
--- /dev/null
+++ b/app/assets/javascripts/pipelines/components/graph/graph_component.vue
@@ -0,0 +1,92 @@
+<script>
+  /* global Flash */
+  import Visibility from 'visibilityjs';
+  import Poll from '../../../lib/utils/poll';
+  import PipelineService from '../../services/pipeline_service';
+  import PipelineStore from '../../stores/pipeline_store';
+  import stageColumnComponent from './stage_column_component.vue';
+  import '../../../flash';
+
+  export default {
+    components: {
+      stageColumnComponent,
+    },
+
+    data() {
+      const DOMdata = document.getElementById('js-pipeline-graph-vue').dataset;
+      const store = new PipelineStore();
+
+      return {
+        isLoading: false,
+        endpoint: DOMdata.endpoint,
+        store,
+        state: store.state,
+      };
+    },
+
+    created() {
+      this.service = new PipelineService(this.endpoint);
+
+      const poll = new Poll({
+        resource: this.service,
+        method: 'getPipeline',
+        successCallback: this.successCallback,
+        errorCallback: this.errorCallback,
+      });
+
+      if (!Visibility.hidden()) {
+        this.isLoading = true;
+        poll.makeRequest();
+      }
+
+      Visibility.change(() => {
+        if (!Visibility.hidden()) {
+          poll.restart();
+        } else {
+          poll.stop();
+        }
+      });
+    },
+
+    methods: {
+      successCallback(response) {
+        const data = response.json();
+
+        this.isLoading = false;
+        this.store.storeGraph(data.details.stages);
+      },
+
+      errorCallback() {
+        this.isLoading = false;
+        return new Flash('An error occurred while fetching the pipeline.');
+      },
+
+      capitalizeStageName(name) {
+        return name.charAt(0).toUpperCase() + name.slice(1);
+      },
+    },
+  };
+</script>
+<template>
+  <div class="build-content middle-block js-pipeline-graph">
+    <div class="pipeline-visualization pipeline-graph">
+      <div class="text-center">
+        <i
+          v-if="isLoading"
+          class="loading-icon fa fa-spin fa-spinner fa-3x"
+          aria-label="Loading"
+          aria-hidden="true" />
+      </div>
+
+      <ul
+        v-if="!isLoading"
+        class="stage-column-list">
+        <stage-column-component
+          v-for="stage in state.graph"
+          :title="capitalizeStageName(stage.name)"
+          :jobs="stage.groups"
+          :key="stage.name"/>
+      </ul>
+    </div>
+  </div>
+</template>
diff --git a/app/assets/javascripts/pipelines/components/graph/job_component.vue b/app/assets/javascripts/pipelines/components/graph/job_component.vue
new file mode 100644
index 0000000000000..b39c936101e3f
--- /dev/null
+++ b/app/assets/javascripts/pipelines/components/graph/job_component.vue
@@ -0,0 +1,124 @@
+<script>
+  import actionComponent from './action_component.vue';
+  import dropdownActionComponent from './dropdown_action_component.vue';
+  import jobNameComponent from './job_name_component.vue';
+  import tooltipMixin from '../../../vue_shared/mixins/tooltip';
+
+  /**
+   * Renders the badge for the pipeline graph and the job's dropdown.
+   *
+   * The following object should be provided as `job`:
+   *
+   * {
+   *   "id": 4256,
+   *   "name": "test",
+   *   "status": {
+   *     "icon": "icon_status_success",
+   *     "text": "passed",
+   *     "label": "passed",
+   *     "group": "success",
+   *     "details_path": "/root/ci-mock/builds/4256",
+   *     "action": {
+   *       "icon": "icon_action_retry",
+   *       "title": "Retry",
+   *       "path": "/root/ci-mock/builds/4256/retry",
+   *       "method": "post"
+   *     }
+   *   }
+   * }
+   */
+
+  export default {
+    props: {
+      job: {
+        type: Object,
+        required: true,
+      },
+
+      cssClassJobName: {
+        type: String,
+        required: false,
+        default: '',
+      },
+
+      isDropdown: {
+        type: Boolean,
+        required: false,
+        default: false,
+      },
+    },
+
+    components: {
+      actionComponent,
+      dropdownActionComponent,
+      jobNameComponent,
+    },
+
+    mixins: [
+      tooltipMixin,
+    ],
+
+    computed: {
+      tooltipText() {
+        return `${this.job.name} - ${this.job.status.label}`;
+      },
+
+      /**
+       * Verifies if the provided job has an action path
+       *
+       * @return {Boolean}
+       */
+      hasAction() {
+        return this.job.status && this.job.status.action && this.job.status.action.path;
+      },
+    },
+  };
+</script>
+<template>
+  <div>
+    <a
+      v-if="job.status.details_path"
+      :href="job.status.details_path"
+      :title="tooltipText"
+      :class="cssClassJobName"
+      ref="tooltip"
+      data-toggle="tooltip"
+      data-container="body">
+
+      <job-name-component
+        :name="job.name"
+        :status="job.status"
+        />
+    </a>
+
+    <div
+      v-else
+      :title="tooltipText"
+      :class="cssClassJobName"
+      ref="tooltip"
+      data-toggle="tooltip"
+      data-container="body">
+
+      <job-name-component
+        :name="job.name"
+        :status="job.status"
+        />
+    </div>
+
+    <action-component
+      v-if="hasAction && !isDropdown"
+      :tooltip-text="job.status.action.title"
+      :link="job.status.action.path"
+      :action-icon="job.status.action.icon"
+      :action-method="job.status.action.method"
+      />
+
+    <dropdown-action-component
+      v-if="hasAction && isDropdown"
+      :tooltip-text="job.status.action.title"
+      :link="job.status.action.path"
+      :action-icon="job.status.action.icon"
+      :action-method="job.status.action.method"
+      />
+  </div>
+</template>
diff --git a/app/assets/javascripts/pipelines/components/graph/job_name_component.vue b/app/assets/javascripts/pipelines/components/graph/job_name_component.vue
new file mode 100644
index 0000000000000..d8856e10668a2
--- /dev/null
+++ b/app/assets/javascripts/pipelines/components/graph/job_name_component.vue
@@ -0,0 +1,37 @@
+<script>
+  import ciIcon from '../../../vue_shared/components/ci_icon.vue';
+
+  /**
+   * Component that renders both the CI icon status and the job name.
+   * Used in
+   *  - Badge component
+   *  - Dropdown badge components
+   */
+  export default {
+    props: {
+      name: {
+        type: String,
+        required: true,
+      },
+
+      status: {
+        type: Object,
+        required: true,
+      },
+    },
+
+    components: {
+      ciIcon,
+    },
+  };
+</script>
+<template>
+  <span>
+    <ci-icon
+      :status="status" />
+
+    <span class="ci-status-text">
+      {{name}}
+    </span>
+  </span>
+</template>
diff --git a/app/assets/javascripts/pipelines/components/graph/stage_column_component.vue b/app/assets/javascripts/pipelines/components/graph/stage_column_component.vue
new file mode 100644
index 0000000000000..b7da185e2803e
--- /dev/null
+++ b/app/assets/javascripts/pipelines/components/graph/stage_column_component.vue
@@ -0,0 +1,64 @@
+<script>
+import jobComponent from './job_component.vue';
+import dropdownJobComponent from './dropdown_job_component.vue';
+
+export default {
+  props: {
+    title: {
+      type: String,
+      required: true,
+    },
+
+    jobs: {
+      type: Array,
+      required: true,
+    },
+  },
+
+  components: {
+    jobComponent,
+    dropdownJobComponent,
+  },
+
+  methods: {
+    firstJob(list) {
+      return list[0];
+    },
+
+    jobId(job) {
+      return `ci-badge-${job.name}`;
+    },
+  },
+};
+</script>
+<template>
+  <li class="stage-column">
+    <div class="stage-name">
+      {{title}}
+    </div>
+    <div class="builds-container">
+      <ul>
+        <li
+          v-for="job in jobs"
+          :key="job.id"
+          class="build"
+          :id="jobId(job)">
+
+          <div class="curve"></div>
+
+          <job-component
+            v-if="job.size === 1"
+            :job="job"
+            css-class-job-name="build-content"
+            />
+
+          <dropdown-job-component
+            v-if="job.size > 1"
+            :job="job"
+            />
+
+        </li>
+      </ul>
+    </div>
+  </li>
+</template>
diff --git a/app/assets/javascripts/pipelines/components/stage.vue b/app/assets/javascripts/pipelines/components/stage.vue
index 2e485f951a186..dc42223269df2 100644
--- a/app/assets/javascripts/pipelines/components/stage.vue
+++ b/app/assets/javascripts/pipelines/components/stage.vue
@@ -14,7 +14,7 @@
  */
 
 /* global Flash */
-import StatusIconEntityMap from '../../ci_status_icons';
+import { statusCssClasses, borderlessStatusIconEntityMap } from '../../vue_shared/ci_status_icons';
 
 export default {
   props: {
@@ -109,11 +109,11 @@ export default {
     },
 
     triggerButtonClass() {
-      return `ci-status-icon-${this.stage.status.group}`;
+      return `ci-status-icon-${statusCssClasses[this.stage.status.icon]}`;
     },
 
     svgIcon() {
-      return StatusIconEntityMap[this.stage.status.icon];
+      return borderlessStatusIconEntityMap[this.stage.status.icon];
     },
   },
 };
diff --git a/app/assets/javascripts/pipelines/graph_bundle.js b/app/assets/javascripts/pipelines/graph_bundle.js
new file mode 100644
index 0000000000000..b7a6b5d847977
--- /dev/null
+++ b/app/assets/javascripts/pipelines/graph_bundle.js
@@ -0,0 +1,10 @@
+import Vue from 'vue';
+import pipelineGraph from './components/graph/graph_component.vue';
+
+document.addEventListener('DOMContentLoaded', () => new Vue({
+  el: '#js-pipeline-graph-vue',
+  components: {
+    pipelineGraph,
+  },
+  render: createElement => createElement('pipeline-graph'),
+}));
diff --git a/app/assets/javascripts/pipelines/services/pipeline_service.js b/app/assets/javascripts/pipelines/services/pipeline_service.js
new file mode 100644
index 0000000000000..f1cc60c1ee031
--- /dev/null
+++ b/app/assets/javascripts/pipelines/services/pipeline_service.js
@@ -0,0 +1,14 @@
+import Vue from 'vue';
+import VueResource from 'vue-resource';
+
+Vue.use(VueResource);
+
+export default class PipelineService {
+  constructor(endpoint) {
+    this.pipeline = Vue.resource(endpoint);
+  }
+
+  getPipeline() {
+    return this.pipeline.get();
+  }
+}
diff --git a/app/assets/javascripts/pipelines/stores/pipeline_store.js b/app/assets/javascripts/pipelines/stores/pipeline_store.js
new file mode 100644
index 0000000000000..86ab50d8f1e97
--- /dev/null
+++ b/app/assets/javascripts/pipelines/stores/pipeline_store.js
@@ -0,0 +1,11 @@
+export default class PipelineStore {
+  constructor() {
+    this.state = {};
+
+    this.state.graph = [];
+  }
+
+  storeGraph(graph = []) {
+    this.state.graph = graph;
+  }
+}
diff --git a/app/assets/javascripts/vue_shared/ci_action_icons.js b/app/assets/javascripts/vue_shared/ci_action_icons.js
new file mode 100644
index 0000000000000..734b3c6c45e4a
--- /dev/null
+++ b/app/assets/javascripts/vue_shared/ci_action_icons.js
@@ -0,0 +1,22 @@
+import cancelSVG from 'icons/_icon_action_cancel.svg';
+import retrySVG from 'icons/_icon_action_retry.svg';
+import playSVG from 'icons/_icon_action_play.svg';
+
+export default function getActionIcon(action) {
+  let icon;
+  switch (action) {
+    case 'icon_action_cancel':
+      icon = cancelSVG;
+      break;
+    case 'icon_action_retry':
+      icon = retrySVG;
+      break;
+    case 'icon_action_play':
+      icon = playSVG;
+      break;
+    default:
+      icon = '';
+  }
+
+  return icon;
+}
diff --git a/app/assets/javascripts/vue_shared/ci_status_icons.js b/app/assets/javascripts/vue_shared/ci_status_icons.js
new file mode 100644
index 0000000000000..48ad9214ac800
--- /dev/null
+++ b/app/assets/javascripts/vue_shared/ci_status_icons.js
@@ -0,0 +1,55 @@
+import BORDERLESS_CANCELED_SVG from 'icons/_icon_status_canceled_borderless.svg';
+import BORDERLESS_CREATED_SVG from 'icons/_icon_status_created_borderless.svg';
+import BORDERLESS_FAILED_SVG from 'icons/_icon_status_failed_borderless.svg';
+import BORDERLESS_MANUAL_SVG from 'icons/_icon_status_manual_borderless.svg';
+import BORDERLESS_PENDING_SVG from 'icons/_icon_status_pending_borderless.svg';
+import BORDERLESS_RUNNING_SVG from 'icons/_icon_status_running_borderless.svg';
+import BORDERLESS_SKIPPED_SVG from 'icons/_icon_status_skipped_borderless.svg';
+import BORDERLESS_SUCCESS_SVG from 'icons/_icon_status_success_borderless.svg';
+import BORDERLESS_WARNING_SVG from 'icons/_icon_status_warning_borderless.svg';
+
+import CANCELED_SVG from 'icons/_icon_status_canceled.svg';
+import CREATED_SVG from 'icons/_icon_status_created.svg';
+import FAILED_SVG from 'icons/_icon_status_failed.svg';
+import MANUAL_SVG from 'icons/_icon_status_manual.svg';
+import PENDING_SVG from 'icons/_icon_status_pending.svg';
+import RUNNING_SVG from 'icons/_icon_status_running.svg';
+import SKIPPED_SVG from 'icons/_icon_status_skipped.svg';
+import SUCCESS_SVG from 'icons/_icon_status_success.svg';
+import WARNING_SVG from 'icons/_icon_status_warning.svg';
+
+export const borderlessStatusIconEntityMap = {
+  icon_status_canceled: BORDERLESS_CANCELED_SVG,
+  icon_status_created: BORDERLESS_CREATED_SVG,
+  icon_status_failed: BORDERLESS_FAILED_SVG,
+  icon_status_manual: BORDERLESS_MANUAL_SVG,
+  icon_status_pending: BORDERLESS_PENDING_SVG,
+  icon_status_running: BORDERLESS_RUNNING_SVG,
+  icon_status_skipped: BORDERLESS_SKIPPED_SVG,
+  icon_status_success: BORDERLESS_SUCCESS_SVG,
+  icon_status_warning: BORDERLESS_WARNING_SVG,
+};
+
+export const statusIconEntityMap = {
+  icon_status_canceled: CANCELED_SVG,
+  icon_status_created: CREATED_SVG,
+  icon_status_failed: FAILED_SVG,
+  icon_status_manual: MANUAL_SVG,
+  icon_status_pending: PENDING_SVG,
+  icon_status_running: RUNNING_SVG,
+  icon_status_skipped: SKIPPED_SVG,
+  icon_status_success: SUCCESS_SVG,
+  icon_status_warning: WARNING_SVG,
+};
+
+export const statusCssClasses = {
+  icon_status_canceled: 'canceled',
+  icon_status_created: 'created',
+  icon_status_failed: 'failed',
+  icon_status_manual: 'manual',
+  icon_status_pending: 'pending',
+  icon_status_running: 'running',
+  icon_status_skipped: 'skipped',
+  icon_status_success: 'success',
+  icon_status_warning: 'warning',
+};
diff --git a/app/assets/javascripts/vue_shared/components/ci_icon.vue b/app/assets/javascripts/vue_shared/components/ci_icon.vue
new file mode 100644
index 0000000000000..4d44baaa3c427
--- /dev/null
+++ b/app/assets/javascripts/vue_shared/components/ci_icon.vue
@@ -0,0 +1,29 @@
+<script>
+  import { statusIconEntityMap, statusCssClasses } from '../../vue_shared/ci_status_icons';
+
+  export default {
+    props: {
+      status: {
+        type: Object,
+        required: true,
+      },
+    },
+
+    computed: {
+      statusIconSvg() {
+        return statusIconEntityMap[this.status.icon];
+      },
+
+      cssClass() {
+        const status = statusCssClasses[this.status.icon];
+        return `ci-status-icon ci-status-icon-${status} js-ci-status-icon-${status}`;
+      },
+    },
+  };
+</script>
+<template>
+  <span
+    :class="cssClass"
+    v-html="statusIconSvg">
+  </span>
+</template>
diff --git a/app/assets/javascripts/vue_shared/mixins/tooltip.js b/app/assets/javascripts/vue_shared/mixins/tooltip.js
new file mode 100644
index 0000000000000..9bb948bff66b2
--- /dev/null
+++ b/app/assets/javascripts/vue_shared/mixins/tooltip.js
@@ -0,0 +1,9 @@
+export default {
+  mounted() {
+    $(this.$refs.tooltip).tooltip();
+  },
+
+  updated() {
+    $(this.$refs.tooltip).tooltip('fixTitle');
+  },
+};
diff --git a/app/assets/stylesheets/pages/pipelines.scss b/app/assets/stylesheets/pages/pipelines.scss
index 530a6f3c6a1a8..eaf3dd495674c 100644
--- a/app/assets/stylesheets/pages/pipelines.scss
+++ b/app/assets/stylesheets/pages/pipelines.scss
@@ -486,7 +486,7 @@
     color: $gl-text-color-secondary;
 
     // Action Icons in big pipeline-graph nodes
-    > .ci-action-icon-container .ci-action-icon-wrapper {
+    > div > .ci-action-icon-container .ci-action-icon-wrapper {
       height: 30px;
       width: 30px;
       background: $white-light;
@@ -511,7 +511,7 @@
       }
     }
 
-    > .ci-action-icon-container {
+    > div > .ci-action-icon-container {
       position: absolute;
       right: 5px;
       top: 5px;
@@ -541,7 +541,7 @@
       }
     }
 
-    > .build-content {
+    > div > .build-content {
       display: inline-block;
       padding: 8px 10px 9px;
       width: 100%;
@@ -557,34 +557,6 @@
     }
 
 
-    .arrow {
-      &::before,
-      &::after {
-        content: '';
-        display: inline-block;
-        position: absolute;
-        width: 0;
-        height: 0;
-        border-color: transparent;
-        border-style: solid;
-        top: 18px;
-      }
-
-      &::before {
-        left: -5px;
-        margin-top: -6px;
-        border-width: 7px 5px 7px 0;
-        border-right-color: $border-color;
-      }
-
-      &::after {
-        left: -4px;
-        margin-top: -9px;
-        border-width: 10px 7px 10px 0;
-        border-right-color: $white-light;
-      }
-    }
-
     // Connect first build in each stage with right horizontal line
     &:first-child {
       &::after {
@@ -859,7 +831,8 @@
     border-radius: 3px;
 
     // build name
-    .ci-build-text {
+    .ci-build-text,
+    .ci-status-text {
       font-weight: 200;
       overflow: hidden;
       white-space: nowrap;
@@ -911,6 +884,38 @@
   }
 }
 
+/**
+ * Top arrow in the dropdown in the big pipeline graph
+ */
+.big-pipeline-graph-dropdown-menu {
+
+  &::before,
+  &::after {
+    content: '';
+    display: inline-block;
+    position: absolute;
+    width: 0;
+    height: 0;
+    border-color: transparent;
+    border-style: solid;
+    top: 18px;
+  }
+
+  &::before {
+    left: -5px;
+    margin-top: -6px;
+    border-width: 7px 5px 7px 0;
+    border-right-color: $border-color;
+  }
+
+  &::after {
+    left: -4px;
+    margin-top: -9px;
+    border-width: 10px 7px 10px 0;
+    border-right-color: $white-light;
+  }
+}
+
 /**
  * Top arrow in the dropdown in the mini pipeline graph
  */
diff --git a/app/views/ci/status/_graph_badge.html.haml b/app/views/ci/status/_graph_badge.html.haml
deleted file mode 100644
index 128b418090fd2..0000000000000
--- a/app/views/ci/status/_graph_badge.html.haml
+++ /dev/null
@@ -1,20 +0,0 @@
--# Renders the graph node with both the status icon, status name and action icon
-
-- subject = local_assigns.fetch(:subject)
-- status = subject.detailed_status(current_user)
-- klass = "ci-status-icon ci-status-icon-#{status.group} js-ci-status-icon-#{status.group}"
-- tooltip = "#{subject.name} - #{status.label}"
-
-- if status.has_details?
-  = link_to status.details_path, class: 'build-content has-tooltip', data: { toggle: 'tooltip', title: tooltip, container: 'body' }  do
-    %span{ class: klass }= custom_icon(status.icon)
-    .ci-status-text= subject.name
-- else
-  .build-content.has-tooltip{ data: { toggle: 'tooltip', title: tooltip } }
-    %span{ class: klass }= custom_icon(status.icon)
-    .ci-status-text= subject.name
-
-- if status.has_action?
-  = link_to status.action_path, class: 'ci-action-icon-container has-tooltip', method: status.action_method, data: { toggle: 'tooltip', title: status.action_title, container: 'body' }  do
-    %i.ci-action-icon-wrapper{ class: "js-#{status.action_icon.dasherize}" }
-      = custom_icon(status.action_icon)
diff --git a/app/views/projects/pipelines/_graph.html.haml b/app/views/projects/pipelines/_graph.html.haml
deleted file mode 100644
index 0202833c0bf56..0000000000000
--- a/app/views/projects/pipelines/_graph.html.haml
+++ /dev/null
@@ -1,4 +0,0 @@
-- pipeline = local_assigns.fetch(:pipeline)
-.pipeline-visualization.pipeline-graph
-  %ul.stage-column-list
-    = render partial: "projects/stage/graph", collection: pipeline.stages, as: :stage
diff --git a/app/views/projects/pipelines/_with_tabs.html.haml b/app/views/projects/pipelines/_with_tabs.html.haml
index 1aa48bf98131b..1c7d1768aa58c 100644
--- a/app/views/projects/pipelines/_with_tabs.html.haml
+++ b/app/views/projects/pipelines/_with_tabs.html.haml
@@ -17,8 +17,11 @@
 
 .tab-content
   #js-tab-pipeline.tab-pane
-    .build-content.middle-block.js-pipeline-graph
-      = render "projects/pipelines/graph", pipeline: pipeline
+    #js-pipeline-graph-vue{ data: { endpoint: namespace_project_pipeline_path(@project.namespace, @project, @pipeline, format: :json) } }
+
+    - content_for :page_specific_javascripts do
+      = page_specific_javascript_bundle_tag('common_vue')
+      = page_specific_javascript_bundle_tag('pipelines_graph')
 
   #js-tab-builds.tab-pane
     - if pipeline.yaml_errors.present?
diff --git a/app/views/projects/stage/_graph.html.haml b/app/views/projects/stage/_graph.html.haml
deleted file mode 100644
index 4ee30b023acc1..0000000000000
--- a/app/views/projects/stage/_graph.html.haml
+++ /dev/null
@@ -1,19 +0,0 @@
-- stage = local_assigns.fetch(:stage)
-- statuses = stage.statuses.latest
-- status_groups = statuses.sort_by(&:sortable_name).group_by(&:group_name)
-%li.stage-column
-  .stage-name
-    %a{ name: stage.name }
-    = stage.name.titleize
-  .builds-container
-    %ul
-      - status_groups.each do |group_name, grouped_statuses|
-        - if grouped_statuses.one?
-          - status = grouped_statuses.first
-          %li.build{ 'id' => "ci-badge-#{group_name}" }
-            .curve
-            = render 'ci/status/graph_badge', subject: status
-        - else
-          %li.build{ 'id' => "ci-badge-#{group_name}" }
-            .curve
-            = render 'projects/stage/in_stage_group', name: group_name, subject: grouped_statuses
diff --git a/app/views/projects/stage/_in_stage_group.html.haml b/app/views/projects/stage/_in_stage_group.html.haml
deleted file mode 100644
index 671a3ef481c7e..0000000000000
--- a/app/views/projects/stage/_in_stage_group.html.haml
+++ /dev/null
@@ -1,14 +0,0 @@
-- group_status = CommitStatus.where(id: subject).status
-%button.dropdown-menu-toggle.build-content.has-tooltip{ type: 'button', data: { toggle: 'dropdown',  title: "#{name} - #{group_status}", container: 'body' } }
-  %span{ class: "ci-status-icon ci-status-icon-#{group_status}" }
-    = ci_icon_for_status(group_status)
-  %span.ci-status-text
-    = name
-  %span.dropdown-counter-badge= subject.size
-
-%ul.dropdown-menu.big-pipeline-graph-dropdown-menu.js-grouped-pipeline-dropdown
-  .arrow
-  .scrollable-menu
-    - subject.each do |status|
-      %li
-        = render 'ci/status/dropdown_graph_badge', subject: status
diff --git a/changelogs/unreleased/25226-realtime-pipelines-fe.yml b/changelogs/unreleased/25226-realtime-pipelines-fe.yml
new file mode 100644
index 0000000000000..1149c8f0eac15
--- /dev/null
+++ b/changelogs/unreleased/25226-realtime-pipelines-fe.yml
@@ -0,0 +1,4 @@
+---
+title: Re-rewrites pipeline graph in vue to support realtime data updates
+merge_request:
+author:
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 119b1ea9d2e4d..a3dae6b2e131c 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -49,6 +49,7 @@ var config = {
     pdf_viewer:           './blob/pdf_viewer.js',
     pipelines:            './pipelines/index.js',
     balsamiq_viewer:      './blob/balsamiq_viewer.js',
+    pipelines_graph:      './pipelines/graph_bundle.js',
     profile:              './profile/profile_bundle.js',
     protected_branches:   './protected_branches/protected_branches_bundle.js',
     protected_tags:       './protected_tags',
@@ -145,6 +146,7 @@ var config = {
         'pdf_viewer',
         'pipelines',
         'balsamiq_viewer',
+        'pipelines_graph',
       ],
       minChunks: function(module, count) {
         return module.resource && (/vue_shared/).test(module.resource);
diff --git a/spec/javascripts/bootstrap_linked_tabs_spec.js b/spec/javascripts/bootstrap_linked_tabs_spec.js
index fa9f95e16cdca..a27dc48b3fd8b 100644
--- a/spec/javascripts/bootstrap_linked_tabs_spec.js
+++ b/spec/javascripts/bootstrap_linked_tabs_spec.js
@@ -1,4 +1,4 @@
-require('~/lib/utils/bootstrap_linked_tabs');
+import LinkedTabs from '~/lib/utils/bootstrap_linked_tabs';
 
 (() => {
   // TODO: remove this hack!
@@ -25,7 +25,7 @@ require('~/lib/utils/bootstrap_linked_tabs');
       });
 
       it('should activate the tab correspondent to the given action', () => {
-        const linkedTabs = new window.gl.LinkedTabs({ // eslint-disable-line
+        const linkedTabs = new LinkedTabs({ // eslint-disable-line
           action: 'tab1',
           defaultAction: 'tab1',
           parentEl: '.linked-tabs',
@@ -35,7 +35,7 @@ require('~/lib/utils/bootstrap_linked_tabs');
       });
 
       it('should active the default tab action when the action is show', () => {
-        const linkedTabs = new window.gl.LinkedTabs({ // eslint-disable-line
+        const linkedTabs = new LinkedTabs({ // eslint-disable-line
           action: 'show',
           defaultAction: 'tab1',
           parentEl: '.linked-tabs',
@@ -49,7 +49,7 @@ require('~/lib/utils/bootstrap_linked_tabs');
       it('should change the url according to the clicked tab', () => {
         const historySpy = !phantomjs && spyOn(history, 'replaceState').and.callFake(() => {});
 
-        const linkedTabs = new window.gl.LinkedTabs({ // eslint-disable-line
+        const linkedTabs = new LinkedTabs({
           action: 'show',
           defaultAction: 'tab1',
           parentEl: '.linked-tabs',
diff --git a/spec/javascripts/ci_status_icon_spec.js b/spec/javascripts/ci_status_icon_spec.js
deleted file mode 100644
index c83416c15efc1..0000000000000
--- a/spec/javascripts/ci_status_icon_spec.js
+++ /dev/null
@@ -1,44 +0,0 @@
-import * as icons from '~/ci_status_icons';
-
-describe('CI status icons', () => {
-  const statuses = [
-    'canceled',
-    'created',
-    'failed',
-    'manual',
-    'pending',
-    'running',
-    'skipped',
-    'success',
-    'warning',
-  ];
-
-  statuses.forEach((status) => {
-    it(`should export a ${status} svg`, () => {
-      const key = `${status.toUpperCase()}_SVG`;
-
-      expect(Object.hasOwnProperty.call(icons, key)).toBe(true);
-      expect(icons[key]).toMatch(/^<svg/);
-    });
-  });
-
-  describe('default export map', () => {
-    const entityIconNames = [
-      'icon_status_canceled',
-      'icon_status_created',
-      'icon_status_failed',
-      'icon_status_manual',
-      'icon_status_pending',
-      'icon_status_running',
-      'icon_status_skipped',
-      'icon_status_success',
-      'icon_status_warning',
-    ];
-
-    entityIconNames.forEach((iconName) => {
-      it(`should have a '${iconName}' key`, () => {
-        expect(Object.hasOwnProperty.call(icons.default, iconName)).toBe(true);
-      });
-    });
-  });
-});
diff --git a/spec/javascripts/fixtures/graph.html.haml b/spec/javascripts/fixtures/graph.html.haml
new file mode 100644
index 0000000000000..4fedb0f1ded07
--- /dev/null
+++ b/spec/javascripts/fixtures/graph.html.haml
@@ -0,0 +1 @@
+#js-pipeline-graph-vue{ data: { endpoint: "foo" } }
diff --git a/spec/javascripts/pipelines/graph/action_component_spec.js b/spec/javascripts/pipelines/graph/action_component_spec.js
new file mode 100644
index 0000000000000..f033956c0715b
--- /dev/null
+++ b/spec/javascripts/pipelines/graph/action_component_spec.js
@@ -0,0 +1,40 @@
+import Vue from 'vue';
+import actionComponent from '~/pipelines/components/graph/action_component.vue';
+
+describe('pipeline graph action component', () => {
+  let component;
+
+  beforeEach(() => {
+    const ActionComponent = Vue.extend(actionComponent);
+    component = new ActionComponent({
+      propsData: {
+        tooltipText: 'bar',
+        link: 'foo',
+        actionMethod: 'post',
+        actionIcon: 'icon_action_cancel',
+      },
+    }).$mount();
+  });
+
+  it('should render a link', () => {
+    expect(component.$el.getAttribute('href')).toEqual('foo');
+  });
+
+  it('should render the provided title as a bootstrap tooltip', () => {
+    expect(component.$el.getAttribute('data-original-title')).toEqual('bar');
+  });
+
+  it('should update bootstrap tooltip when title changes', (done) => {
+    component.tooltipText = 'changed';
+
+    Vue.nextTick(() => {
+      expect(component.$el.getAttribute('data-original-title')).toBe('changed');
+      done();
+    });
+  });
+
+  it('should render an svg', () => {
+    expect(component.$el.querySelector('.ci-action-icon-wrapper')).toBeDefined();
+    expect(component.$el.querySelector('svg')).toBeDefined();
+  });
+});
diff --git a/spec/javascripts/pipelines/graph/dropdown_action_component_spec.js b/spec/javascripts/pipelines/graph/dropdown_action_component_spec.js
new file mode 100644
index 0000000000000..14ff1b0d25cfc
--- /dev/null
+++ b/spec/javascripts/pipelines/graph/dropdown_action_component_spec.js
@@ -0,0 +1,30 @@
+import Vue from 'vue';
+import dropdownActionComponent from '~/pipelines/components/graph/dropdown_action_component.vue';
+
+describe('action component', () => {
+  let component;
+
+  beforeEach(() => {
+    const DropdownActionComponent = Vue.extend(dropdownActionComponent);
+    component = new DropdownActionComponent({
+      propsData: {
+        tooltipText: 'bar',
+        link: 'foo',
+        actionMethod: 'post',
+        actionIcon: 'icon_action_cancel',
+      },
+    }).$mount();
+  });
+
+  it('should render a link', () => {
+    expect(component.$el.getAttribute('href')).toEqual('foo');
+  });
+
+  it('should render the provided title as a bootstrap tooltip', () => {
+    expect(component.$el.getAttribute('data-original-title')).toEqual('bar');
+  });
+
+  it('should render an svg', () => {
+    expect(component.$el.querySelector('svg')).toBeDefined();
+  });
+});
diff --git a/spec/javascripts/pipelines/graph/graph_component_spec.js b/spec/javascripts/pipelines/graph/graph_component_spec.js
new file mode 100644
index 0000000000000..a756617e65eae
--- /dev/null
+++ b/spec/javascripts/pipelines/graph/graph_component_spec.js
@@ -0,0 +1,83 @@
+import Vue from 'vue';
+import graphComponent from '~/pipelines/components/graph/graph_component.vue';
+
+describe('graph component', () => {
+  preloadFixtures('static/graph.html.raw');
+
+  let GraphComponent;
+
+  beforeEach(() => {
+    loadFixtures('static/graph.html.raw');
+    GraphComponent = Vue.extend(graphComponent);
+  });
+
+  describe('while is loading', () => {
+    it('should render a loading icon', () => {
+      const component = new GraphComponent().$mount('#js-pipeline-graph-vue');
+      expect(component.$el.querySelector('.loading-icon')).toBeDefined();
+    });
+  });
+
+  describe('with a successfull response', () => {
+    const interceptor = (request, next) => {
+      next(request.respondWith(JSON.stringify({
+        details: {
+          stages: [{
+            name: 'test',
+            title: 'test: passed',
+            status: {
+              icon: 'icon_status_success',
+              text: 'passed',
+              label: 'passed',
+              details_path: '/root/ci-mock/pipelines/123#test',
+            },
+            path: '/root/ci-mock/pipelines/123#test',
+            groups: [{
+              name: 'test',
+              size: 1,
+              jobs: [{
+                id: 4153,
+                name: 'test',
+                status: {
+                  icon: 'icon_status_success',
+                  text: 'passed',
+                  label: 'passed',
+                  details_path: '/root/ci-mock/builds/4153',
+                  action: {
+                    icon: 'icon_action_retry',
+                    title: 'Retry',
+                    path: '/root/ci-mock/builds/4153/retry',
+                    method: 'post',
+                  },
+                },
+              }],
+            }],
+          }],
+        },
+      }), {
+        status: 200,
+      }));
+    };
+
+    beforeEach(() => {
+      Vue.http.interceptors.push(interceptor);
+    });
+
+    afterEach(() => {
+      Vue.http.interceptors = _.without(Vue.http.interceptors, interceptor);
+    });
+
+    it('should render the graph', (done) => {
+      const component = new GraphComponent().$mount('#js-pipeline-graph-vue');
+
+      setTimeout(() => {
+        expect(component.$el.classList.contains('js-pipeline-graph')).toEqual(true);
+
+        expect(component.$el.querySelector('loading-icon')).toBe(null);
+
+        expect(component.$el.querySelector('.stage-column-list')).toBeDefined();
+        done();
+      }, 0);
+    });
+  });
+});
diff --git a/spec/javascripts/pipelines/graph/job_component_spec.js b/spec/javascripts/pipelines/graph/job_component_spec.js
new file mode 100644
index 0000000000000..63986b6c0db1a
--- /dev/null
+++ b/spec/javascripts/pipelines/graph/job_component_spec.js
@@ -0,0 +1,117 @@
+import Vue from 'vue';
+import jobComponent from '~/pipelines/components/graph/job_component.vue';
+
+describe('pipeline graph job component', () => {
+  let JobComponent;
+
+  const mockJob = {
+    id: 4256,
+    name: 'test',
+    status: {
+      icon: 'icon_status_success',
+      text: 'passed',
+      label: 'passed',
+      group: 'success',
+      details_path: '/root/ci-mock/builds/4256',
+      action: {
+        icon: 'icon_action_retry',
+        title: 'Retry',
+        path: '/root/ci-mock/builds/4256/retry',
+        method: 'post',
+      },
+    },
+  };
+
+  beforeEach(() => {
+    JobComponent = Vue.extend(jobComponent);
+  });
+
+  describe('name with link', () => {
+    it('should render the job name and status with a link', () => {
+      const component = new JobComponent({
+        propsData: {
+          job: mockJob,
+        },
+      }).$mount();
+
+      const link = component.$el.querySelector('a');
+
+      expect(link.getAttribute('href')).toEqual(mockJob.status.details_path);
+
+      expect(
+        link.getAttribute('data-original-title'),
+      ).toEqual(`${mockJob.name} - ${mockJob.status.label}`);
+
+      expect(component.$el.querySelector('.js-status-icon-success')).toBeDefined();
+
+      expect(
+        component.$el.querySelector('.ci-status-text').textContent.trim(),
+      ).toEqual(mockJob.name);
+    });
+  });
+
+  describe('name without link', () => {
+    it('it should render status and name', () => {
+      const component = new JobComponent({
+        propsData: {
+          job: {
+            id: 4256,
+            name: 'test',
+            status: {
+              icon: 'icon_status_success',
+              text: 'passed',
+              label: 'passed',
+              group: 'success',
+              details_path: '/root/ci-mock/builds/4256',
+            },
+          },
+        },
+      }).$mount();
+
+      expect(component.$el.querySelector('.js-status-icon-success')).toBeDefined();
+
+      expect(
+        component.$el.querySelector('.ci-status-text').textContent.trim(),
+      ).toEqual(mockJob.name);
+    });
+  });
+
+  describe('action icon', () => {
+    it('it should render the action icon', () => {
+      const component = new JobComponent({
+        propsData: {
+          job: mockJob,
+        },
+      }).$mount();
+
+      expect(component.$el.querySelector('a.ci-action-icon-container')).toBeDefined();
+      expect(component.$el.querySelector('i.ci-action-icon-wrapper')).toBeDefined();
+    });
+  });
+
+  describe('dropdown', () => {
+    it('should render the dropdown action icon', () => {
+      const component = new JobComponent({
+        propsData: {
+          job: mockJob,
+          isDropdown: true,
+        },
+      }).$mount();
+
+      expect(component.$el.querySelector('a.ci-action-icon-wrapper')).toBeDefined();
+    });
+  });
+
+  it('should render provided class name', () => {
+    const component = new JobComponent({
+      propsData: {
+        job: mockJob,
+        cssClassJobName: 'css-class-job-name',
+      },
+    }).$mount();
+
+    expect(
+      component.$el.querySelector('a').classList.contains('css-class-job-name'),
+    ).toBe(true);
+  });
+});
diff --git a/spec/javascripts/pipelines/graph/job_name_component_spec.js b/spec/javascripts/pipelines/graph/job_name_component_spec.js
new file mode 100644
index 0000000000000..8e2071ba0b3ed
--- /dev/null
+++ b/spec/javascripts/pipelines/graph/job_name_component_spec.js
@@ -0,0 +1,27 @@
+import Vue from 'vue';
+import jobNameComponent from '~/pipelines/components/graph/job_name_component.vue';
+
+describe('job name component', () => {
+  let component;
+
+  beforeEach(() => {
+    const JobNameComponent = Vue.extend(jobNameComponent);
+    component = new JobNameComponent({
+      propsData: {
+        name: 'foo',
+        status: {
+          icon: 'icon_status_success',
+        },
+      },
+    }).$mount();
+  });
+
+  it('should render the provided name', () => {
+    expect(component.$el.querySelector('.ci-status-text').textContent.trim()).toEqual('foo');
+  });
+
+  it('should render an icon with the provided status', () => {
+    expect(component.$el.querySelector('.ci-status-icon-success')).toBeDefined();
+    expect(component.$el.querySelector('.ci-status-icon-success svg')).toBeDefined();
+  });
+});
diff --git a/spec/javascripts/pipelines/graph/stage_column_component_spec.js b/spec/javascripts/pipelines/graph/stage_column_component_spec.js
new file mode 100644
index 0000000000000..aa4d6eedaf422
--- /dev/null
+++ b/spec/javascripts/pipelines/graph/stage_column_component_spec.js
@@ -0,0 +1,42 @@
+import Vue from 'vue';
+import stageColumnComponent from '~/pipelines/components/graph/stage_column_component.vue';
+
+describe('stage column component', () => {
+  let component;
+  const mockJob = {
+    id: 4256,
+    name: 'test',
+    status: {
+      icon: 'icon_status_success',
+      text: 'passed',
+      label: 'passed',
+      group: 'success',
+      details_path: '/root/ci-mock/builds/4256',
+      action: {
+        icon: 'icon_action_retry',
+        title: 'Retry',
+        path: '/root/ci-mock/builds/4256/retry',
+        method: 'post',
+      },
+    },
+  };
+
+  beforeEach(() => {
+    const StageColumnComponent = Vue.extend(stageColumnComponent);
+
+    component = new StageColumnComponent({
+      propsData: {
+        title: 'foo',
+        jobs: [mockJob, mockJob, mockJob],
+      },
+    }).$mount();
+  });
+
+  it('should render provided title', () => {
+    expect(component.$el.querySelector('.stage-name').textContent.trim()).toEqual('foo');
+  });
+
+  it('should render the provided jobs', () => {
+    expect(component.$el.querySelectorAll('.builds-container > ul > li').length).toEqual(3);
+  });
+});
diff --git a/spec/javascripts/pipelines_spec.js b/spec/javascripts/pipelines_spec.js
index 72770a702d305..81ac589f4e600 100644
--- a/spec/javascripts/pipelines_spec.js
+++ b/spec/javascripts/pipelines_spec.js
@@ -1,30 +1,22 @@
-require('~/pipelines');
+import Pipelines from '~/pipelines';
 
 // Fix for phantomJS
 if (!Element.prototype.matches && Element.prototype.webkitMatchesSelector) {
   Element.prototype.matches = Element.prototype.webkitMatchesSelector;
 }
 
-(() => {
-  describe('Pipelines', () => {
-    preloadFixtures('static/pipeline_graph.html.raw');
+describe('Pipelines', () => {
+  preloadFixtures('static/pipeline_graph.html.raw');
 
-    beforeEach(() => {
-      loadFixtures('static/pipeline_graph.html.raw');
-    });
-
-    it('should be defined', () => {
-      expect(window.gl.Pipelines).toBeDefined();
-    });
-
-    it('should create a `Pipelines` instance without options', () => {
-      expect(() => { new window.gl.Pipelines(); }).not.toThrow(); //eslint-disable-line
-    });
+  beforeEach(() => {
+    loadFixtures('static/pipeline_graph.html.raw');
+  });
 
-    it('should create a `Pipelines` instance with options', () => {
-      const pipelines = new window.gl.Pipelines({ foo: 'bar' });
+  it('should be defined', () => {
+    expect(Pipelines).toBeDefined();
+  });
 
-      expect(pipelines.pipelineGraph).toBeDefined();
-    });
+  it('should create a `Pipelines` instance without options', () => {
+    expect(() => { new Pipelines(); }).not.toThrow(); //eslint-disable-line
   });
-})();
+});
diff --git a/spec/javascripts/vue_shared/ci_action_icons_spec.js b/spec/javascripts/vue_shared/ci_action_icons_spec.js
new file mode 100644
index 0000000000000..2e89a07e76e83
--- /dev/null
+++ b/spec/javascripts/vue_shared/ci_action_icons_spec.js
@@ -0,0 +1,22 @@
+import getActionIcon from '~/vue_shared/ci_action_icons';
+import cancelSVG from 'icons/_icon_action_cancel.svg';
+import retrySVG from 'icons/_icon_action_retry.svg';
+import playSVG from 'icons/_icon_action_play.svg';
+
+describe('getActionIcon', () => {
+  it('should return an empty string', () => {
+    expect(getActionIcon()).toEqual('');
+  });
+
+  it('should return cancel svg', () => {
+    expect(getActionIcon('icon_action_cancel')).toEqual(cancelSVG);
+  });
+
+  it('should return retry svg', () => {
+    expect(getActionIcon('icon_action_retry')).toEqual(retrySVG);
+  });
+
+  it('should return play svg', () => {
+    expect(getActionIcon('icon_action_play')).toEqual(playSVG);
+  });
+});
diff --git a/spec/javascripts/vue_shared/ci_status_icon_spec.js b/spec/javascripts/vue_shared/ci_status_icon_spec.js
new file mode 100644
index 0000000000000..b6621d6054d0d
--- /dev/null
+++ b/spec/javascripts/vue_shared/ci_status_icon_spec.js
@@ -0,0 +1,27 @@
+import { borderlessStatusIconEntityMap, statusIconEntityMap } from '~/vue_shared/ci_status_icons';
+
+describe('CI status icons', () => {
+  const statuses = [
+    'icon_status_canceled',
+    'icon_status_created',
+    'icon_status_failed',
+    'icon_status_manual',
+    'icon_status_pending',
+    'icon_status_running',
+    'icon_status_skipped',
+    'icon_status_success',
+    'icon_status_warning',
+  ];
+
+  it('should have a dictionary for borderless icons', () => {
+    statuses.forEach((status) => {
+      expect(borderlessStatusIconEntityMap[status]).toBeDefined();
+    });
+  });
+
+  it('should have a dictionary for icons', () => {
+    statuses.forEach((status) => {
+      expect(statusIconEntityMap[status]).toBeDefined();
+    });
+  });
+});
diff --git a/spec/javascripts/vue_shared/components/ci_icon_spec.js b/spec/javascripts/vue_shared/components/ci_icon_spec.js
new file mode 100644
index 0000000000000..98dc6caa622ba
--- /dev/null
+++ b/spec/javascripts/vue_shared/components/ci_icon_spec.js
@@ -0,0 +1,130 @@
+import Vue from 'vue';
+import ciIcon from '~/vue_shared/components/ci_icon.vue';
+
+describe('CI Icon component', () => {
+  let CiIcon;
+  beforeEach(() => {
+    CiIcon = Vue.extend(ciIcon);
+  });
+
+  it('should render a span element with an svg', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_success',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.tagName).toEqual('SPAN');
+    expect(component.$el.querySelector('span > svg')).toBeDefined();
+  });
+
+  it('should render a success status', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_success',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-success')).toEqual(true);
+  });
+
+  it('should render a failed status', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_failed',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-failed')).toEqual(true);
+  });
+
+  it('should render success with warnings status', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_warning',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-warning')).toEqual(true);
+  });
+
+  it('should render pending status', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_pending',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-pending')).toEqual(true);
+  });
+
+  it('should render running status', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_running',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-running')).toEqual(true);
+  });
+
+  it('should render created status', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_created',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-created')).toEqual(true);
+  });
+
+  it('should render skipped status', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_skipped',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-skipped')).toEqual(true);
+  });
+
+  it('should render canceled status', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_canceled',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-canceled')).toEqual(true);
+  });
+
+  it('should render status for manual action', () => {
+    const component = new CiIcon({
+      propsData: {
+        status: {
+          icon: 'icon_status_manual',
+        },
+      },
+    }).$mount();
+
+    expect(component.$el.classList.contains('ci-status-icon-manual')).toEqual(true);
+  });
+});
-- 
GitLab


From 13b31d25ec9c5f8d17f95ca8ad756a88b5e3dbd9 Mon Sep 17 00:00:00 2001
From: Filipa Lacerda <filipa@gitlab.com>
Date: Sat, 6 May 2017 21:35:07 +0100
Subject: [PATCH 346/363] Fix broken css  class

---
 app/assets/javascripts/pipelines/components/stage.vue | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/assets/javascripts/pipelines/components/stage.vue b/app/assets/javascripts/pipelines/components/stage.vue
index dc42223269df2..310f44b06df44 100644
--- a/app/assets/javascripts/pipelines/components/stage.vue
+++ b/app/assets/javascripts/pipelines/components/stage.vue
@@ -14,7 +14,7 @@
  */
 
 /* global Flash */
-import { statusCssClasses, borderlessStatusIconEntityMap } from '../../vue_shared/ci_status_icons';
+import { borderlessStatusIconEntityMap } from '../../vue_shared/ci_status_icons';
 
 export default {
   props: {
@@ -109,7 +109,7 @@ export default {
     },
 
     triggerButtonClass() {
-      return `ci-status-icon-${statusCssClasses[this.stage.status.icon]}`;
+      return `ci-status-icon-${this.stage.status.group}`;
     },
 
     svgIcon() {
-- 
GitLab


From 8f64091a9550f9073889752bb975d96e074fd734 Mon Sep 17 00:00:00 2001
From: Filipa Lacerda <filipa@gitlab.com>
Date: Sat, 6 May 2017 21:55:40 +0100
Subject: [PATCH 347/363] Adds missing CSS class

---
 .../pipelines/components/graph/action_component.vue          | 5 +++++
 1 file changed, 5 insertions(+)

diff --git a/app/assets/javascripts/pipelines/components/graph/action_component.vue b/app/assets/javascripts/pipelines/components/graph/action_component.vue
index 14e485791ea5f..1f9e3d3977938 100644
--- a/app/assets/javascripts/pipelines/components/graph/action_component.vue
+++ b/app/assets/javascripts/pipelines/components/graph/action_component.vue
@@ -37,6 +37,10 @@
       actionIconSvg() {
         return getActionIcon(this.actionIcon);
       },
+
+      cssClass() {
+        return `js-${gl.text.dasherize(this.actionIcon)}`;
+      },
     },
   };
 </script>
@@ -52,6 +56,7 @@
 
     <i
       class="ci-action-icon-wrapper"
+      :class="cssClass"
       v-html="actionIconSvg"
       aria-hidden="true"
       />
-- 
GitLab


From b46d38aef4948c0a428c4334adc8aa221923decd Mon Sep 17 00:00:00 2001
From: Filipa Lacerda <filipa@gitlab.com>
Date: Sat, 6 May 2017 22:54:05 +0100
Subject: [PATCH 348/363] Move file loading to the top of the file

---
 app/views/projects/pipelines/_with_tabs.html.haml | 8 ++++----
 1 file changed, 4 insertions(+), 4 deletions(-)

diff --git a/app/views/projects/pipelines/_with_tabs.html.haml b/app/views/projects/pipelines/_with_tabs.html.haml
index 1c7d1768aa58c..075ddc0025c76 100644
--- a/app/views/projects/pipelines/_with_tabs.html.haml
+++ b/app/views/projects/pipelines/_with_tabs.html.haml
@@ -1,5 +1,9 @@
 - failed_builds = @pipeline.statuses.latest.failed
 
+- content_for :page_specific_javascripts do
+  = page_specific_javascript_bundle_tag('common_vue')
+  = page_specific_javascript_bundle_tag('pipelines_graph')
+
 .tabs-holder
   %ul.pipelines-tabs.nav-links.no-top.no-bottom
     %li.js-pipeline-tab-link
@@ -19,10 +23,6 @@
   #js-tab-pipeline.tab-pane
     #js-pipeline-graph-vue{ data: { endpoint: namespace_project_pipeline_path(@project.namespace, @project, @pipeline, format: :json) } }
 
-    - content_for :page_specific_javascripts do
-      = page_specific_javascript_bundle_tag('common_vue')
-      = page_specific_javascript_bundle_tag('pipelines_graph')
-
   #js-tab-builds.tab-pane
     - if pipeline.yaml_errors.present?
       .bs-callout.bs-callout-danger
-- 
GitLab


From c195b313ad2f84e79475058ab431648bed76fcbc Mon Sep 17 00:00:00 2001
From: Kamil Trzcinski <ayufan@ayufan.eu>
Date: Sun, 7 May 2017 14:09:21 +0200
Subject: [PATCH 349/363] Make test that actually displays pipeline graph

---
 .../projects/pipelines/pipelines_spec.rb      | 54 +++++++++++++++
 .../projects/pipelines/show.html.haml_spec.rb | 67 -------------------
 2 files changed, 54 insertions(+), 67 deletions(-)
 delete mode 100644 spec/views/projects/pipelines/show.html.haml_spec.rb

diff --git a/spec/features/projects/pipelines/pipelines_spec.rb b/spec/features/projects/pipelines/pipelines_spec.rb
index 2272b19bc8f17..8e820edee35e9 100644
--- a/spec/features/projects/pipelines/pipelines_spec.rb
+++ b/spec/features/projects/pipelines/pipelines_spec.rb
@@ -370,6 +370,60 @@
       end
     end
 
+    describe 'GET /:project/pipelines/show' do
+      let(:project) { create(:project) }
+
+      let(:pipeline) do
+        create(:ci_empty_pipeline,
+              project: project,
+              sha: project.commit.id,
+              user: user)
+      end
+
+      before do
+        create_build('build', 0, 'build', :success)
+        create_build('test', 1, 'rspec 0:2', :pending)
+        create_build('test', 1, 'rspec 1:2', :running)
+        create_build('test', 1, 'spinach 0:2', :created)
+        create_build('test', 1, 'spinach 1:2', :created)
+        create_build('test', 1, 'audit', :created)
+        create_build('deploy', 2, 'production', :created)
+
+        create(:generic_commit_status, pipeline: pipeline, stage: 'external', name: 'jenkins', stage_idx: 3)
+
+        visit namespace_project_pipeline_path(project.namespace, project, pipeline)
+        wait_for_vue_resource
+      end
+
+      it 'shows a graph with grouped stages' do
+        expect(page).to have_css('.js-pipeline-graph')
+        # expect(page).to have_css('.js-grouped-pipeline-dropdown')
+
+        # header
+        expect(page).to have_text("##{pipeline.id}")
+        expect(page).to have_css('time', text: pipeline.created_at.strftime("%b %d, %Y"))
+        expect(page).to have_selector(%Q(img[alt$="#{pipeline.user.name}'s avatar"]))
+        expect(page).to have_link(pipeline.user.name, href: user_path(pipeline.user))
+
+        # stages
+        expect(page).to have_text('Build')
+        expect(page).to have_text('Test')
+        expect(page).to have_text('Deploy')
+        expect(page).to have_text('External')
+
+        # builds
+        expect(page).to have_text('rspec')
+        expect(page).to have_text('spinach')
+        expect(page).to have_text('rspec 0:2')
+        expect(page).to have_text('production')
+        expect(page).to have_text('jenkins')
+      end
+
+      def create_build(stage, stage_idx, name, status)
+        create(:ci_build, pipeline: pipeline, stage: stage, stage_idx: stage_idx, name: name, status: status)
+      end
+    end
+
     describe 'POST /:project/pipelines' do
       let(:project) { create(:project) }
 
diff --git a/spec/views/projects/pipelines/show.html.haml_spec.rb b/spec/views/projects/pipelines/show.html.haml_spec.rb
deleted file mode 100644
index bb39ec8efbf4b..0000000000000
--- a/spec/views/projects/pipelines/show.html.haml_spec.rb
+++ /dev/null
@@ -1,67 +0,0 @@
-require 'spec_helper'
-
-describe 'projects/pipelines/show' do
-  include Devise::Test::ControllerHelpers
-
-  let(:user) { create(:user) }
-  let(:project) { create(:project, :repository) }
-
-  let(:pipeline) do
-    create(:ci_empty_pipeline,
-           project: project,
-           sha: project.commit.id,
-           user: user)
-  end
-
-  before do
-    controller.prepend_view_path('app/views/projects')
-
-    create_build('build', 0, 'build', :success)
-    create_build('test', 1, 'rspec 0:2', :pending)
-    create_build('test', 1, 'rspec 1:2', :running)
-    create_build('test', 1, 'spinach 0:2', :created)
-    create_build('test', 1, 'spinach 1:2', :created)
-    create_build('test', 1, 'audit', :created)
-    create_build('deploy', 2, 'production', :created)
-
-    create(:generic_commit_status, pipeline: pipeline, stage: 'external', name: 'jenkins', stage_idx: 3)
-
-    assign(:project, project)
-    assign(:pipeline, pipeline.present(current_user: user))
-    assign(:commit, project.commit)
-
-    allow(view).to receive(:can?).and_return(true)
-  end
-
-  it 'shows a graph with grouped stages' do
-    render
-
-    expect(rendered).to have_css('.js-pipeline-graph')
-    expect(rendered).to have_css('.js-grouped-pipeline-dropdown')
-
-    # header
-    expect(rendered).to have_text("##{pipeline.id}")
-    expect(rendered).to have_css('time', text: pipeline.created_at.strftime("%b %d, %Y"))
-    expect(rendered).to have_selector(%Q(img[alt$="#{pipeline.user.name}'s avatar"]))
-    expect(rendered).to have_link(pipeline.user.name, href: user_path(pipeline.user))
-
-    # stages
-    expect(rendered).to have_text('Build')
-    expect(rendered).to have_text('Test')
-    expect(rendered).to have_text('Deploy')
-    expect(rendered).to have_text('External')
-
-    # builds
-    expect(rendered).to have_text('rspec')
-    expect(rendered).to have_text('spinach')
-    expect(rendered).to have_text('rspec 0:2')
-    expect(rendered).to have_text('production')
-    expect(rendered).to have_text('jenkins')
-  end
-
-  private
-
-  def create_build(stage, stage_idx, name, status)
-    create(:ci_build, pipeline: pipeline, stage: stage, stage_idx: stage_idx, name: name, status: status)
-  end
-end
-- 
GitLab


From b1f753f7f252945708aa9e7f2f6c0f140ab09ed8 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Sun, 7 May 2017 15:24:15 +0000
Subject: [PATCH 350/363] Correctly stub application settings in signup_spec.rb

---
 spec/features/signup_spec.rb | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/spec/features/signup_spec.rb b/spec/features/signup_spec.rb
index 9fde8d6e5cfb7..d7b6dda4946ea 100644
--- a/spec/features/signup_spec.rb
+++ b/spec/features/signup_spec.rb
@@ -3,7 +3,7 @@
 feature 'Signup', feature: true do
   describe 'signup with no errors' do
     context "when sending confirmation email" do
-      before { allow_any_instance_of(ApplicationSetting).to receive(:send_user_confirmation_email).and_return(true) }
+      before { stub_application_setting(send_user_confirmation_email: true) }
 
       it 'creates the user account and sends a confirmation email' do
         user = build(:user)
@@ -23,7 +23,7 @@
     end
 
     context "when not sending confirmation email" do
-      before { allow_any_instance_of(ApplicationSetting).to receive(:send_user_confirmation_email).and_return(false) }
+      before { stub_application_setting(send_user_confirmation_email: false) }
 
       it 'creates the user account and goes to dashboard' do
         user = build(:user)
-- 
GitLab


From e6f008bb84754de3c0d70cce611aa6a30dba0c43 Mon Sep 17 00:00:00 2001
From: "Luke \"Jared\" Bennett" <lbennett@gitlab.com>
Date: Sun, 7 May 2017 15:34:27 +0000
Subject: [PATCH 351/363] Cast ENV['RSPEC_PROFILING_POSTGRES_URL'] to symbol in
 establish_connection call of rspec_profiling.rb

---
 config/initializers/rspec_profiling.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/config/initializers/rspec_profiling.rb b/config/initializers/rspec_profiling.rb
index a7efd74f09e93..b8ad3b1e51c75 100644
--- a/config/initializers/rspec_profiling.rb
+++ b/config/initializers/rspec_profiling.rb
@@ -1,7 +1,7 @@
 module RspecProfilingExt
   module PSQL
     def establish_connection
-      ::RspecProfiling::Collectors::PSQL::Result.establish_connection(ENV['RSPEC_PROFILING_POSTGRES_URL'])
+      ::RspecProfiling::Collectors::PSQL::Result.establish_connection(ENV['RSPEC_PROFILING_POSTGRES_URL'].to_sym)
     end
   end
 
-- 
GitLab


From e0e52fe5825736b60636d13b649e51a5bec0df57 Mon Sep 17 00:00:00 2001
From: "Z.J. van de Weg" <git@zjvandeweg.nl>
Date: Sun, 7 May 2017 18:26:39 +0200
Subject: [PATCH 352/363] Fix test|

---
 spec/features/projects/pipelines/pipelines_spec.rb | 4 +---
 1 file changed, 1 insertion(+), 3 deletions(-)

diff --git a/spec/features/projects/pipelines/pipelines_spec.rb b/spec/features/projects/pipelines/pipelines_spec.rb
index 8e820edee35e9..8cc96c7b00f88 100644
--- a/spec/features/projects/pipelines/pipelines_spec.rb
+++ b/spec/features/projects/pipelines/pipelines_spec.rb
@@ -397,11 +397,9 @@
 
       it 'shows a graph with grouped stages' do
         expect(page).to have_css('.js-pipeline-graph')
-        # expect(page).to have_css('.js-grouped-pipeline-dropdown')
 
         # header
         expect(page).to have_text("##{pipeline.id}")
-        expect(page).to have_css('time', text: pipeline.created_at.strftime("%b %d, %Y"))
         expect(page).to have_selector(%Q(img[alt$="#{pipeline.user.name}'s avatar"]))
         expect(page).to have_link(pipeline.user.name, href: user_path(pipeline.user))
 
@@ -414,7 +412,7 @@
         # builds
         expect(page).to have_text('rspec')
         expect(page).to have_text('spinach')
-        expect(page).to have_text('rspec 0:2')
+        expect(page).to have_text('rspec')
         expect(page).to have_text('production')
         expect(page).to have_text('jenkins')
       end
-- 
GitLab


From 34cd10979757cdad26056a532cf09c59a7f74ae9 Mon Sep 17 00:00:00 2001
From: Ahmad Sherif <me@ahmadsherif.com>
Date: Fri, 21 Apr 2017 15:02:21 +0200
Subject: [PATCH 353/363] Re-enable Gitaly commit_raw_diff feature

---
 GITALY_SERVER_VERSION              |  2 +-
 Gemfile                            |  2 +-
 Gemfile.lock                       |  6 ++--
 app/models/commit.rb               | 15 ++++----
 lib/gitlab/gitaly_client/commit.rb |  4 ++-
 spec/models/commit_spec.rb         | 55 +++++++++++++++---------------
 6 files changed, 43 insertions(+), 41 deletions(-)

diff --git a/GITALY_SERVER_VERSION b/GITALY_SERVER_VERSION
index a3df0a6959e15..78bc1abd14f2c 100644
--- a/GITALY_SERVER_VERSION
+++ b/GITALY_SERVER_VERSION
@@ -1 +1 @@
-0.8.0
+0.10.0
diff --git a/Gemfile b/Gemfile
index 5dbb0bdfddb5d..0cffee6db5623 100644
--- a/Gemfile
+++ b/Gemfile
@@ -367,6 +367,6 @@ gem 'vmstat', '~> 2.3.0'
 gem 'sys-filesystem', '~> 1.1.6'
 
 # Gitaly GRPC client
-gem 'gitaly', '~> 0.5.0'
+gem 'gitaly', '~> 0.6.0'
 
 gem 'toml-rb', '~> 0.3.15', require: false
diff --git a/Gemfile.lock b/Gemfile.lock
index 01c35a935f225..c0c56aa960266 100644
--- a/Gemfile.lock
+++ b/Gemfile.lock
@@ -263,7 +263,7 @@ GEM
       po_to_json (>= 1.0.0)
       rails (>= 3.2.0)
     gherkin-ruby (0.3.2)
-    gitaly (0.5.0)
+    gitaly (0.6.0)
       google-protobuf (~> 3.1)
       grpc (~> 1.0)
     github-linguist (4.7.6)
@@ -434,7 +434,7 @@ GEM
       rugged (~> 0.24)
     little-plugger (1.1.4)
     locale (2.1.2)
-    logging (2.1.0)
+    logging (2.2.2)
       little-plugger (~> 1.1)
       multi_json (~> 1.10)
     loofah (2.0.3)
@@ -922,7 +922,7 @@ DEPENDENCIES
   gettext (~> 3.2.2)
   gettext_i18n_rails (~> 1.8.0)
   gettext_i18n_rails_js (~> 1.2.0)
-  gitaly (~> 0.5.0)
+  gitaly (~> 0.6.0)
   github-linguist (~> 4.7.0)
   gitlab-flowdock-git-hook (~> 1.0.1)
   gitlab-markup (~> 1.5.1)
diff --git a/app/models/commit.rb b/app/models/commit.rb
index 88a015cdb77b4..9359b323ed424 100644
--- a/app/models/commit.rb
+++ b/app/models/commit.rb
@@ -326,13 +326,14 @@ def uri_type(path)
   end
 
   def raw_diffs(*args)
-    # NOTE: This feature is intentionally disabled until
-    # https://gitlab.com/gitlab-org/gitaly/issues/178 is resolved
-    # if Gitlab::GitalyClient.feature_enabled?(:commit_raw_diffs)
-    #   Gitlab::GitalyClient::Commit.diff_from_parent(self, *args)
-    # else
-    raw.diffs(*args)
-    # end
+    use_gitaly = Gitlab::GitalyClient.feature_enabled?(:commit_raw_diffs)
+    deltas_only = args.last.is_a?(Hash) && args.last[:deltas_only]
+
+    if use_gitaly && !deltas_only
+      Gitlab::GitalyClient::Commit.diff_from_parent(self, *args)
+    else
+      raw.diffs(*args)
+    end
   end
 
   def diffs(diff_options = nil)
diff --git a/lib/gitlab/gitaly_client/commit.rb b/lib/gitlab/gitaly_client/commit.rb
index 27db1e19bc125..0b001a9903d56 100644
--- a/lib/gitlab/gitaly_client/commit.rb
+++ b/lib/gitlab/gitaly_client/commit.rb
@@ -32,7 +32,9 @@ def diff_from_parent(commit, options = {})
           request = Gitaly::CommitDiffRequest.new(
             repository: gitaly_repo,
             left_commit_id: parent_id,
-            right_commit_id: commit.id
+            right_commit_id: commit.id,
+            ignore_whitespace_change: options.fetch(:ignore_whitespace_change, false),
+            paths: options.fetch(:paths, []),
           )
 
           Gitlab::Git::DiffCollection.new(stub.commit_diff(request), options)
diff --git a/spec/models/commit_spec.rb b/spec/models/commit_spec.rb
index 08b2169fea76b..852889d4540e4 100644
--- a/spec/models/commit_spec.rb
+++ b/spec/models/commit_spec.rb
@@ -389,32 +389,31 @@
     end
   end
 
-  # describe '#raw_diffs' do
-  # TODO: Uncomment when feature is reenabled
-  #   context 'Gitaly commit_raw_diffs feature enabled' do
-  #     before do
-  #       allow(Gitlab::GitalyClient).to receive(:feature_enabled?).with(:commit_raw_diffs).and_return(true)
-  #     end
-  #
-  #     context 'when a truthy deltas_only is not passed to args' do
-  #       it 'fetches diffs from Gitaly server' do
-  #         expect(Gitlab::GitalyClient::Commit).to receive(:diff_from_parent).
-  #           with(commit)
-  #
-  #         commit.raw_diffs
-  #       end
-  #     end
-  #
-  #     context 'when a truthy deltas_only is passed to args' do
-  #       it 'fetches diffs using Rugged' do
-  #         opts = { deltas_only: true }
-  #
-  #         expect(Gitlab::GitalyClient::Commit).not_to receive(:diff_from_parent)
-  #         expect(commit.raw).to receive(:diffs).with(opts)
-  #
-  #         commit.raw_diffs(opts)
-  #       end
-  #     end
-  #   end
-  # end
+  describe '#raw_diffs' do
+    context 'Gitaly commit_raw_diffs feature enabled' do
+      before do
+        allow(Gitlab::GitalyClient).to receive(:feature_enabled?).with(:commit_raw_diffs).and_return(true)
+      end
+
+      context 'when a truthy deltas_only is not passed to args' do
+        it 'fetches diffs from Gitaly server' do
+          expect(Gitlab::GitalyClient::Commit).to receive(:diff_from_parent).
+            with(commit)
+
+          commit.raw_diffs
+        end
+      end
+
+      context 'when a truthy deltas_only is passed to args' do
+        it 'fetches diffs using Rugged' do
+          opts = { deltas_only: true }
+
+          expect(Gitlab::GitalyClient::Commit).not_to receive(:diff_from_parent)
+          expect(commit.raw).to receive(:diffs).with(opts)
+
+          commit.raw_diffs(opts)
+        end
+      end
+    end
+  end
 end
-- 
GitLab


From 2f60a402d1fa98bb0090343f8180fc03b715467e Mon Sep 17 00:00:00 2001
From: Ahmad Sherif <me@ahmadsherif.com>
Date: Tue, 2 May 2017 13:01:28 +0200
Subject: [PATCH 354/363] Remove stubbing from Gitlab::GitalyClient::Commit
 specs

Closes gitaly#198
---
 spec/lib/gitlab/gitaly_client/commit_spec.rb | 11 +++--------
 1 file changed, 3 insertions(+), 8 deletions(-)

diff --git a/spec/lib/gitlab/gitaly_client/commit_spec.rb b/spec/lib/gitlab/gitaly_client/commit_spec.rb
index 58f11ff89063b..abe08ccdfa1f0 100644
--- a/spec/lib/gitlab/gitaly_client/commit_spec.rb
+++ b/spec/lib/gitlab/gitaly_client/commit_spec.rb
@@ -7,11 +7,6 @@
     let(:repository_message) { project.repository.gitaly_repository }
     let(:commit) { project.commit('913c66a37b4a45b9769037c55c2d238bd0942d2e') }
 
-    before do
-      allow(Gitaly::Diff::Stub).to receive(:new).and_return(diff_stub)
-      allow(diff_stub).to receive(:commit_diff).and_return([])
-    end
-
     context 'when a commit has a parent' do
       it 'sends an RPC request with the parent ID as left commit' do
         request = Gitaly::CommitDiffRequest.new(
@@ -20,7 +15,7 @@
           right_commit_id: commit.id,
         )
 
-        expect(diff_stub).to receive(:commit_diff).with(request)
+        expect_any_instance_of(Gitaly::Diff::Stub).to receive(:commit_diff).with(request)
 
         described_class.diff_from_parent(commit)
       end
@@ -35,7 +30,7 @@
           right_commit_id: initial_commit.id,
         )
 
-        expect(diff_stub).to receive(:commit_diff).with(request)
+        expect_any_instance_of(Gitaly::Diff::Stub).to receive(:commit_diff).with(request)
 
         described_class.diff_from_parent(initial_commit)
       end
@@ -50,7 +45,7 @@
     it 'passes options to Gitlab::Git::DiffCollection' do
       options = { max_files: 31, max_lines: 13 }
 
-      expect(Gitlab::Git::DiffCollection).to receive(:new).with([], options)
+      expect(Gitlab::Git::DiffCollection).to receive(:new).with(kind_of(Enumerable), options)
 
       described_class.diff_from_parent(commit, options)
     end
-- 
GitLab


From e08130223cfc48a6e662a87d32c3026088161de2 Mon Sep 17 00:00:00 2001
From: Stan Hu <stanhu@gmail.com>
Date: Sun, 7 May 2017 22:35:20 +0000
Subject: [PATCH 355/363] Revert "Merge branch
 'fix-rspec_profiling-establish_connection-string-deprecation' into 'master'"

This reverts merge request !11150
---
 config/initializers/rspec_profiling.rb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/config/initializers/rspec_profiling.rb b/config/initializers/rspec_profiling.rb
index b8ad3b1e51c75..a7efd74f09e93 100644
--- a/config/initializers/rspec_profiling.rb
+++ b/config/initializers/rspec_profiling.rb
@@ -1,7 +1,7 @@
 module RspecProfilingExt
   module PSQL
     def establish_connection
-      ::RspecProfiling::Collectors::PSQL::Result.establish_connection(ENV['RSPEC_PROFILING_POSTGRES_URL'].to_sym)
+      ::RspecProfiling::Collectors::PSQL::Result.establish_connection(ENV['RSPEC_PROFILING_POSTGRES_URL'])
     end
   end
 
-- 
GitLab


From 8df3997a92bffa2d29f3c559933a336b837cdb93 Mon Sep 17 00:00:00 2001
From: Zeger-Jan van de Weg <zegerjan@gitlab.com>
Date: Sun, 7 May 2017 22:35:56 +0000
Subject: [PATCH 356/363] Add Pipeline Schedules that supersedes experimental
 Trigger Schedule

---
 app/assets/javascripts/gl_field_error.js      |   4 +-
 app/assets/javascripts/gl_field_errors.js     |   9 +
 .../components/interval_pattern_input.js      | 143 ++++++++++++
 .../components/pipeline_schedules_callout.js  |  44 ++++
 .../components/target_branch_dropdown.js      |  42 ++++
 .../components/timezone_dropdown.js           |  56 +++++
 .../icons/intro_illustration.svg              |   1 +
 .../pipeline_schedule_form_bundle.js          |  21 ++
 .../pipeline_schedules_index_bundle.js        |   9 +
 .../stylesheets/pages/pipeline_schedules.scss |  71 ++++++
 .../projects/pipeline_schedules_controller.rb |  68 ++++++
 app/finders/pipeline_schedules_finder.rb      |  22 ++
 app/helpers/gitlab_routing_helper.rb          |  20 ++
 app/helpers/pipeline_schedules_helper.rb      |  11 +
 app/models/ci/pipeline.rb                     |   1 +
 ...igger_schedule.rb => pipeline_schedule.rb} |  21 +-
 app/models/ci/trigger.rb                      |   7 -
 app/models/project.rb                         |   1 +
 app/policies/ci/pipeline_schedule_policy.rb   |   4 +
 app/policies/project_policy.rb                |   8 +
 .../ci/create_pipeline_schedule_service.rb    |  13 ++
 app/services/ci/create_pipeline_service.rb    |   5 +-
 .../ci/create_trigger_request_service.rb      |   5 +-
 .../pipeline_schedules/_form.html.haml        |  32 +++
 .../_pipeline_schedule.html.haml              |  35 +++
 .../pipeline_schedules/_table.html.haml       |  12 +
 .../pipeline_schedules/_tabs.html.haml        |  18 ++
 .../pipeline_schedules/edit.html.haml         |   7 +
 .../pipeline_schedules/index.html.haml        |  24 ++
 .../projects/pipeline_schedules/new.html.haml |   7 +
 app/views/projects/pipelines/_head.html.haml  |   6 +
 app/views/projects/triggers/_form.html.haml   |  22 --
 app/views/projects/triggers/_index.html.haml  |   2 -
 .../projects/triggers/_trigger.html.haml      |   6 -
 app/workers/pipeline_schedule_worker.rb       |  19 ++
 app/workers/trigger_schedule_worker.rb        |  18 --
 .../zj-better-view-pipeline-schedule.yml      |   4 +
 config/gitlab.yml.example                     |   2 +-
 config/initializers/1_settings.rb             |   6 +-
 config/routes/project.rb                      |   6 +
 config/webpack.config.js                      |   2 +
 ...5112128_create_pipeline_schedules_table.rb |  28 +++
 ...remove_foreigh_key_ci_trigger_schedules.rb |  13 ++
 ...1_add_pipeline_schedule_id_to_pipelines.rb |   9 +
 ..._index_to_pipeline_pipeline_schedule_id.rb |  19 ++
 ...4_add_foreign_key_to_pipeline_schedules.rb |  15 ++
 ...gn_key_pipeline_schedules_and_pipelines.rb |  23 ++
 ...trigger_schedules_to_pipeline_schedules.rb |  41 ++++
 ...5130047_drop_ci_trigger_schedules_table.rb |  32 +++
 db/schema.rb                                  |  55 ++---
 doc/ci/img/pipeline_schedules_list.png        | Bin 0 -> 67555 bytes
 doc/ci/img/pipeline_schedules_new_form.png    | Bin 0 -> 49873 bytes
 doc/ci/pipeline_schedules.md                  |  40 ++++
 doc/ci/triggers/README.md                     |  38 ---
 .../triggers/img/trigger_schedule_create.png  | Bin 34264 -> 0 bytes
 doc/ci/triggers/img/trigger_schedule_edit.png | Bin 18524 -> 0 bytes
 .../trigger_schedule_updated_next_run_at.png  | Bin 21896 -> 0 bytes
 doc/user/project/settings/import_export.md    |   3 +-
 lib/gitlab/import_export.rb                   |   2 +-
 lib/gitlab/import_export/import_export.yml    |   6 +-
 lib/gitlab/import_export/relation_factory.rb  |   2 +-
 lib/gitlab/usage_data.rb                      |   1 +
 .../pipeline_schedules_controller_spec.rb     |  87 +++++++
 ...gger_schedules.rb => pipeline_schedule.rb} |  13 +-
 .../import_export/test_project_export.tar.gz  | Bin 679892 -> 681478 bytes
 .../projects/pipeline_schedules_spec.rb       | 140 +++++++++++
 .../security/project/internal_access_spec.rb  |  16 +-
 .../security/project/private_access_spec.rb   |  44 +++-
 .../security/project/public_access_spec.rb    |  16 +-
 spec/features/triggers_spec.rb                |  71 ------
 .../finders/pipeline_schedules_finder_spec.rb |  41 ++++
 .../interval_pattern_input_spec.js            | 217 ++++++++++++++++++
 .../pipeline_schedule_callout_spec.js         |  91 ++++++++
 spec/lib/gitlab/import_export/all_models.yml  |  13 +-
 .../import_export/safe_model_attributes.yml   |  16 +-
 spec/lib/gitlab/usage_data_spec.rb            |   1 +
 spec/models/ci/pipeline_schedule_spec.rb      | 112 +++++++++
 spec/models/ci/pipeline_spec.rb               |   1 +
 spec/models/ci/trigger_schedule_spec.rb       | 108 ---------
 spec/models/ci/trigger_spec.rb                |   1 -
 spec/models/project_spec.rb                   |   1 +
 spec/workers/pipeline_schedule_worker_spec.rb |  51 ++++
 spec/workers/trigger_schedule_worker_spec.rb  |  73 ------
 83 files changed, 1841 insertions(+), 413 deletions(-)
 create mode 100644 app/assets/javascripts/pipeline_schedules/components/interval_pattern_input.js
 create mode 100644 app/assets/javascripts/pipeline_schedules/components/pipeline_schedules_callout.js
 create mode 100644 app/assets/javascripts/pipeline_schedules/components/target_branch_dropdown.js
 create mode 100644 app/assets/javascripts/pipeline_schedules/components/timezone_dropdown.js
 create mode 100644 app/assets/javascripts/pipeline_schedules/icons/intro_illustration.svg
 create mode 100644 app/assets/javascripts/pipeline_schedules/pipeline_schedule_form_bundle.js
 create mode 100644 app/assets/javascripts/pipeline_schedules/pipeline_schedules_index_bundle.js
 create mode 100644 app/assets/stylesheets/pages/pipeline_schedules.scss
 create mode 100644 app/controllers/projects/pipeline_schedules_controller.rb
 create mode 100644 app/finders/pipeline_schedules_finder.rb
 create mode 100644 app/helpers/pipeline_schedules_helper.rb
 rename app/models/ci/{trigger_schedule.rb => pipeline_schedule.rb} (66%)
 create mode 100644 app/policies/ci/pipeline_schedule_policy.rb
 create mode 100644 app/services/ci/create_pipeline_schedule_service.rb
 create mode 100644 app/views/projects/pipeline_schedules/_form.html.haml
 create mode 100644 app/views/projects/pipeline_schedules/_pipeline_schedule.html.haml
 create mode 100644 app/views/projects/pipeline_schedules/_table.html.haml
 create mode 100644 app/views/projects/pipeline_schedules/_tabs.html.haml
 create mode 100644 app/views/projects/pipeline_schedules/edit.html.haml
 create mode 100644 app/views/projects/pipeline_schedules/index.html.haml
 create mode 100644 app/views/projects/pipeline_schedules/new.html.haml
 create mode 100644 app/workers/pipeline_schedule_worker.rb
 delete mode 100644 app/workers/trigger_schedule_worker.rb
 create mode 100644 changelogs/unreleased/zj-better-view-pipeline-schedule.yml
 create mode 100644 db/migrate/20170425112128_create_pipeline_schedules_table.rb
 create mode 100644 db/migrate/20170425112628_remove_foreigh_key_ci_trigger_schedules.rb
 create mode 100644 db/migrate/20170425114731_add_pipeline_schedule_id_to_pipelines.rb
 create mode 100644 db/migrate/20170506085040_add_index_to_pipeline_pipeline_schedule_id.rb
 create mode 100644 db/migrate/20170506091344_add_foreign_key_to_pipeline_schedules.rb
 create mode 100644 db/migrate/20170506185517_add_foreign_key_pipeline_schedules_and_pipelines.rb
 create mode 100644 db/post_migrate/20170425121605_migrate_trigger_schedules_to_pipeline_schedules.rb
 create mode 100644 db/post_migrate/20170425130047_drop_ci_trigger_schedules_table.rb
 create mode 100644 doc/ci/img/pipeline_schedules_list.png
 create mode 100644 doc/ci/img/pipeline_schedules_new_form.png
 create mode 100644 doc/ci/pipeline_schedules.md
 delete mode 100644 doc/ci/triggers/img/trigger_schedule_create.png
 delete mode 100644 doc/ci/triggers/img/trigger_schedule_edit.png
 delete mode 100644 doc/ci/triggers/img/trigger_schedule_updated_next_run_at.png
 create mode 100644 spec/controllers/projects/pipeline_schedules_controller_spec.rb
 rename spec/factories/ci/{trigger_schedules.rb => pipeline_schedule.rb} (70%)
 create mode 100644 spec/features/projects/pipeline_schedules_spec.rb
 create mode 100644 spec/finders/pipeline_schedules_finder_spec.rb
 create mode 100644 spec/javascripts/pipeline_schedules/interval_pattern_input_spec.js
 create mode 100644 spec/javascripts/pipeline_schedules/pipeline_schedule_callout_spec.js
 create mode 100644 spec/models/ci/pipeline_schedule_spec.rb
 delete mode 100644 spec/models/ci/trigger_schedule_spec.rb
 create mode 100644 spec/workers/pipeline_schedule_worker_spec.rb
 delete mode 100644 spec/workers/trigger_schedule_worker_spec.rb

diff --git a/app/assets/javascripts/gl_field_error.js b/app/assets/javascripts/gl_field_error.js
index 76de249ac3bb0..0add707525498 100644
--- a/app/assets/javascripts/gl_field_error.js
+++ b/app/assets/javascripts/gl_field_error.js
@@ -65,6 +65,7 @@ class GlFieldError {
     this.state = {
       valid: false,
       empty: true,
+      submitted: false,
     };
 
     this.initFieldValidation();
@@ -108,9 +109,10 @@ class GlFieldError {
     const currentValue = this.accessCurrentValue();
     this.state.valid = false;
     this.state.empty = currentValue === '';
-
+    this.state.submitted = true;
     this.renderValidity();
     this.form.focusOnFirstInvalid.apply(this.form);
+
     // For UX, wait til after first invalid submission to check each keyup
     this.inputElement.off('keyup.fieldValidator')
       .on('keyup.fieldValidator', this.updateValidity.bind(this));
diff --git a/app/assets/javascripts/gl_field_errors.js b/app/assets/javascripts/gl_field_errors.js
index 636258ec55565..ca3cec07a88cb 100644
--- a/app/assets/javascripts/gl_field_errors.js
+++ b/app/assets/javascripts/gl_field_errors.js
@@ -37,6 +37,15 @@ class GlFieldErrors {
     }
   }
 
+  /* Public method for triggering validity updates manually  */
+  updateFormValidityState() {
+    this.state.inputs.forEach((field) => {
+      if (field.state.submitted) {
+        field.updateValidity();
+      }
+    });
+  }
+
   focusOnFirstInvalid () {
     const firstInvalid = this.state.inputs.filter((input) => !input.inputDomElement.validity.valid)[0];
     firstInvalid.inputElement.focus();
diff --git a/app/assets/javascripts/pipeline_schedules/components/interval_pattern_input.js b/app/assets/javascripts/pipeline_schedules/components/interval_pattern_input.js
new file mode 100644
index 0000000000000..152e75b747e4b
--- /dev/null
+++ b/app/assets/javascripts/pipeline_schedules/components/interval_pattern_input.js
@@ -0,0 +1,143 @@
+import Vue from 'vue';
+
+const inputNameAttribute = 'schedule[cron]';
+
+export default {
+  props: {
+    initialCronInterval: {
+      type: String,
+      required: false,
+      default: '',
+    },
+  },
+  data() {
+    return {
+      inputNameAttribute,
+      cronInterval: this.initialCronInterval,
+      cronIntervalPresets: {
+        everyDay: '0 4 * * *',
+        everyWeek: '0 4 * * 0',
+        everyMonth: '0 4 1 * *',
+      },
+      cronSyntaxUrl: 'https://en.wikipedia.org/wiki/Cron',
+      customInputEnabled: false,
+    };
+  },
+  computed: {
+    showUnsetWarning() {
+      return this.cronInterval === '';
+    },
+    intervalIsPreset() {
+      return _.contains(this.cronIntervalPresets, this.cronInterval);
+    },
+    // The text input is editable when there's a custom interval, or when it's
+    // a preset interval and the user clicks the 'custom' radio button
+    isEditable() {
+      return !!(this.customInputEnabled || !this.intervalIsPreset);
+    },
+  },
+  methods: {
+    toggleCustomInput(shouldEnable) {
+      this.customInputEnabled = shouldEnable;
+
+      if (shouldEnable) {
+        // We need to change the value so other radios don't remain selected
+        // because the model (cronInterval) hasn't changed. The server trims it.
+        this.cronInterval = `${this.cronInterval} `;
+      }
+    },
+  },
+  created() {
+    if (this.intervalIsPreset) {
+      this.enableCustomInput = false;
+    }
+  },
+  watch: {
+    cronInterval() {
+      // updates field validation state when model changes, as
+      // glFieldError only updates on input.
+      Vue.nextTick(() => {
+        gl.pipelineScheduleFieldErrors.updateFormValidityState();
+      });
+    },
+  },
+  template: `
+    <div class="interval-pattern-form-group">
+      <input
+        id="custom"
+        class="label-light"
+        type="radio"
+        :name="inputNameAttribute"
+        :value="cronInterval"
+        :checked="isEditable"
+        @click="toggleCustomInput(true)"
+      />
+
+      <label for="custom">
+        Custom
+      </label>
+
+      <span class="cron-syntax-link-wrap">
+        (<a :href="cronSyntaxUrl" target="_blank">Cron syntax</a>)
+      </span>
+
+      <input
+        id="every-day"
+        class="label-light"
+        type="radio"
+        v-model="cronInterval"
+        :name="inputNameAttribute"
+        :value="cronIntervalPresets.everyDay"
+        @click="toggleCustomInput(false)"
+      />
+
+      <label class="label-light" for="every-day">
+        Every day (at 4:00am)
+      </label>
+
+      <input
+        id="every-week"
+        class="label-light"
+        type="radio"
+        v-model="cronInterval"
+        :name="inputNameAttribute"
+        :value="cronIntervalPresets.everyWeek"
+        @click="toggleCustomInput(false)"
+      />
+
+      <label class="label-light" for="every-week">
+        Every week (Sundays at 4:00am)
+      </label>
+
+      <input
+        id="every-month"
+        class="label-light"
+        type="radio"
+        v-model="cronInterval"
+        :name="inputNameAttribute"
+        :value="cronIntervalPresets.everyMonth"
+        @click="toggleCustomInput(false)"
+      />
+
+      <label class="label-light" for="every-month">
+        Every month (on the 1st at 4:00am)
+      </label>
+
+      <div class="cron-interval-input-wrapper col-md-6">
+        <input
+          id="schedule_cron"
+          class="form-control inline cron-interval-input"
+          type="text"
+          placeholder="Define a custom pattern with cron syntax"
+          required="true"
+          v-model="cronInterval"
+          :name="inputNameAttribute"
+          :disabled="!isEditable"
+        />
+      </div>
+      <span class="cron-unset-status col-md-3" v-if="showUnsetWarning">
+        Schedule not yet set
+      </span>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/pipeline_schedules/components/pipeline_schedules_callout.js b/app/assets/javascripts/pipeline_schedules/components/pipeline_schedules_callout.js
new file mode 100644
index 0000000000000..27ffe6ea3046a
--- /dev/null
+++ b/app/assets/javascripts/pipeline_schedules/components/pipeline_schedules_callout.js
@@ -0,0 +1,44 @@
+import Cookies from 'js-cookie';
+import illustrationSvg from '../icons/intro_illustration.svg';
+
+const cookieKey = 'pipeline_schedules_callout_dismissed';
+
+export default {
+  data() {
+    return {
+      illustrationSvg,
+      calloutDismissed: Cookies.get(cookieKey) === 'true',
+    };
+  },
+  methods: {
+    dismissCallout() {
+      this.calloutDismissed = true;
+      Cookies.set(cookieKey, this.calloutDismissed, { expires: 365 });
+    },
+  },
+  template: `
+    <div v-if="!calloutDismissed" class="pipeline-schedules-user-callout user-callout">
+      <div class="bordered-box landing content-block">
+        <button
+          id="dismiss-callout-btn"
+          class="btn btn-default close"
+          @click="dismissCallout">
+          <i class="fa fa-times"></i>
+        </button>
+        <div class="svg-container" v-html="illustrationSvg"></div>
+        <div class="user-callout-copy">
+          <h4>Scheduling Pipelines</h4>
+          <p> 
+              The pipelines schedule runs pipelines in the future, repeatedly, for specific branches or tags. 
+              Those scheduled pipelines will inherit limited project access based on their associated user.
+          </p>
+          <p> Learn more in the
+            <!-- FIXME -->
+            <a href="random.com">pipeline schedules documentation</a>.
+          </p>
+        </div>
+      </div>
+    </div>
+  `,
+};
+
diff --git a/app/assets/javascripts/pipeline_schedules/components/target_branch_dropdown.js b/app/assets/javascripts/pipeline_schedules/components/target_branch_dropdown.js
new file mode 100644
index 0000000000000..22e746ad2c328
--- /dev/null
+++ b/app/assets/javascripts/pipeline_schedules/components/target_branch_dropdown.js
@@ -0,0 +1,42 @@
+export default class TargetBranchDropdown {
+  constructor() {
+    this.$dropdown = $('.js-target-branch-dropdown');
+    this.$dropdownToggle = this.$dropdown.find('.dropdown-toggle-text');
+    this.$input = $('#schedule_ref');
+    this.initialValue = this.$input.val();
+    this.initDropdown();
+  }
+
+  initDropdown() {
+    this.$dropdown.glDropdown({
+      data: this.formatBranchesList(),
+      filterable: true,
+      selectable: true,
+      toggleLabel: item => item.name,
+      search: {
+        fields: ['name'],
+      },
+      clicked: cfg => this.updateInputValue(cfg),
+      text: item => item.name,
+    });
+
+    this.setDropdownToggle();
+  }
+
+  formatBranchesList() {
+    return this.$dropdown.data('data')
+      .map(val => ({ name: val }));
+  }
+
+  setDropdownToggle() {
+    if (this.initialValue) {
+      this.$dropdownToggle.text(this.initialValue);
+    }
+  }
+
+  updateInputValue({ selectedObj, e }) {
+    e.preventDefault();
+    this.$input.val(selectedObj.name);
+    gl.pipelineScheduleFieldErrors.updateFormValidityState();
+  }
+}
diff --git a/app/assets/javascripts/pipeline_schedules/components/timezone_dropdown.js b/app/assets/javascripts/pipeline_schedules/components/timezone_dropdown.js
new file mode 100644
index 0000000000000..c70e0502cf8d5
--- /dev/null
+++ b/app/assets/javascripts/pipeline_schedules/components/timezone_dropdown.js
@@ -0,0 +1,56 @@
+/* eslint-disable class-methods-use-this */
+
+export default class TimezoneDropdown {
+  constructor() {
+    this.$dropdown = $('.js-timezone-dropdown');
+    this.$dropdownToggle = this.$dropdown.find('.dropdown-toggle-text');
+    this.$input = $('#schedule_cron_timezone');
+    this.timezoneData = this.$dropdown.data('data');
+    this.initialValue = this.$input.val();
+    this.initDropdown();
+  }
+
+  initDropdown() {
+    this.$dropdown.glDropdown({
+      data: this.timezoneData,
+      filterable: true,
+      selectable: true,
+      toggleLabel: item => item.name,
+      search: {
+        fields: ['name'],
+      },
+      clicked: cfg => this.updateInputValue(cfg),
+      text: item => this.formatTimezone(item),
+    });
+
+    this.setDropdownToggle();
+  }
+
+  formatUtcOffset(offset) {
+    let prefix = '';
+
+    if (offset > 0) {
+      prefix = '+';
+    } else if (offset < 0) {
+      prefix = '-';
+    }
+
+    return `${prefix} ${Math.abs(offset / 3600)}`;
+  }
+
+  formatTimezone(item) {
+    return `[UTC ${this.formatUtcOffset(item.offset)}] ${item.name}`;
+  }
+
+  setDropdownToggle() {
+    if (this.initialValue) {
+      this.$dropdownToggle.text(this.initialValue);
+    }
+  }
+
+  updateInputValue({ selectedObj, e }) {
+    e.preventDefault();
+    this.$input.val(selectedObj.identifier);
+    gl.pipelineScheduleFieldErrors.updateFormValidityState();
+  }
+}
diff --git a/app/assets/javascripts/pipeline_schedules/icons/intro_illustration.svg b/app/assets/javascripts/pipeline_schedules/icons/intro_illustration.svg
new file mode 100644
index 0000000000000..26d1ff97b3e43
--- /dev/null
+++ b/app/assets/javascripts/pipeline_schedules/icons/intro_illustration.svg
@@ -0,0 +1 @@
+<svg width="140" height="102" viewBox="0 0 140 102" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"><title>illustration</title><defs><rect id="a" width="12.033" height="40.197" rx="3"/><rect id="b" width="12.033" height="40.197" rx="3"/></defs><g fill="none" fill-rule="evenodd"><g transform="translate(-.446)"><path d="M91.747 35.675v-6.039a2.996 2.996 0 0 0-2.999-3.005H54.635a2.997 2.997 0 0 0-2.999 3.005v6.039H40.092a3.007 3.007 0 0 0-2.996 3.005v34.187a2.995 2.995 0 0 0 2.996 3.005h11.544V79.9a2.996 2.996 0 0 0 2.999 3.005h34.113a2.997 2.997 0 0 0 2.999-3.005v-4.03h11.544a3.007 3.007 0 0 0 2.996-3.004V38.68a2.995 2.995 0 0 0-2.996-3.005H91.747z" stroke="#B5A7DD" stroke-width="2"/><rect stroke="#E5E5E5" stroke-width="2" fill="#FFF" x="21.556" y="38.69" width="98.27" height="34.167" rx="3"/><path d="M121.325 38.19c.55 0 .995.444.995 1.002 0 .554-.453 1.003-.995 1.003h-4.039a1.004 1.004 0 0 1 0-2.006h4.039zm9.044 0c.55 0 .996.444.996 1.002 0 .554-.454 1.003-.996 1.003h-4.038a1.004 1.004 0 0 1 0-2.006h4.038zm9.044 0c.55 0 .996.444.996 1.002 0 .554-.453 1.003-.996 1.003h-4.038a1.004 1.004 0 0 1 0-2.006h4.038zM121.325 71.854a1.004 1.004 0 0 1 0 2.006h-4.039a1.004 1.004 0 0 1 0-2.006h4.039zm9.044 0a1.004 1.004 0 0 1 0 2.006h-4.038a1.004 1.004 0 0 1 0-2.006h4.038zm9.044 0a1.004 1.004 0 0 1 0 2.006h-4.038a1.004 1.004 0 0 1 0-2.006h4.038z" fill="#E5E5E5"/><g transform="translate(110.3 35.675)"><use fill="#FFF" xlink:href="#a"/><rect stroke="#FDE5D8" stroke-width="2" x="1" y="1" width="10.033" height="38.197" rx="3"/><ellipse fill="#FC8A51" cx="6.017" cy="9.547" rx="1.504" ry="1.507"/><ellipse fill="#FC8A51" cx="6.017" cy="20.099" rx="1.504" ry="1.507"/><ellipse fill="#FC8A51" cx="6.017" cy="30.65" rx="1.504" ry="1.507"/></g><path d="M6.008 38.19c.55 0 .996.444.996 1.002 0 .554-.454 1.003-.996 1.003H1.97a1.004 1.004 0 0 1 0-2.006h4.038zm9.044 0c.55 0 .996.444.996 1.002 0 .554-.453 1.003-.996 1.003h-4.038a1.004 1.004 0 0 1 0-2.006h4.038zm9.045 0c.55 0 .995.444.995 1.002 0 .554-.453 1.003-.995 1.003h-4.039a1.004 1.004 0 0 1 0-2.006h4.039zM6.008 71.854a1.004 1.004 0 0 1 0 2.006H1.97a1.004 1.004 0 0 1 0-2.006h4.038zm9.044 0a1.004 1.004 0 0 1 0 2.006h-4.038a1.004 1.004 0 0 1 0-2.006h4.038zm9.045 0a1.004 1.004 0 0 1 0 2.006h-4.039a1.004 1.004 0 0 1 0-2.006h4.039z" fill="#E5E5E5"/><g transform="translate(19.05 35.675)"><use fill="#FFF" xlink:href="#b"/><rect stroke="#FDE5D8" stroke-width="2" x="1" y="1" width="10.033" height="38.197" rx="3"/><ellipse fill="#FC8A51" cx="6.017" cy="10.049" rx="1.504" ry="1.507"/><ellipse fill="#FC8A51" cx="6.017" cy="20.601" rx="1.504" ry="1.507"/><ellipse fill="#FC8A51" cx="6.017" cy="31.153" rx="1.504" ry="1.507"/></g><g transform="translate(47.096)"><g transform="translate(7.05)"><ellipse fill="#FC8A51" cx="17.548" cy="5.025" rx="4.512" ry="4.522"/><rect stroke="#B5A7DD" stroke-width="2" fill="#FFF" x="13.036" y="4.02" width="9.025" height="20.099" rx="1.5"/><rect stroke="#FDE5D8" stroke-width="2" fill="#FFF" y="4.02" width="35.096" height="4.02" rx="2.01"/><rect stroke="#6B4FBB" stroke-width="2" fill="#FFF" x="4.512" y="18.089" width="26.072" height="17.084" rx="1.5"/></g><rect stroke="#6B4FBB" stroke-width="2" fill="#FFF" transform="rotate(-45 43.117 35.117)" x="38.168" y="31.416" width="9.899" height="7.403" rx="3.702"/><ellipse stroke="#6B4FBB" stroke-width="2" fill="#FFF" cx="25" cy="55" rx="25" ry="25"/><ellipse stroke="#6B4FBB" stroke-width="2" fill="#FFF" cx="25" cy="55" rx="21" ry="21"/><rect stroke="#6B4FBB" stroke-width="2" fill="#FFF" x="43.05" y="53.281" width="2.95" height="1.538" rx=".769"/><rect stroke="#6B4FBB" stroke-width="2" fill="#FFF" x="4.305" y="53.281" width="2.95" height="1.538" rx=".769"/><rect stroke="#6B4FBB" stroke-width="2" fill="#FFF" transform="rotate(90 25.153 74.422)" x="23.677" y="73.653" width="2.95" height="1.538" rx=".769"/><rect stroke="#6B4FBB" stroke-width="2" fill="#FFF" transform="rotate(90 25.153 35.51)" x="23.844" y="34.742" width="2.616" height="1.538" rx=".769"/><path d="M13.362 42.502c-.124-.543.198-.854.74-.69l2.321.704c.533.161.643.592.235.972l-.22.206 7.06 7.572a1.002 1.002 0 1 1-1.467 1.368l-7.06-7.573-.118.11c-.402.375-.826.248-.952-.304l-.54-2.365zM21.606 67.576c-.408.38-.84.255-.968-.295l-.551-2.363c-.127-.542.191-.852.725-.69l.288.089 3.027-9.901a1.002 1.002 0 1 1 1.918.586l-3.027 9.901.154.047c.525.16.627.592.213.977l-1.779 1.65z" fill="#FC8A51"/><ellipse stroke="#6B4FBB" stroke-width="2" fill="#FFF" cx="25.099" cy="54.768" rx="2.507" ry="2.512"/></g></g><path d="M52.697 96.966a1.004 1.004 0 0 1 2.006 0v4.038a1.004 1.004 0 0 1-2.006 0v-4.038zm0-9.044a1.004 1.004 0 0 1 2.006 0v4.038a1.004 1.004 0 0 1-2.006 0v-4.038zM86.29 96.966c0-.55.444-.996 1.002-.996.554 0 1.003.454 1.003.996v4.038a1.004 1.004 0 0 1-2.006 0v-4.038zm0-9.044c0-.55.444-.996 1.002-.996.554 0 1.003.453 1.003.996v4.038a1.004 1.004 0 0 1-2.006 0v-4.038z" fill="#E5E5E5"/></g></svg>
\ No newline at end of file
diff --git a/app/assets/javascripts/pipeline_schedules/pipeline_schedule_form_bundle.js b/app/assets/javascripts/pipeline_schedules/pipeline_schedule_form_bundle.js
new file mode 100644
index 0000000000000..c60e77deccedf
--- /dev/null
+++ b/app/assets/javascripts/pipeline_schedules/pipeline_schedule_form_bundle.js
@@ -0,0 +1,21 @@
+import Vue from 'vue';
+import IntervalPatternInput from './components/interval_pattern_input';
+import TimezoneDropdown from './components/timezone_dropdown';
+import TargetBranchDropdown from './components/target_branch_dropdown';
+
+document.addEventListener('DOMContentLoaded', () => {
+  const IntervalPatternInputComponent = Vue.extend(IntervalPatternInput);
+  const intervalPatternMount = document.getElementById('interval-pattern-input');
+  const initialCronInterval = intervalPatternMount ? intervalPatternMount.dataset.initialInterval : '';
+
+  new IntervalPatternInputComponent({
+    propsData: {
+      initialCronInterval,
+    },
+  }).$mount(intervalPatternMount);
+
+  const formElement = document.getElementById('new-pipeline-schedule-form');
+  gl.timezoneDropdown = new TimezoneDropdown();
+  gl.targetBranchDropdown = new TargetBranchDropdown();
+  gl.pipelineScheduleFieldErrors = new gl.GlFieldErrors(formElement);
+});
diff --git a/app/assets/javascripts/pipeline_schedules/pipeline_schedules_index_bundle.js b/app/assets/javascripts/pipeline_schedules/pipeline_schedules_index_bundle.js
new file mode 100644
index 0000000000000..e36dc5db2abf2
--- /dev/null
+++ b/app/assets/javascripts/pipeline_schedules/pipeline_schedules_index_bundle.js
@@ -0,0 +1,9 @@
+import Vue from 'vue';
+import PipelineSchedulesCallout from './components/pipeline_schedules_callout';
+
+const PipelineSchedulesCalloutComponent = Vue.extend(PipelineSchedulesCallout);
+
+document.addEventListener('DOMContentLoaded', () => {
+  new PipelineSchedulesCalloutComponent()
+    .$mount('#scheduling-pipelines-callout');
+});
diff --git a/app/assets/stylesheets/pages/pipeline_schedules.scss b/app/assets/stylesheets/pages/pipeline_schedules.scss
new file mode 100644
index 0000000000000..0fee54a0d195c
--- /dev/null
+++ b/app/assets/stylesheets/pages/pipeline_schedules.scss
@@ -0,0 +1,71 @@
+.js-pipeline-schedule-form {
+  .dropdown-select,
+  .dropdown-menu-toggle {
+    width: 100%!important;
+  }
+
+  .gl-field-error {
+    margin: 10px 0 0;
+  }
+}
+
+.interval-pattern-form-group {
+  label {
+    margin-right: 10px;
+    font-size: 12px;
+
+    &[for='custom'] {
+      margin-right: 0;
+    }
+  }
+
+  .cron-interval-input-wrapper {
+    padding-left: 0;
+  }
+
+  .cron-interval-input {
+    margin: 10px 10px 0 0;
+  }
+
+  .cron-syntax-link-wrap {
+    margin-right: 10px;
+    font-size: 12px;
+  }
+
+  .cron-unset-status {
+    padding-top: 16px;
+    margin-left: -16px;
+    color: $gl-text-color-secondary;
+    font-size: 12px;
+    font-weight: 600;
+  }
+}
+
+.pipeline-schedule-table-row {
+  .branch-name-cell {
+    max-width: 300px;
+    white-space: nowrap;
+    overflow: hidden;
+    text-overflow: ellipsis;
+  }
+
+  .next-run-cell {
+    color: $gl-text-color-secondary;
+  }
+
+  a {
+    color: $text-color;
+  }
+}
+
+.pipeline-schedules-user-callout {
+  .bordered-box.content-block {
+    border: 1px solid $border-color;
+    background-color: transparent;
+    padding: 16px;
+  }
+
+  #dismiss-callout-btn {
+    color: $gl-text-color;
+  }
+}
diff --git a/app/controllers/projects/pipeline_schedules_controller.rb b/app/controllers/projects/pipeline_schedules_controller.rb
new file mode 100644
index 0000000000000..1616b2cb6b81c
--- /dev/null
+++ b/app/controllers/projects/pipeline_schedules_controller.rb
@@ -0,0 +1,68 @@
+class Projects::PipelineSchedulesController < Projects::ApplicationController
+  before_action :authorize_read_pipeline_schedule!
+  before_action :authorize_create_pipeline_schedule!, only: [:new, :create, :edit, :take_ownership, :update]
+  before_action :authorize_admin_pipeline_schedule!, only: [:destroy]
+
+  before_action :schedule, only: [:edit, :update, :destroy, :take_ownership]
+
+  def index
+    @scope = params[:scope]
+    @all_schedules = PipelineSchedulesFinder.new(@project).execute
+    @schedules = PipelineSchedulesFinder.new(@project).execute(scope: params[:scope])
+      .includes(:last_pipeline)
+  end
+
+  def new
+    @schedule = project.pipeline_schedules.new
+  end
+
+  def create
+    @schedule = Ci::CreatePipelineScheduleService
+      .new(@project, current_user, schedule_params)
+      .execute
+
+    if @schedule.persisted?
+      redirect_to pipeline_schedules_path(@project)
+    else
+      render :new
+    end
+  end
+
+  def edit
+  end
+
+  def update
+    if schedule.update(schedule_params)
+      redirect_to namespace_project_pipeline_schedules_path(@project.namespace.becomes(Namespace), @project)
+    else
+      render :edit
+    end
+  end
+
+  def take_ownership
+    if schedule.update(owner: current_user)
+      redirect_to pipeline_schedules_path(@project)
+    else
+      redirect_to pipeline_schedules_path(@project), alert: "Failed to change the owner"
+    end
+  end
+
+  def destroy
+    if schedule.destroy
+      redirect_to pipeline_schedules_path(@project)
+    else
+      redirect_to pipeline_schedules_path(@project), alert: "Failed to remove the pipeline schedule"
+    end
+  end
+
+  private
+
+  def schedule
+    @schedule ||= project.pipeline_schedules.find(params[:id])
+  end
+
+  def schedule_params
+    params.require(:schedule)
+      .permit(:description, :cron, :cron_timezone, :ref, :active)
+  end
+end
diff --git a/app/finders/pipeline_schedules_finder.rb b/app/finders/pipeline_schedules_finder.rb
new file mode 100644
index 0000000000000..2ac4289fbbef5
--- /dev/null
+++ b/app/finders/pipeline_schedules_finder.rb
@@ -0,0 +1,22 @@
+class PipelineSchedulesFinder
+  attr_reader :project, :pipeline_schedules
+
+  def initialize(project)
+    @project = project
+    @pipeline_schedules = project.pipeline_schedules
+  end
+
+  def execute(scope: nil)
+    scoped_schedules =
+      case scope
+      when 'active'
+        pipeline_schedules.active
+      when 'inactive'
+        pipeline_schedules.inactive
+      else
+        pipeline_schedules
+      end
+
+    scoped_schedules.order(id: :desc)
+  end
+end
diff --git a/app/helpers/gitlab_routing_helper.rb b/app/helpers/gitlab_routing_helper.rb
index 3769830de2a1f..bcf71bc347b53 100644
--- a/app/helpers/gitlab_routing_helper.rb
+++ b/app/helpers/gitlab_routing_helper.rb
@@ -221,6 +221,26 @@ def artifacts_action_path(path, project, build)
     end
   end
 
+  # Pipeline Schedules
+  def pipeline_schedules_path(project, *args)
+    namespace_project_pipeline_schedules_path(project.namespace, project, *args)
+  end
+
+  def pipeline_schedule_path(schedule, *args)
+    project = schedule.project
+    namespace_project_pipeline_schedule_path(project.namespace, project, schedule, *args)
+  end
+
+  def edit_pipeline_schedule_path(schedule)
+    project = schedule.project
+    edit_namespace_project_pipeline_schedule_path(project.namespace, project, schedule)
+  end
+
+  def take_ownership_pipeline_schedule_path(schedule, *args)
+    project = schedule.project
+    take_ownership_namespace_project_pipeline_schedule_path(project.namespace, project, schedule, *args)
+  end
+
   # Settings
   def project_settings_integrations_path(project, *args)
     namespace_project_settings_integrations_path(project.namespace, project, *args)
diff --git a/app/helpers/pipeline_schedules_helper.rb b/app/helpers/pipeline_schedules_helper.rb
new file mode 100644
index 0000000000000..fee1edc2a1bba
--- /dev/null
+++ b/app/helpers/pipeline_schedules_helper.rb
@@ -0,0 +1,11 @@
+module PipelineSchedulesHelper
+  def timezone_data
+    ActiveSupport::TimeZone.all.map do |timezone| 
+      { 
+        name: timezone.name, 
+        offset: timezone.utc_offset, 
+        identifier: timezone.tzinfo.identifier 
+      }
+    end
+  end
+end
diff --git a/app/models/ci/pipeline.rb b/app/models/ci/pipeline.rb
index 4be4aa9ffe2c2..db994b861e59d 100644
--- a/app/models/ci/pipeline.rb
+++ b/app/models/ci/pipeline.rb
@@ -9,6 +9,7 @@ class Pipeline < ActiveRecord::Base
     belongs_to :project
     belongs_to :user
     belongs_to :auto_canceled_by, class_name: 'Ci::Pipeline'
+    belongs_to :pipeline_schedule, class_name: 'Ci::PipelineSchedule'
 
     has_many :auto_canceled_pipelines, class_name: 'Ci::Pipeline', foreign_key: 'auto_canceled_by_id'
     has_many :auto_canceled_jobs, class_name: 'CommitStatus', foreign_key: 'auto_canceled_by_id'
diff --git a/app/models/ci/trigger_schedule.rb b/app/models/ci/pipeline_schedule.rb
similarity index 66%
rename from app/models/ci/trigger_schedule.rb
rename to app/models/ci/pipeline_schedule.rb
index 012a18eb43903..6d7cc83971e00 100644
--- a/app/models/ci/trigger_schedule.rb
+++ b/app/models/ci/pipeline_schedule.rb
@@ -1,24 +1,35 @@
 module Ci
-  class TriggerSchedule < ActiveRecord::Base
+  class PipelineSchedule < ActiveRecord::Base
     extend Ci::Model
     include Importable
 
     acts_as_paranoid
 
     belongs_to :project
-    belongs_to :trigger
+    belongs_to :owner, class_name: 'User'
+    has_one :last_pipeline, -> { order(id: :desc) }, class_name: 'Ci::Pipeline'
+    has_many :pipelines
 
-    validates :trigger, presence: { unless: :importing? }
     validates :cron, unless: :importing_or_inactive?, cron: true, presence: { unless: :importing_or_inactive? }
     validates :cron_timezone, cron_timezone: true, presence: { unless: :importing_or_inactive? }
     validates :ref, presence: { unless: :importing_or_inactive? }
+    validates :description, presence: true
 
     before_save :set_next_run_at
 
     scope :active, -> { where(active: true) }
+    scope :inactive, -> { where(active: false) }
+
+    def owned_by?(current_user)
+      owner == current_user
+    end
+
+    def inactive?
+      !active?
+    end
 
     def importing_or_inactive?
-      importing? || !active?
+      importing? || inactive?
     end
 
     def set_next_run_at
@@ -32,7 +43,7 @@ def schedule_next_run!
     end
 
     def real_next_run(
-        worker_cron: Settings.cron_jobs['trigger_schedule_worker']['cron'],
+        worker_cron: Settings.cron_jobs['pipeline_schedule_worker']['cron'],
         worker_time_zone: Time.zone.name)
       Gitlab::Ci::CronParser.new(worker_cron, worker_time_zone)
                             .next_time_from(next_run_at)
diff --git a/app/models/ci/trigger.rb b/app/models/ci/trigger.rb
index 2f64f70685a86..6df41a3f301e2 100644
--- a/app/models/ci/trigger.rb
+++ b/app/models/ci/trigger.rb
@@ -8,14 +8,11 @@ class Trigger < ActiveRecord::Base
     belongs_to :owner, class_name: "User"
 
     has_many :trigger_requests
-    has_one :trigger_schedule, dependent: :destroy
 
     validates :token, presence: true, uniqueness: true
 
     before_validation :set_default_values
 
-    accepts_nested_attributes_for :trigger_schedule
-
     def set_default_values
       self.token = SecureRandom.hex(15) if self.token.blank?
     end
@@ -39,9 +36,5 @@ def legacy?
     def can_access_project?
       self.owner_id.blank? || Ability.allowed?(self.owner, :create_build, project)
     end
-
-    def trigger_schedule
-      super || build_trigger_schedule(project: project)
-    end
   end
 end
diff --git a/app/models/project.rb b/app/models/project.rb
index edbca3b537b33..a0413b4e6515a 100644
--- a/app/models/project.rb
+++ b/app/models/project.rb
@@ -178,6 +178,7 @@ def update_forks_visibility_level
   has_many :triggers, dependent: :destroy, class_name: 'Ci::Trigger'
   has_many :environments, dependent: :destroy
   has_many :deployments, dependent: :destroy
+  has_many :pipeline_schedules, dependent: :destroy, class_name: 'Ci::PipelineSchedule'
 
   has_many :active_runners, -> { active }, through: :runner_projects, source: :runner, class_name: 'Ci::Runner'
 
diff --git a/app/policies/ci/pipeline_schedule_policy.rb b/app/policies/ci/pipeline_schedule_policy.rb
new file mode 100644
index 0000000000000..1877e89bb23f1
--- /dev/null
+++ b/app/policies/ci/pipeline_schedule_policy.rb
@@ -0,0 +1,4 @@
+module Ci
+  class PipelineSchedulePolicy < PipelinePolicy
+  end
+end
diff --git a/app/policies/project_policy.rb b/app/policies/project_policy.rb
index 5baac9ebe4b25..8f25ac30a2288 100644
--- a/app/policies/project_policy.rb
+++ b/app/policies/project_policy.rb
@@ -46,6 +46,7 @@ def guest_access!
 
     if project.public_builds?
       can! :read_pipeline
+      can! :read_pipeline_schedule
       can! :read_build
     end
   end
@@ -63,6 +64,7 @@ def reporter_access!
     can! :read_build
     can! :read_container_image
     can! :read_pipeline
+    can! :read_pipeline_schedule
     can! :read_environment
     can! :read_deployment
     can! :read_merge_request
@@ -83,6 +85,8 @@ def developer_access!
     can! :update_build
     can! :create_pipeline
     can! :update_pipeline
+    can! :create_pipeline_schedule
+    can! :update_pipeline_schedule
     can! :create_merge_request
     can! :create_wiki
     can! :push_code
@@ -108,6 +112,7 @@ def master_access!
     can! :admin_build
     can! :admin_container_image
     can! :admin_pipeline
+    can! :admin_pipeline_schedule
     can! :admin_environment
     can! :admin_deployment
     can! :admin_pages
@@ -120,6 +125,7 @@ def public_access!
     can! :fork_project
     can! :read_commit_status
     can! :read_pipeline
+    can! :read_pipeline_schedule
     can! :read_container_image
     can! :build_download_code
     can! :build_read_container_image
@@ -198,6 +204,7 @@ def disabled_features!
     unless project.feature_available?(:builds, user) && repository_enabled
       cannot!(*named_abilities(:build))
       cannot!(*named_abilities(:pipeline))
+      cannot!(*named_abilities(:pipeline_schedule))
       cannot!(*named_abilities(:environment))
       cannot!(*named_abilities(:deployment))
     end
@@ -277,6 +284,7 @@ def base_readonly_access!
     can! :read_merge_request
     can! :read_note
     can! :read_pipeline
+    can! :read_pipeline_schedule
     can! :read_commit_status
     can! :read_container_image
     can! :download_code
diff --git a/app/services/ci/create_pipeline_schedule_service.rb b/app/services/ci/create_pipeline_schedule_service.rb
new file mode 100644
index 0000000000000..cd40deb61871b
--- /dev/null
+++ b/app/services/ci/create_pipeline_schedule_service.rb
@@ -0,0 +1,13 @@
+module Ci
+  class CreatePipelineScheduleService < BaseService
+    def execute
+      project.pipeline_schedules.create(pipeline_schedule_params)
+    end
+
+    private
+
+    def pipeline_schedule_params
+      params.merge(owner: current_user)
+    end
+  end
+end
diff --git a/app/services/ci/create_pipeline_service.rb b/app/services/ci/create_pipeline_service.rb
index 21350be555714..ccdda08d8859d 100644
--- a/app/services/ci/create_pipeline_service.rb
+++ b/app/services/ci/create_pipeline_service.rb
@@ -2,7 +2,7 @@ module Ci
   class CreatePipelineService < BaseService
     attr_reader :pipeline
 
-    def execute(ignore_skip_ci: false, save_on_errors: true, trigger_request: nil)
+    def execute(ignore_skip_ci: false, save_on_errors: true, trigger_request: nil, schedule: nil)
       @pipeline = Ci::Pipeline.new(
         project: project,
         ref: ref,
@@ -10,7 +10,8 @@ def execute(ignore_skip_ci: false, save_on_errors: true, trigger_request: nil)
         before_sha: before_sha,
         tag: tag?,
         trigger_requests: Array(trigger_request),
-        user: current_user
+        user: current_user,
+        pipeline_schedule: schedule
       )
 
       unless project.builds_enabled?
diff --git a/app/services/ci/create_trigger_request_service.rb b/app/services/ci/create_trigger_request_service.rb
index dca5aa9f5d702..8362f01ddb8bc 100644
--- a/app/services/ci/create_trigger_request_service.rb
+++ b/app/services/ci/create_trigger_request_service.rb
@@ -5,9 +5,8 @@ def execute(project, trigger, ref, variables = nil)
 
       pipeline = Ci::CreatePipelineService.new(project, trigger.owner, ref: ref).
         execute(ignore_skip_ci: true, trigger_request: trigger_request)
-      if pipeline.persisted?
-        trigger_request
-      end
+
+      trigger_request if pipeline.persisted?
     end
   end
 end
diff --git a/app/views/projects/pipeline_schedules/_form.html.haml b/app/views/projects/pipeline_schedules/_form.html.haml
new file mode 100644
index 0000000000000..4a21cce024e40
--- /dev/null
+++ b/app/views/projects/pipeline_schedules/_form.html.haml
@@ -0,0 +1,32 @@
+- content_for :page_specific_javascripts do
+  = page_specific_javascript_bundle_tag('schedule_form')
+
+= form_for [@project.namespace.becomes(Namespace), @project, @schedule], as: :schedule, html: { id: "new-pipeline-schedule-form", class: "form-horizontal js-pipeline-schedule-form" } do |f|
+  = form_errors(@schedule)
+  .form-group
+    .col-md-6
+      = f.label :description, 'Description', class: 'label-light'
+      = f.text_field :description, class: 'form-control', required: true, autofocus: true, placeholder: 'Provide a short description for this pipeline'
+  .form-group
+    .col-md-12
+      = f.label :cron, 'Interval Pattern', class: 'label-light'
+      #interval-pattern-input{ data: { initial_interval: @schedule.cron } }
+  .form-group
+    .col-md-6
+      = f.label :cron_timezone, 'Cron Timezone', class: 'label-light'
+      = dropdown_tag("Select a timezone", options: { toggle_class: 'btn js-timezone-dropdown', title: "Select a timezone", filter: true, placeholder: "Filter", data: { data: timezone_data } } )
+      = f.text_field :cron_timezone, value: @schedule.cron_timezone, id: 'schedule_cron_timezone', class: 'hidden', name: 'schedule[cron_timezone]', required: true
+  .form-group
+    .col-md-6
+      = f.label :ref, 'Target Branch', class: 'label-light'
+      = dropdown_tag("Select target branch", options: { toggle_class: 'btn js-target-branch-dropdown', title: "Select target branch", filter: true, placeholder: "Filter", data: { data: @project.repository.branch_names } } )
+      = f.text_field :ref, value: @schedule.ref, id: 'schedule_ref', class: 'hidden', name: 'schedule[ref]', required: true
+  .form-group
+    .col-md-6
+      = f.label  :active, 'Activated', class: 'label-light'
+      %div
+        = f.check_box :active, required: false, value: @schedule.active?
+        active
+  .footer-block.row-content-block
+    = f.submit 'Save pipeline schedule', class: 'btn btn-create', tabindex: 3
+    = link_to 'Cancel', pipeline_schedules_path(@project), class: 'btn btn-cancel'
diff --git a/app/views/projects/pipeline_schedules/_pipeline_schedule.html.haml b/app/views/projects/pipeline_schedules/_pipeline_schedule.html.haml
new file mode 100644
index 0000000000000..1406868488fa1
--- /dev/null
+++ b/app/views/projects/pipeline_schedules/_pipeline_schedule.html.haml
@@ -0,0 +1,35 @@
+- if pipeline_schedule
+  %tr.pipeline-schedule-table-row
+    %td
+      = pipeline_schedule.description
+    %td.branch-name-cell
+      = icon('code-fork')
+      = link_to pipeline_schedule.ref, namespace_project_commits_path(@project.namespace, @project, pipeline_schedule.ref), class: "branch-name"
+    %td
+      - if pipeline_schedule.last_pipeline
+        .status-icon-container{ class: "ci-status-icon-#{pipeline_schedule.last_pipeline.status}" }
+          = link_to namespace_project_pipeline_path(@project.namespace, @project, pipeline_schedule.last_pipeline.id) do
+            = ci_icon_for_status(pipeline_schedule.last_pipeline.status)
+      - else
+        None
+    %td.next-run-cell
+      - if pipeline_schedule.active?
+        = time_ago_with_tooltip(pipeline_schedule.next_run_at)
+      - else
+        Inactive
+    %td
+      - if pipeline_schedule.owner
+        = image_tag avatar_icon(pipeline_schedule.owner, 20), class: "avatar s20"
+        = link_to user_path(pipeline_schedule.owner) do
+          = pipeline_schedule.owner&.name
+    %td
+      .pull-right.btn-group
+        - if can?(current_user, :update_pipeline_schedule, @project) && !pipeline_schedule.owned_by?(current_user)
+          = link_to take_ownership_pipeline_schedule_path(pipeline_schedule), method: :post, title: 'Take Ownership', class: 'btn' do
+            Take ownership
+        - if can?(current_user, :update_pipeline_schedule, pipeline_schedule)
+          = link_to edit_pipeline_schedule_path(pipeline_schedule), title: 'Edit', class: 'btn' do
+            = icon('pencil')
+        - if can?(current_user, :admin_pipeline_schedule, pipeline_schedule)
+          = link_to pipeline_schedule_path(pipeline_schedule), title: 'Delete', method: :delete, class: 'btn btn-remove', data: { confirm: "Are you sure you want to cancel this pipeline?" } do
+            = icon('trash')
diff --git a/app/views/projects/pipeline_schedules/_table.html.haml b/app/views/projects/pipeline_schedules/_table.html.haml
new file mode 100644
index 0000000000000..25c7604eb24d2
--- /dev/null
+++ b/app/views/projects/pipeline_schedules/_table.html.haml
@@ -0,0 +1,12 @@
+.table-holder
+  %table.table.ci-table
+    %thead
+      %tr
+        %th Description
+        %th Target
+        %th Last Pipeline
+        %th Next Run
+        %th Owner
+        %th
+
+    = render partial: "pipeline_schedule", collection: @schedules
diff --git a/app/views/projects/pipeline_schedules/_tabs.html.haml b/app/views/projects/pipeline_schedules/_tabs.html.haml
new file mode 100644
index 0000000000000..2a1fb16876a19
--- /dev/null
+++ b/app/views/projects/pipeline_schedules/_tabs.html.haml
@@ -0,0 +1,18 @@
+%ul.nav-links
+  %li{ class: active_when(scope.nil?) }>
+    = link_to schedule_path_proc.call(nil) do
+      All
+      %span.badge.js-totalbuilds-count
+        = number_with_delimiter(all_schedules.count(:id))
+
+  %li{ class: active_when(scope == 'active') }>
+    = link_to schedule_path_proc.call('active') do
+      Active
+      %span.badge
+        = number_with_delimiter(all_schedules.active.count(:id))
+
+  %li{ class: active_when(scope == 'inactive') }>
+    = link_to schedule_path_proc.call('inactive') do
+      Inactive
+      %span.badge
+        = number_with_delimiter(all_schedules.inactive.count(:id))
diff --git a/app/views/projects/pipeline_schedules/edit.html.haml b/app/views/projects/pipeline_schedules/edit.html.haml
new file mode 100644
index 0000000000000..e16fe0b7a9873
--- /dev/null
+++ b/app/views/projects/pipeline_schedules/edit.html.haml
@@ -0,0 +1,7 @@
+- page_title "Edit", @schedule.description, "Pipeline Schedule"
+
+%h3.page-title
+  Edit Pipeline Schedule #{@schedule.id}
+%hr
+
+= render "form"
diff --git a/app/views/projects/pipeline_schedules/index.html.haml b/app/views/projects/pipeline_schedules/index.html.haml
new file mode 100644
index 0000000000000..dd35c3055f204
--- /dev/null
+++ b/app/views/projects/pipeline_schedules/index.html.haml
@@ -0,0 +1,24 @@
+- content_for :page_specific_javascripts do
+  = page_specific_javascript_bundle_tag('schedules_index')
+
+- @no_container = true
+- page_title "Pipeline Schedules"
+= render "projects/pipelines/head"
+
+%div{ class: container_class }
+  #scheduling-pipelines-callout
+  .top-area
+    - schedule_path_proc = ->(scope) { pipeline_schedules_path(@project, scope: scope) }
+    = render "tabs", schedule_path_proc: schedule_path_proc, all_schedules: @all_schedules, scope: @scope
+
+    .nav-controls
+      = link_to new_namespace_project_pipeline_schedule_path(@project.namespace, @project), class: 'btn btn-create' do
+        %span New Schedule
+
+  - if @schedules.present?
+    %ul.content-list
+      = render partial: "table"
+  - else
+    .light-well
+      .nothing-here-block No schedules
+
diff --git a/app/views/projects/pipeline_schedules/new.html.haml b/app/views/projects/pipeline_schedules/new.html.haml
new file mode 100644
index 0000000000000..b89e170ad3c55
--- /dev/null
+++ b/app/views/projects/pipeline_schedules/new.html.haml
@@ -0,0 +1,7 @@
+- page_title "New Pipeline Schedule"
+
+%h3.page-title
+  Schedule a new pipeline
+%hr
+
+= render "form"
diff --git a/app/views/projects/pipelines/_head.html.haml b/app/views/projects/pipelines/_head.html.haml
index b0dac9de1c61c..db9d77dba1604 100644
--- a/app/views/projects/pipelines/_head.html.haml
+++ b/app/views/projects/pipelines/_head.html.haml
@@ -15,6 +15,12 @@
               %span
                 Jobs
 
+        - if project_nav_tab? :pipelines
+          = nav_link(controller: :pipeline_schedules) do
+            = link_to pipeline_schedules_path(@project), title: 'Schedules', class: 'shortcuts-builds' do
+              %span
+                Schedules
+
         - if project_nav_tab? :environments
           = nav_link(controller: :environments) do
             = link_to project_environments_path(@project), title: 'Environments', class: 'shortcuts-environments' do
diff --git a/app/views/projects/triggers/_form.html.haml b/app/views/projects/triggers/_form.html.haml
index 70d654fa9a0ad..5f708b3a2eddb 100644
--- a/app/views/projects/triggers/_form.html.haml
+++ b/app/views/projects/triggers/_form.html.haml
@@ -8,26 +8,4 @@
   .form-group
     = f.label :key, "Description", class: "label-light"
     = f.text_field :description, class: "form-control", required: true, title: 'Trigger description is required.', placeholder: "Trigger description"
-    - if @trigger.persisted?
-      %hr
-      = f.fields_for :trigger_schedule do |schedule_fields|
-        = schedule_fields.hidden_field :id
-        .form-group
-          .checkbox
-            = schedule_fields.label :active do
-              = schedule_fields.check_box :active
-              %strong Schedule trigger (experimental)
-            .help-block
-              If checked, this trigger will be executed periodically according to cron and timezone.
-              = link_to icon('question-circle'), help_page_path('ci/triggers/README', anchor: 'using-scheduled-triggers')
-        .form-group
-          = schedule_fields.label :cron, "Cron", class: "label-light"
-          = schedule_fields.text_field :cron, class: "form-control", title: 'Cron specification is required.', placeholder: "0 1 * * *"
-        .form-group
-          = schedule_fields.label :cron, "Timezone", class: "label-light"
-          = schedule_fields.text_field :cron_timezone, class: "form-control", title: 'Timezone is required.', placeholder: "UTC"
-        .form-group
-          = schedule_fields.label :ref, "Branch or tag", class: "label-light"
-          = schedule_fields.text_field :ref, class: "form-control", title: 'Branch or tag is required.', placeholder: "master"
-          .help-block Existing branch name, tag
   = f.submit btn_text, class: "btn btn-save"
diff --git a/app/views/projects/triggers/_index.html.haml b/app/views/projects/triggers/_index.html.haml
index 84e945ee0df53..cc74e50a5e37a 100644
--- a/app/views/projects/triggers/_index.html.haml
+++ b/app/views/projects/triggers/_index.html.haml
@@ -22,8 +22,6 @@
                 %th
                   %strong Last used
                 %th
-                  %strong Next run at
-                %th
               = render partial: 'projects/triggers/trigger', collection: @triggers, as: :trigger
         - else
           %p.settings-message.text-center.append-bottom-default
diff --git a/app/views/projects/triggers/_trigger.html.haml b/app/views/projects/triggers/_trigger.html.haml
index ebd91a8e2af95..9b5f63ae81a5c 100644
--- a/app/views/projects/triggers/_trigger.html.haml
+++ b/app/views/projects/triggers/_trigger.html.haml
@@ -29,12 +29,6 @@
     - else
       Never
 
-  %td
-    - if trigger.trigger_schedule&.active?
-      = trigger.trigger_schedule.real_next_run
-    - else
-      Never
-
   %td.text-right.trigger-actions
     - take_ownership_confirmation = "By taking ownership you will bind this trigger to your user account. With this the trigger will have access to all your projects as if it was you. Are you sure?"
     - revoke_trigger_confirmation = "By revoking a trigger you will break any processes making use of it. Are you sure?"
diff --git a/app/workers/pipeline_schedule_worker.rb b/app/workers/pipeline_schedule_worker.rb
new file mode 100644
index 0000000000000..a449a765f7b26
--- /dev/null
+++ b/app/workers/pipeline_schedule_worker.rb
@@ -0,0 +1,19 @@
+class PipelineScheduleWorker
+  include Sidekiq::Worker
+  include CronjobQueue
+
+  def perform
+    Ci::PipelineSchedule.active.where("next_run_at < ?", Time.now).find_each do |schedule|
+      begin
+        Ci::CreatePipelineService.new(schedule.project,
+                                      schedule.owner,
+                                      ref: schedule.ref)
+          .execute(save_on_errors: false, schedule: schedule)
+      rescue => e
+        Rails.logger.error "#{schedule.id}: Failed to create a scheduled pipeline: #{e.message}"
+      ensure
+        schedule.schedule_next_run!
+      end
+    end
+  end
+end
diff --git a/app/workers/trigger_schedule_worker.rb b/app/workers/trigger_schedule_worker.rb
deleted file mode 100644
index 9c1baf7e6c5fe..0000000000000
--- a/app/workers/trigger_schedule_worker.rb
+++ /dev/null
@@ -1,18 +0,0 @@
-class TriggerScheduleWorker
-  include Sidekiq::Worker
-  include CronjobQueue
-
-  def perform
-    Ci::TriggerSchedule.active.where("next_run_at < ?", Time.now).find_each do |trigger_schedule|
-      begin
-        Ci::CreateTriggerRequestService.new.execute(trigger_schedule.project,
-                                                    trigger_schedule.trigger,
-                                                    trigger_schedule.ref)
-      rescue => e
-        Rails.logger.error "#{trigger_schedule.id}: Failed to trigger_schedule job: #{e.message}"
-      ensure
-        trigger_schedule.schedule_next_run!
-      end
-    end
-  end
-end
diff --git a/changelogs/unreleased/zj-better-view-pipeline-schedule.yml b/changelogs/unreleased/zj-better-view-pipeline-schedule.yml
new file mode 100644
index 0000000000000..6d6fa0784f22e
--- /dev/null
+++ b/changelogs/unreleased/zj-better-view-pipeline-schedule.yml
@@ -0,0 +1,4 @@
+---
+title: Pipeline schedules got a new and improved UI
+merge_request: 10853
+author:
diff --git a/config/gitlab.yml.example b/config/gitlab.yml.example
index fa503f84dd093..14d99c243fc49 100644
--- a/config/gitlab.yml.example
+++ b/config/gitlab.yml.example
@@ -181,7 +181,7 @@ production: &base
     stuck_ci_jobs_worker:
       cron: "0 * * * *"
     # Execute scheduled triggers
-    trigger_schedule_worker:
+    pipeline_schedule_worker:
       cron: "0 */12 * * *"
     # Remove expired build artifacts
     expire_build_artifacts_worker:
diff --git a/config/initializers/1_settings.rb b/config/initializers/1_settings.rb
index 6a6fbe86df9ab..6097ae6534eec 100644
--- a/config/initializers/1_settings.rb
+++ b/config/initializers/1_settings.rb
@@ -323,9 +323,9 @@ def cron_random_weekly_time
 Settings.cron_jobs['stuck_ci_jobs_worker'] ||= Settingslogic.new({})
 Settings.cron_jobs['stuck_ci_jobs_worker']['cron'] ||= '0 * * * *'
 Settings.cron_jobs['stuck_ci_jobs_worker']['job_class'] = 'StuckCiJobsWorker'
-Settings.cron_jobs['trigger_schedule_worker'] ||= Settingslogic.new({})
-Settings.cron_jobs['trigger_schedule_worker']['cron'] ||= '0 */12 * * *'
-Settings.cron_jobs['trigger_schedule_worker']['job_class'] = 'TriggerScheduleWorker'
+Settings.cron_jobs['pipeline_schedule_worker'] ||= Settingslogic.new({})
+Settings.cron_jobs['pipeline_schedule_worker']['cron'] ||= '0 */12 * * *'
+Settings.cron_jobs['pipeline_schedule_worker']['job_class'] = 'PipelineScheduleWorker'
 Settings.cron_jobs['expire_build_artifacts_worker'] ||= Settingslogic.new({})
 Settings.cron_jobs['expire_build_artifacts_worker']['cron'] ||= '50 * * * *'
 Settings.cron_jobs['expire_build_artifacts_worker']['job_class'] = 'ExpireBuildArtifactsWorker'
diff --git a/config/routes/project.rb b/config/routes/project.rb
index 68474a443682a..7f6e5447b1982 100644
--- a/config/routes/project.rb
+++ b/config/routes/project.rb
@@ -128,6 +128,12 @@
         end
       end
 
+      resources :pipeline_schedules, except: [:show] do
+        member do
+          post :take_ownership
+        end
+      end
+
       resources :environments, except: [:destroy] do
         member do
           post :stop
diff --git a/config/webpack.config.js b/config/webpack.config.js
index a3dae6b2e131c..cb6bd949ddbb3 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -54,6 +54,8 @@ var config = {
     protected_branches:   './protected_branches/protected_branches_bundle.js',
     protected_tags:       './protected_tags',
     sidebar:              './sidebar/sidebar_bundle.js',
+    schedule_form:        './pipeline_schedules/pipeline_schedule_form_bundle.js',
+    schedules_index:      './pipeline_schedules/pipeline_schedules_index_bundle.js',
     snippet:              './snippet/snippet_bundle.js',
     sketch_viewer:        './blob/sketch_viewer.js',
     stl_viewer:           './blob/stl_viewer.js',
diff --git a/db/migrate/20170425112128_create_pipeline_schedules_table.rb b/db/migrate/20170425112128_create_pipeline_schedules_table.rb
new file mode 100644
index 0000000000000..3612a796ae869
--- /dev/null
+++ b/db/migrate/20170425112128_create_pipeline_schedules_table.rb
@@ -0,0 +1,28 @@
+class CreatePipelineSchedulesTable < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  def up
+    create_table :ci_pipeline_schedules do |t|
+      t.string :description
+      t.string :ref
+      t.string :cron
+      t.string :cron_timezone
+      t.datetime :next_run_at
+      t.integer :project_id
+      t.integer :owner_id
+      t.boolean :active, default: true
+      t.datetime :deleted_at
+
+      t.timestamps
+    end
+
+    add_index(:ci_pipeline_schedules, :project_id)
+    add_index(:ci_pipeline_schedules, [:next_run_at, :active])
+  end
+
+  def down
+    drop_table :ci_pipeline_schedules
+  end
+end
diff --git a/db/migrate/20170425112628_remove_foreigh_key_ci_trigger_schedules.rb b/db/migrate/20170425112628_remove_foreigh_key_ci_trigger_schedules.rb
new file mode 100644
index 0000000000000..6116ca59ee4d1
--- /dev/null
+++ b/db/migrate/20170425112628_remove_foreigh_key_ci_trigger_schedules.rb
@@ -0,0 +1,13 @@
+class RemoveForeighKeyCiTriggerSchedules < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  def up
+    remove_foreign_key :ci_trigger_schedules, column: :trigger_id
+  end
+
+  def down
+    # no op, the foreign key should not have been here
+  end
+end
diff --git a/db/migrate/20170425114731_add_pipeline_schedule_id_to_pipelines.rb b/db/migrate/20170425114731_add_pipeline_schedule_id_to_pipelines.rb
new file mode 100644
index 0000000000000..ddb27d4dc8163
--- /dev/null
+++ b/db/migrate/20170425114731_add_pipeline_schedule_id_to_pipelines.rb
@@ -0,0 +1,9 @@
+class AddPipelineScheduleIdToPipelines < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  def change
+    add_column :ci_pipelines, :pipeline_schedule_id, :integer
+  end
+end
diff --git a/db/migrate/20170506085040_add_index_to_pipeline_pipeline_schedule_id.rb b/db/migrate/20170506085040_add_index_to_pipeline_pipeline_schedule_id.rb
new file mode 100644
index 0000000000000..08a7f3fc9abb7
--- /dev/null
+++ b/db/migrate/20170506085040_add_index_to_pipeline_pipeline_schedule_id.rb
@@ -0,0 +1,19 @@
+class AddIndexToPipelinePipelineScheduleId < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  disable_ddl_transaction!
+
+  def up
+    unless index_exists?(:ci_pipelines, :pipeline_schedule_id)
+      add_concurrent_index(:ci_pipelines, :pipeline_schedule_id)
+    end
+  end
+
+  def down
+    if index_exists?(:ci_pipelines, :pipeline_schedule_id)
+      remove_concurrent_index(:ci_pipelines, :pipeline_schedule_id)
+    end
+  end
+end
diff --git a/db/migrate/20170506091344_add_foreign_key_to_pipeline_schedules.rb b/db/migrate/20170506091344_add_foreign_key_to_pipeline_schedules.rb
new file mode 100644
index 0000000000000..7f2dba702af58
--- /dev/null
+++ b/db/migrate/20170506091344_add_foreign_key_to_pipeline_schedules.rb
@@ -0,0 +1,15 @@
+class AddForeignKeyToPipelineSchedules < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  disable_ddl_transaction!
+
+  def up
+    add_concurrent_foreign_key :ci_pipeline_schedules, :projects, column: :project_id
+  end
+
+  def down
+    remove_foreign_key :ci_pipeline_schedules, :projects
+  end
+end
diff --git a/db/migrate/20170506185517_add_foreign_key_pipeline_schedules_and_pipelines.rb b/db/migrate/20170506185517_add_foreign_key_pipeline_schedules_and_pipelines.rb
new file mode 100644
index 0000000000000..55bf40ba24db6
--- /dev/null
+++ b/db/migrate/20170506185517_add_foreign_key_pipeline_schedules_and_pipelines.rb
@@ -0,0 +1,23 @@
+class AddForeignKeyPipelineSchedulesAndPipelines < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  disable_ddl_transaction!
+
+  def up
+    on_delete =
+      if Gitlab::Database.mysql?
+        :nullify
+      else
+        'SET NULL'
+      end
+
+    add_concurrent_foreign_key  :ci_pipelines, :ci_pipeline_schedules,
+      column: :pipeline_schedule_id, on_delete: on_delete
+  end
+
+  def down
+    remove_foreign_key :ci_pipelines, column: :pipeline_schedule_id
+  end
+end
diff --git a/db/post_migrate/20170425121605_migrate_trigger_schedules_to_pipeline_schedules.rb b/db/post_migrate/20170425121605_migrate_trigger_schedules_to_pipeline_schedules.rb
new file mode 100644
index 0000000000000..a44b399c4de65
--- /dev/null
+++ b/db/post_migrate/20170425121605_migrate_trigger_schedules_to_pipeline_schedules.rb
@@ -0,0 +1,41 @@
+class MigrateTriggerSchedulesToPipelineSchedules < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  def up
+    connection.execute <<-SQL
+      INSERT INTO ci_pipeline_schedules (
+        project_id,
+        created_at,
+        updated_at,
+        deleted_at,
+        cron,
+        cron_timezone,
+        next_run_at,
+        ref,
+        active,
+        owner_id,
+        description
+      )
+      SELECT
+        ci_trigger_schedules.project_id,
+        ci_trigger_schedules.created_at,
+        ci_trigger_schedules.updated_at,
+        ci_trigger_schedules.deleted_at,
+        ci_trigger_schedules.cron,
+        ci_trigger_schedules.cron_timezone,
+        ci_trigger_schedules.next_run_at,
+        ci_trigger_schedules.ref,
+        ci_trigger_schedules.active,
+        ci_triggers.owner_id,
+        ci_triggers.description
+      FROM ci_trigger_schedules
+      INNER JOIN ci_triggers ON ci_trigger_schedules.trigger_id=ci_triggers.id;
+    SQL
+  end
+
+  def down
+    # no op as the data has been removed
+  end
+end
diff --git a/db/post_migrate/20170425130047_drop_ci_trigger_schedules_table.rb b/db/post_migrate/20170425130047_drop_ci_trigger_schedules_table.rb
new file mode 100644
index 0000000000000..24750c58ef008
--- /dev/null
+++ b/db/post_migrate/20170425130047_drop_ci_trigger_schedules_table.rb
@@ -0,0 +1,32 @@
+class DropCiTriggerSchedulesTable < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  disable_ddl_transaction!
+
+  def up
+    drop_table :ci_trigger_schedules
+  end
+
+  def down
+    create_table "ci_trigger_schedules", force: :cascade do |t|
+      t.integer "project_id"
+      t.integer "trigger_id", null: false
+      t.datetime "deleted_at"
+      t.datetime "created_at"
+      t.datetime "updated_at"
+      t.string "cron"
+      t.string "cron_timezone"
+      t.datetime "next_run_at"
+      t.string "ref"
+      t.boolean "active"
+    end
+
+    add_index "ci_trigger_schedules", %w(active next_run_at), name: "index_ci_trigger_schedules_on_active_and_next_run_at", using: :btree
+    add_index "ci_trigger_schedules", ["project_id"], name: "index_ci_trigger_schedules_on_project_id", using: :btree
+    add_index "ci_trigger_schedules", ["next_run_at"], name: "index_ci_trigger_schedules_on_next_run_at"
+
+    add_concurrent_foreign_key "ci_trigger_schedules", "ci_triggers", column: :trigger_id, on_delete: :cascade
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 61af066ed3bea..722e776c27df8 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170504102911) do
+ActiveRecord::Schema.define(version: 20170506185517) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -95,6 +95,7 @@
     t.string "enabled_git_access_protocol"
     t.boolean "domain_blacklist_enabled", default: false
     t.text "domain_blacklist"
+    t.boolean "usage_ping_enabled", default: true, null: false
     t.boolean "koding_enabled"
     t.string "koding_url"
     t.text "sign_in_text_html"
@@ -113,14 +114,13 @@
     t.string "plantuml_url"
     t.boolean "plantuml_enabled"
     t.integer "terminal_max_session_time", default: 0, null: false
-    t.string "default_artifacts_expire_in", default: "0", null: false
     t.integer "unique_ips_limit_per_user"
     t.integer "unique_ips_limit_time_window"
     t.boolean "unique_ips_limit_enabled", default: false, null: false
+    t.string "default_artifacts_expire_in", default: "0", null: false
+    t.string "uuid"
     t.decimal "polling_interval_multiplier", default: 1.0, null: false
     t.integer "cached_markdown_version"
-    t.boolean "usage_ping_enabled", default: true, null: false
-    t.string "uuid"
     t.boolean "clientside_sentry_enabled", default: false, null: false
     t.string "clientside_sentry_dsn"
   end
@@ -246,6 +246,23 @@
   add_index "ci_builds", ["updated_at"], name: "index_ci_builds_on_updated_at", using: :btree
   add_index "ci_builds", ["user_id"], name: "index_ci_builds_on_user_id", using: :btree
 
+  create_table "ci_pipeline_schedules", force: :cascade do |t|
+    t.string "description"
+    t.string "ref"
+    t.string "cron"
+    t.string "cron_timezone"
+    t.datetime "next_run_at"
+    t.integer "project_id"
+    t.integer "owner_id"
+    t.boolean "active", default: true
+    t.datetime "deleted_at"
+    t.datetime "created_at"
+    t.datetime "updated_at"
+  end
+
+  add_index "ci_pipeline_schedules", ["next_run_at", "active"], name: "index_ci_pipeline_schedules_on_next_run_at_and_active", using: :btree
+  add_index "ci_pipeline_schedules", ["project_id"], name: "index_ci_pipeline_schedules_on_project_id", using: :btree
+
   create_table "ci_pipelines", force: :cascade do |t|
     t.string "ref"
     t.string "sha"
@@ -263,8 +280,10 @@
     t.integer "user_id"
     t.integer "lock_version"
     t.integer "auto_canceled_by_id"
+    t.integer "pipeline_schedule_id"
   end
 
+  add_index "ci_pipelines", ["pipeline_schedule_id"], name: "index_ci_pipelines_on_pipeline_schedule_id", using: :btree
   add_index "ci_pipelines", ["project_id", "ref", "status"], name: "index_ci_pipelines_on_project_id_and_ref_and_status", using: :btree
   add_index "ci_pipelines", ["project_id", "sha"], name: "index_ci_pipelines_on_project_id_and_sha", using: :btree
   add_index "ci_pipelines", ["project_id"], name: "index_ci_pipelines_on_project_id", using: :btree
@@ -313,23 +332,6 @@
 
   add_index "ci_trigger_requests", ["commit_id"], name: "index_ci_trigger_requests_on_commit_id", using: :btree
 
-  create_table "ci_trigger_schedules", force: :cascade do |t|
-    t.integer "project_id"
-    t.integer "trigger_id", null: false
-    t.datetime "deleted_at"
-    t.datetime "created_at"
-    t.datetime "updated_at"
-    t.string "cron"
-    t.string "cron_timezone"
-    t.datetime "next_run_at"
-    t.string "ref"
-    t.boolean "active"
-  end
-
-  add_index "ci_trigger_schedules", ["active", "next_run_at"], name: "index_ci_trigger_schedules_on_active_and_next_run_at", using: :btree
-  add_index "ci_trigger_schedules", ["next_run_at"], name: "index_ci_trigger_schedules_on_next_run_at", using: :btree
-  add_index "ci_trigger_schedules", ["project_id"], name: "index_ci_trigger_schedules_on_project_id", using: :btree
-
   create_table "ci_triggers", force: :cascade do |t|
     t.string "token"
     t.datetime "deleted_at"
@@ -981,8 +983,8 @@
     t.boolean "lfs_enabled"
     t.text "description_html"
     t.boolean "only_allow_merge_if_all_discussions_are_resolved"
-    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.boolean "printing_merge_request_link_enabled", default: true, null: false
+    t.integer "auto_cancel_pending_pipelines", default: 0, null: false
     t.string "import_jid"
     t.integer "cached_markdown_version"
     t.datetime "last_repository_updated_at"
@@ -1349,11 +1351,11 @@
     t.string "incoming_email_token"
     t.string "organization"
     t.boolean "authorized_projects_populated"
+    t.boolean "require_two_factor_authentication_from_group", default: false, null: false
+    t.integer "two_factor_grace_period", default: 48, null: false
     t.boolean "ghost"
     t.date "last_activity_on"
     t.boolean "notified_of_own_activity"
-    t.boolean "require_two_factor_authentication_from_group", default: false, null: false
-    t.integer "two_factor_grace_period", default: 48, null: false
     t.string "preferred_language"
   end
 
@@ -1408,13 +1410,14 @@
   add_foreign_key "boards", "projects"
   add_foreign_key "chat_teams", "namespaces", on_delete: :cascade
   add_foreign_key "ci_builds", "ci_pipelines", column: "auto_canceled_by_id", name: "fk_a2141b1522", on_delete: :nullify
+  add_foreign_key "ci_pipeline_schedules", "projects", name: "fk_8ead60fcc4", on_delete: :cascade
+  add_foreign_key "ci_pipelines", "ci_pipeline_schedules", column: "pipeline_schedule_id", name: "fk_3d34ab2e06", on_delete: :nullify
   add_foreign_key "ci_pipelines", "ci_pipelines", column: "auto_canceled_by_id", name: "fk_262d4c2d19", on_delete: :nullify
   add_foreign_key "ci_trigger_requests", "ci_triggers", column: "trigger_id", name: "fk_b8ec8b7245", on_delete: :cascade
-  add_foreign_key "ci_trigger_schedules", "ci_triggers", column: "trigger_id", name: "fk_90a406cc94", on_delete: :cascade
   add_foreign_key "ci_triggers", "users", column: "owner_id", name: "fk_e8e10d1964", on_delete: :cascade
+  add_foreign_key "container_repositories", "projects"
   add_foreign_key "issue_assignees", "issues", on_delete: :cascade
   add_foreign_key "issue_assignees", "users", on_delete: :cascade
-  add_foreign_key "container_repositories", "projects"
   add_foreign_key "issue_metrics", "issues", on_delete: :cascade
   add_foreign_key "label_priorities", "labels", on_delete: :cascade
   add_foreign_key "label_priorities", "projects", on_delete: :cascade
diff --git a/doc/ci/img/pipeline_schedules_list.png b/doc/ci/img/pipeline_schedules_list.png
new file mode 100644
index 0000000000000000000000000000000000000000..9388fac98ebcfbb6d57e81f8620d9759b2900eaf
GIT binary patch
literal 67555
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%0DDYo>SPC6#a
z7&I5ck^Xa8PJDL=1uv7XVrV)$eETW<yAEG&DKWWDtYABt?IE95`w#$A3QC0{3wSs8
zqh?k;y6wsBv7=@Mm!+w#s&a)UyKJ`n2POwoAF8KzidN(m8QQn}8g+U!3oGZL1m9dl
zNk;YH_S}>?S`)C&0~Qp^dkyA0utx3^G-&XE;3j1iN$%)vT4k|yb5qIo`)8N!b*uKe
zy|uwX`2C5@$}h-R<0>XAUuvYQ8s#BXhb$*ctfqpAu>IS9Tv5V27PQ#)=?)`556iRl
z)Q9!l7U$lXoafH2`EPwy6hdu0BvL0zSX%Zdu{|4>b-Gp-y;mvLh12!0ibXPOXl+Y9
zQg3Fl{d34)PjA;DF7-O1J)uohtlz#-M(dfDf3vWZQud*>m=wKGWrB~)@Nm-fVRUI$
z3rs6@*EHO~f{N}=b(|6m)qO<~>t72hFxkB>AQ0J?<S)gwW;&s>smPeZ^53OIbcKQ(
zzwi+L%!1}5MylqFJ03IE+GE70%g}~8tHEs{p!b;JFo2F5enoUpYIz0L0gesER==C_
z`;$MD=DGFKvbk8&F42ueK&62GK<%Y44NI<y9kVdAu>#lC$|nhx1g4qj*T^Ch$GMZO
zrK|Mr4E^6{^z5VZbGte*!*BC|^L5^zfofVrtZ9)wCzvpXrS9M_;JT}+P7f$-*2v*`
zeSdDA=a(`|)uE^D3_}!c^-(H{_bQTSAC>2Z--RNT#}PC<r{(xCvXvAHm52%(mGk|B
zeQ|%-8o@DyBAbFgIE$)uyKb?1sm46_szkZ*V$OMKz+aXn!4X_tng@~c;a^ORpS!#-
zlutX8u<^ezA9Fduv4IjF%B<q|P%*^&Bi*qBkDfB`{+G)ZYs98fU0P>wbqfP@kleo@
zqe<pJj%|{6$F=`D{@=}&6LN9uN6^0^ge1CV5EM4<>o7px|743K1pmlP*#5}^kwT(T
z2>Pu<BPVD8{s9V+{^gsX{|&<Q|6{%>t80S$-?3c8624HK@YuAqBL4>T`t+ay|8J7o
z1^s8D?z@CRiT_S$d$l8}^zSl`als4~F`mxjGSz$4ck+KG`0u6Q59&Xl5S%V7!apD}
zi7w2)KKqnF8UOT&&--V>P&$x4{_mRkYtoncFDMq0Kg|12Q0%MgA0CXr6N?n$pFEiS
z-|E1WPH_8Yb(miKf!sJAXaynuSs{N02h~ZPNPGRyY))?ct?mmIUx<GO@Mmza-tzP+
zd;dh>vbr&|y4E*JAt3*mFjv5zxHHsCMlt{Ia`{&({W`nVAff&_6Bi<XIk;-&6zYG(
z?FaXF{r;$gg865y{Q36%2ktW{X8Dtw|640484$4l-zM&>HWaKe>*(G4KK-tqx*Ty=
z?S!{)UG1*5?vrQ;-g^1E|EL2k0y1iuZg3zXUJ~(|%Fc|voI18LpAcRv`_>@6!yKPB
zW0ITki}TipQIl~s)&XSBJ1=bhbx)|hSaWx$L+zeblM-N)TVd7hh~Ywd>ORX3ow0_&
z=k+tGWj!|9=&-(h!UbOw9P92wO$K-zbarOCFfMq!L%gNV&&%|D8g34A4X0N_J6L~{
z{S_A~2rllsas1yW(g~uv%)>)MphSIV;drmO-cmp}LhRa9Y$hGYA5<(4Xl9Y!N<TE{
zcS@jt$b(CirV-t``j=j-$Jx_F(8+&8!b83r^M)-w&+u1+$1jbBL-u~RMs5ILwEMr7
zZoIK@4n3ogF!20XEctCUun`EGt7&-P1lhpD7%95&>ckJ^T}LRjfokD!+N{%5#W+*j
zFU7LXJ|rZTrWL+vraB-|OsG?%6Jm3=<Z4E%d~vAnwJjKR#~Q$A5Ay%c5^RhlQr(Mt
zbS^ss#G|)d!+~zmZf1{5se=mxQ>ggj1`g`tl6$4L=(t$5k{*j&j<vz{!Ud@YXD0ne
z8B`$5UQ{2^nlAlaU|lLQT_af`7v#ilsL(tQ-GJIiw*n*2HdnSiAM9W?@q1sJ!EV>v
zx$gQT#8SD0waX`ii_*#*>~EFrjzD3FwN1f*Z2YKCF&a=M+iw`t%1%sf4+mE0cOdOT
z4__Q&1RH<h?_|a%33^9Ignf8v**;+d)<3SRjn|zWkF%vv@q4i|<P%oH$iI<~<wxDB
ztJoxsr6)Ctfckph*U1rLWD>?c5^QKi(C*N;6Xk+74vnjCnTZDCSk#lhW({ItK8$_-
z^9fYI-~c8(<P6g1Go6DoKL6{!5`F3_#>5yNluv(K<dSoS8C`=X4g*wYZS20+e=H&K
zKfK-45sms9`Jb1GUumW<#@CnB$L>iH#CL$D<6+qIAYs=Fn+7Le3A|=1>qWN@i|zh)
zZynxycmP`Tls*lB{<+GupZ(IRZ{tfYGkuBS;){<WYRBpgEZ9n|1zPMwv;4`kqay-2
zxnd%6T{|YZI}D`B%tiN7ZW9+3?mFYs$Du9qoOOg1w?b2b2hH|Y#OIrAl4|`|B#JcU
zSw&m{<Aa1WA*PS0x{I6;iU5l6CKWivlYpwWm+glyc9^P`)AnJCXsaQR`aQxau~0@Y
zO~?fNiwFtn!QQ>hH+N*r;QpPxN2n8U-nf{nmlTd;b%9-1($7mD4{6)0MnG0zX5JPQ
z>gn$uf19u>No!G#OW~V{H`fg?J8i$}41(Lwnrgjrjjy=A?2v28?mixlHmiN+HUno)
zon&Vlu)pe-1f1nRy2F=VBn%C9+;X|z7aCOC1hhUq<g{gFMwO4qs7G6L`44`Z6oS=q
zM6wGT*URi=0zNxbVN2zcOzEk&OTn=lef*{ZKIVQfI9fn8wVD&5;f;xi;(sQ}%xnt&
zghzwB3?_&hgw2)qefEo#vQs$}gLt8A&k<1sJ6pa<$r+CLu#+;oLtHg5b_ac}j(+0Z
z>_9)50+UMXW9#c~(C~6rmFeuhG(ldY!EElrKjm<GAtx5VZeGUl=~V^ou6Huob1?N|
zf2a7fv_6V473;QQR_}bDi@a@McU;i>y%1^4f{{p0ZVKT)w!YQNbo0bz(36f^wp4=a
zFpOO`^r3I;w5HmqAG_JG?}3^`2(4F<{(VqM)`2Exoo0p49zt^|bWP>iY3?nJhPYBc
z*Brruf1rmAGWc!z$HCly?+ADg#5G;<j!=40?aD0HM2leXvK+MiMlDA6SnkB>XkL{k
z*_mm=A9;?!KX6n@IW>R~PQ(W1#j}K82}@<RoTh3y$%)sS`;k?9xS@S*>hqnMpBMVC
zX9iWz^w8>e9%KeI`oUOPq%I_S?Ix(~sTt-RwaML)&J2dP*uQV?b_%`wBNe8Z36Xzo
z8c#7Mjnn0G)OP^VQ)YRoky<9CqAIt_v<;q33IVTB;XViY;hk&{%V7GRD~ekHtyn-9
z3eyCR{<?29CYAn6diTQB5WrKfc3wxrhq4X2_;(c}R^uluQZ_5(#Na9$!gpq>RgMpC
z9wNaSm$_$aItwOf8}<EkdWY=9-!p#sE0)JrZI!D<J5weOCGF_-%Jf6xK8p{HB$G)j
z-H;he0i{eF(!#*EICNE80qB@iGF?^P#p?HOo|asBmD}uv!OT=W-Va`AhQpX~8=WTP
z*c~<5iSvV6VZwQGI+w2iZ2EVs&*I}*RR>TlD^qx-nJE1_`Yp`frmyHL2{b(&@fwaZ
zVhipaFtv^8{l{tYiwTy9#Oh`PojY=&M$Q)2Uj`UsonBMJJU!o>*{R+gh2Ws^H@VRp
z5?-O-+4a{`kW<6r=csruzI|y3UH>#9+LjR?wTP9twZ9vxkU@RCVa3cqxsc?7p7x7D
zQ{bHdsBH_!M60hmy((m&>6HgDh0Jp6i1iw@7&hjr6YzTM5Ap?U08~w|HY;E?Xl9jE
zI0UfqzWPxVuXKi%<lz?78I`HG!K4rkZg2h^ARmqoKm*|k%7s8$h5Jdo-SBX!MJ9aN
z_Qm(Lz=5G;*~=UcI8^8`Hbl0#^+^Pjk06A1R#uJ*kG$a>8=LpMZG6GJxF8vXng(N!
z1Ah=k6q_FpO*<!T^cv=bT>Pf^aGdfJg=Bm%irVSm5tb^?yInZI3z>?S;<m96k#6l*
z2#)B&x$8QGOf+|9iwP0`ONY7FkCD6c-KdK<Ll9*6@9|f6a^oI|I<=YOq}s6&`@C(8
z_u%Z9{GJX}bvR3Z_>DQPN^|IsOMG(;9OYe%BZ8k>5p(+EL+8SPLIYf@KOega)&ES<
z<~U;V)dNhxx8b|7DE%74POCAcvWI4rfk-^GZ8kWVK(vno!#<yK4EyAH%&L=E-`%Gl
zrT=bwZex9|q8orz86i}EpY^F96^OEwcIWqSMO|q)Q-WkOe_H}O5ufnAY38;=S);!(
zTSbmOlx|K}Db(G>LHe?-1NH~LTu3hlTCvP<fC@%Q*%kSFr`DKszrs0s9Iw>ZOh(AK
zs%Y!;ZCxgPY0AaWqOg=Mco+i*YIEqrW79y2e?_uBgw1v3;AciAVs+Ti^>$ytw_7Ea
zEA%Y1bA4zQAX848MeZN5dS!N)Ldr`&3M4M4@J?9C*#2_t2TX;S${nPK+3O6((wbpz
zl4jb;6!?rPM%E%<MAa9O@<{lU@%yJ0ahs~cWF8xF6F*w14{dh=dEk=3MOs<!eUG7w
z=KMvN0^xnHD8rHqH5_?M90^jCQT9ZQXWy$Us{9(T!PLiI8RG36_46s~Vsgu5xNJ@8
zOY#Q{$BChS)@BS5U$i^9cGa}z%%@v@5t~-uCZfV>J(`OtA->4@Gpn#xz>8vXMqu$J
zOX)djZ@wrUmPV<#T*kqmv(#GO`{XgSkvx<Na#+>vdWx$|z?04NwOx{KwlP}q`NRvC
z$k!Us>w9Dzr|JoXAiP=eTe&j*uSZKan}O<Nw){%=<4XE)7yd~2Tpn|4;rB*dNx(&B
zVocnHHT2FJ2H`NVA1NE1CoLwy?)U>xY^<Cy=E!Qk0Bl3}#97dHp#NFuvX`%`>Sso8
zG;_H>;3hB;aKY4lS4W0B^Q{wpn`R0ZH*2nQQjQ^}*@wTzdjzK&Y#xPRc1&K^u$J4q
zEwR>#knI!C+)DRsE5q3UOHz;?AbMB-+hyMSHon*HIf;F>oe9vZW~f_edsAhOr#=qs
zV7KA&(oTywKefb*e7%WT+PP81D<U8+*`>XS8I1YjfG+pW=w-zU{ldKx{@I5BDK{Y(
ze`AHk0a2!qp1GmkpE<H?<U5?jNj=Y4&s3{81$zdPXeBXUomS$>NRp35iR<6ffuFl%
z7Gvh#Ed~OnEEL70EdtG^UHa&7Ib9NH(bqirO<3csp+$i;mgDC^PsiN!f)nXQKO`RK
zM?BT9P)63sTI^PW(xqPAk~&1{i^~CMr3pWNpp)f4Jqzmnw6%*XKzxDY&?iVO!If2~
z5LVt9>;1S$Nw~Q=U{8C43Mh-y>etlt==B!hUw6(15F?U@@qa5($M0QU9>d0;>}Va;
z_s&7HeW<r&s+iZ`tChs%{fd;msq=BIdOBLNJiqg~h;d<+oM8Qh=IB*Cv|1D9=2J?y
zn#F^OB<-Mc3xZWndokKmSXE`Xe4!qpCr~AH;HomRCFHDB>}y{gLf0e&`a^#%#yE1l
z6e+b^psTS^5W9!BlJQ=`Vm!ke!y1?~Rj&x)dH5lqzFP+}WkFLvQMa~iOFqUa)q6(X
zScVUPQHz^E1!Ae`XOY^9L&s%fp7K%U4r;hgj6m|TlXhH28bNdvsK4d8%N}IY*Qi9V
z>Ccdl*71UH|GAXlBAu)1GYY%aVudguJ)?g49UHC=O>{i_oI;ZP<J#~eovgBpp;-;X
z64fuU0Ou;iGlMyLT=UK$=s&dWtE3&ShZ8*}XFJs7w-cAwiU65i`Dy(VKT<0_dB(H{
zfC8#e`AAUSl4I;sn~rs;-7<%1#ji%6{DdsLW1)22adw;N7qbCi)%-c<O9Volp#{j$
z!PJx(rgQYGXX{*mwgW%Xcu6-OPwv9~9Q@PCStvLm!|k?I>dAg@ou7!8`fuetiFG~B
zRCnwK98&(hG{;b=Jo_OEYIBL^pqI*BT2?bo7CErJRnoExDve!6aQqO9qp77CzCBf)
zRN0Y|aGEh=DEKHl^1+=3L{uME_4@7dXk>zvff5DY`;(LR;)XOZ?efCmds*#?9C<l3
zvKuAG@EXD$*$7n#Rz&fFoV#Ko{>Xa$>YP>(O7DogIN)^O=;Ya&sKE-uBjwiPtnx?c
z?-~Qkii-RT(ZF>98RNNL-`2jvvNqO-m+ke7QAG4eAFiL(_PVTCwVR;0hqAkzlJ$Yr
z7MEhZ_GLD)7zB3Nr6G5>9|*5wMw5a)V3VU;*?UW}76H{UuY19e#`+{|n{(8N4RUEX
zs8q=V*(JU1PaD@T8olq=u$J74Wv}Fyybb*8cw70X;j_DpL?b8}i{d%vL784!an~$2
zaE{Gx2p1h6f>2PzmQs~P)?7?dY^_P6e_22o<nzKPo9@H8E)Nya219h#;HqrWJYV5+
zsWL!WcxqayLVM&F2)`zhA<ujLXuZ?LukyC%T5j`^wvrSQi`4i^XxW<rI;{TmTZs_;
zB>gu1j)DaP$Ncy*;x$Nu2jTx=?=6GlNR~!zF<BN`Y%w#F#mrzaGo!_<kt}A}VwNn6
z$zleJnVFfHX+~E*`<}CVc75;9FTS{OBPu33rlxu-v%0FY^Qp`h{uE32^|pjhVNf=K
z<GLmNfl&r`?3Y_2dlmHuAa$m>oAu|fr#nS-h(mI=LXpD4L^bAj<V!yEN10-(GZc8+
zuMN>)RLl=g@}QhqJME9PDf%@Kh~hMwVzuP4Q3__5PCM(;S%*^X1Oo4HvVk$&Nz^oh
z5wWhrk&1FX4Y*$#ooEd0b&?{-1tNbSQS@UWy@M$+3al^X4n6>L41d&0;s_mgVNq6x
zuS3>uYqztPCqT0eJ-<L{R51g}=QgCEwwf%~&UF}4^)FM%KyY-J5JY~`B-KgFXHr$U
zQV+jgseyMDnEA-K<cHb7u>X$eXc7EoO%m(&GCbB%+$>AGzpTZk1yqNF!&1_`j>_I{
z&0cJz`X2VjLXojGp={e)Q{(CbtK>r?3V*8S6}L~Y{^q>fCdA`VHrQx7Lq=3RvT)vk
zN#-%@xXN-r7Bmu#j8{={i}8t-U`%4NcZqa^1)fPpcTOC+MK1pZMF1Xke#E|yHq&X7
zJ{rRiwO*DRrbAT&=zG6IT@;WlM*X~x-wVXzMc^MaU%P~?O(uuzRy_stDNbCMDgnq!
z=G1F&G+av#`}7?N&$IEuJ8af&$sVI?q}sAS^l4~uw-o|)LAi#8fBAEodDKKA@50wo
z*E7J5z!DS4QnSxq)vj_um46TfmqLMkmKjV9S!PNhL#L+Vs__tA3`W5&Taq@E3ef(}
z`!7qLk4({WxS3$rU+Q1>jSQuMDtehG7~aL8#9C*3MnGgMo@rDoT#*Q@1!D(&GcpP>
z8OGJxDphA7PC*P|2Q1@gH*E!((00BI$06UG?)-xU$42c7CpsR0Ek`i$B{EA{?592q
z8-$F@TLx~BG=W&J>{9z%8ICt0$^}6pBpji+xAZp+HkP{g6>s~wGD-0YfuTG0<+En{
zU8YGcBUJwGPfFggRFIGbj-L#U9Bp-P434mnS!UaV)y){XWBNv@nD#<<JNYkA{1dIj
zn-ZfGlJc=jI(hX2K@|kT%T2P~^rIcR0@OnKX3t#}sNnpH(~mPVV!S_?AYUHusqz_^
zNaZo2_g#|83)JP)u?aO9`BfJN^q`E__`}EA<6W+=e7okyhVPF<Q%{k*As4wi#ANln
zu`3vPA}bdOz;3}g#Qj{F{RSdp)^7lWoi0@|g5i#vCUEUCFZufnvkprpS8b*#_ueap
z*AI7^^s9(rbxf4V#_D^9^WS8>YZe0K^q-^cYpKNEbVwzqx<vUZ<ssz}i?AuhtA*#M
zd^MU(>{X%G2tegrL$G;p&aJ}|m2^22EBO?eKn{+#sug)QDc8@P=^s^vV_t00jd*(M
zx!%!)_Dx!<#?8bUJ!z7&kR~7DF7=k=%(n?zAiSegjYDUagYn`H`-9%(m%s<c@}K^t
z@00UZ^2wzDuZA*i;neN!_#}<7<ZNzrz(Mm_O&c-g5m@3&J8}j$t=n4Nu=$)nVmiNh
z<M0Dx(mTChZc4#c5=HZvOHVQXu-E&7MJ&{^i{GPcb_OkKG|6^ey+oTg?qSx4s?fB)
zEs(azO$3EF?|r@0{XL;f<u;+!>POiz!(^6Xn=5LJ)Owe~83WEYI@Am!)sLtr;5&Ly
z@6`>TTnWQ3IMXVPdW3Czl^@=-_#!+mLfN?F)<xP&$lh<kwcKX8Xi%kPQKLH?L3L|z
z3I=SRUw7SKN0Dn>=EFe_UcdLZ+#2L6BtPqV-MBvA^6o03sV%vXIcA6w!Np~RAeW$T
z{hL_T)dOu)9@V=TfRRmOidc!*2y)eH@T7pgU<U%~y6Fn&NI+_nZ3?bBxc2MFl{0+v
zU3Z=P-PU{F+v?-p7pKbI*KkMy-YeKVlCNhPT(iY7ayxC%AoJq;2p-z38QNLSbM!3b
z+~*5{fa{by5IPxsgcwuO2yio6XPg7pn<87$^>z-HF#b^R+5&)sq8RWN3pI&60CJ_p
zL3Yx(hw9)*uX#yR*^=QwtERi5Y1{Fr{&)LvN;|#gYK+JF#EG=opBnfsh{kc@%wBUI
z^!mk<Jw0_ISj``bwWn9lf-z)hp)kWNF9&#QiSzgIRakCqB{YMt6Ae=<)Q#u;7<PI;
zu)Rg+Sxe};6kCIzGI%iIS8Hz{KV(6O193ns`lcL4$9cnQrFimad^76O#jIVf6gw@q
zAAbxSB#y5Gz_Z;yUWCRnLY8X8M}AZo?erU3*d_NBNz-$G0KC-zjv-%cXiMcSM7Fr_
zp0>%Yc~7~r{(xw&N+0O>Nk~C_`SyUc%%>~#saIOFl;H#`qw?+yC5*V2{<1cZeBSNu
zo$LVs=ZROl*PZNNHmqd!M5EK5jXmlO(}OI_FCVQA0|e%C!?oD)5%XI>_*YHAk7cz~
zhAqc(b>719t#HsyMEa}6z3SziiKdW;Ly$1!u^O<LR`0W_!N@W{d#700*!WdcprdGc
zifVi^DucGwK3u12MGILx8eL}rGlwq)0XunF!$*fweMOA#>+7VfF*>OSC0=p*2OppH
zDN(3h8Rk85M4>(Obx}wW`g3*V=BnY2%0|D=t01Z>w$JU=Of(+Tz``APemVRe?Lgs&
z<*XnbIt8%*$pt_|M_6#7hEU_X_M<jwNTr=kqkQo!Z@gw_!?DK0`VnC|)rT1-pmfP?
zrmpN1pTH`j3!}fi>-n7GU$W;xG1R8DNmqB7{PN2RuAovSvbPm&NF$(=+BY)rD7xKx
zQYGCU(b%A-vKzik3YTXxqB42!uxc|zB@!et=pf5Z&t!+ikv%}&N1&W`Pr5dTWx+kx
zMab}mcbqul2%?8|n7)q^qq0Bz5GgG+1ypI$^%O~@fNF@Wna=A=*BQMv3fL_ME-tI=
zxkF#BA6c(FWq&v*-LI?T((!**A4aH<dbKy!ttdKhKqZaN$sDidV@bH*Lxauy92s1%
zTJ3q2m+DSr81d`i{=592!G++v9E*RVmDg{0qkhp9+K;;-zrlxruXSpWfBwgxfA)XM
z7V!TWP4j~gIC>qwf0Wk;H-u*bjlyBH-&DRLPyGML@t1@De;`jp^U-u-=-+7kmC2Hh
z?|;Ka9zw4c`8#2HHAi0|lj`ID<8}X@4ZjBVmDhE=f<=^nULREcFDtx+CCc^L{Yu9<
z+}F=G?9A@QHr*)CSGSfkqc*t_M#U!)uk`otbE4b|LVEs%&It=4HLQPcX1`dT7Jcut
z;DFeDrgflz^Lq{3oGcn3L&IeF-&-Eig0ojE7`C@1X6iPX{t!>dr5QWKMR=Ye7r*<1
z75(|Chzw_8^_;#?0ZXnLL?+FxH&}&;@T_VeL;2@#{zDlR*Ip@pg*|!i^)CzcZ$%K}
zS0x15hrg~A|8vp5KA*ngOdne2{G`8k_Eqp=!PiO*q6zqaoD&Rwn)8*B9%+X$xBjj_
z(65yxU}5tA)ShH^8!FRuJ=9V`eM1Abn3&iP4LT}n>dJZ=zProI&p!<W<0$0TPfm0_
zvnXptI&^Vj0VkizqZBpNYK6WS{wbRuEV>0=K++=2MoE^Mfkco^G!kBnU9fR-T2t$1
zxfLDxCXiI5#e(n~UP?wBg=1aiolw2-%ovR|;|8}aUNZHCB)kw?%FQEx=Kcp$f>WM#
z{%N7P_CWFsEu({3%Ve)!xp@~0T}Eio>&B<`=ZCA*?6c@7TsXx5h(mpNkX#yv*lXUB
z6_4CjJP1{K$V)fKOBRjlQU*v9aiKqFCV`-Vb3fs^0P)Y{ckd$^^oOg5CPtDMy%3u?
z28HwCo-bTEY5^bK!{6}+N<K9%W=lpQ9mMRt@Zigopco2H5kcKU^7WK?O|&3@(@^r$
zPfRr9c&9?E4v56M$w1vhZ7qU1_(DU2XiF#c{aUliex}k>f2dhQHWqo7kd$?m7adXB
z-Puu<ny=76lSUVOP&L%CrFbdDc?hbFrIF#__j<In;kPIc>Ua<9P-OaP=7MoJ<RQJs
zfr7D5H*sbsdzkC5<5w)b{`qu#FBNp(q%X^@A0(Zlb^K;<r3@HBncHX8={s6<;%+qM
z1GjojF`J3@Fy73n*96Qd7Au_eMC~t4Jt}S-q@7$=RWND3bTrR+^9)@?nDa4LeDNjq
z8iDSO(s}2KG4riGPi?sJCo)u0LU)v_y7%SlitQ?_3GFYSW|8K}w4fj=-n}Ml6>OT)
zlHTV!EXL8)Ff4&uBySZ`!-w|Mo}{Id4$Rb?@~X=*b<+4^9v5vGa!ifz?X?SpH}v01
zf&i|Q&B+g!b~+4SUW(~TK7aQ7sHR$x<1F>(kdZ0kK{W--cH`M*guGkWVLom$=dL;h
ziasge+|t2Vj@l>UJSxPpTn{LX*}rCqoK3E0TJe-9Y9?%Q<HiheVQ?|#Ihd_UNypqO
zAru`~zv?g-Jy&=}=13nEu6n2$!7;M;hNm{ZZm1?J)vXr@>sQitnM7oK8|Q9&(7dG?
zSy%0r>ANzKO`u>{LI3P9bYY47lmbk<I9M6_;`n52N?ey5+PcZG(VI4On!tk>t#C>w
zGnUi5H*^{1h;+F9Z4?^RO1nlScmHI4mzZfy38P=5PN@sLeu4q+A03tGtKX|yJ<CV!
z(5a0y5&EAj<pOI@P_f+w_+_|{rW=QTq(2tpMdS_$?1P`J2GGqJV<e!(nm^*E-OZ2e
za=z)R;au&=vMWtn+zMa1$l}=QNisLeN7Nhkeh~9^IycF?UbJgK=-bU7jq&}aZ)~Hr
zH!)ArkUBVcxg^w*J7_ifzO2;u!>M=aZ7~%so+!aD3+LbEFj<h9<Gr+a4%~1hP`{W+
zR6E!12}Nxco0R6XgRuZ0e%WXQ#da28>PAHj;9)Uk;m!6i@3tBZF^+AAyNC*4+V|8Y
zYcF##zt4u_Zdc9R2QnzY->in>Ut(geaS`OQSDNt<EbzW{JJZVz;wl`|X_+8L1J~P&
zO>e~-xTIOC%!Avw#RZGfB2v=3i*x$=g&KDi9OQE@eeFIXpp&nkUgLu{RDKkb-P_YY
zeF?j45gE^xlmZsO34thgY*!<CCBAxYY2#Kg&7q{7&$ORv*3U}VduQ&rPI?G>VU@ha
zrE*c`q;#P@*H$_Q$w>E~kpen`%@eoI$>j@Z*d6oGPKUQ`0FW`GuxMXnNICbIKs>B?
z0NiWh^swjG%U!s(oeZm4K;b9-9e=3rd<-GWaiAN0^@HRZ2qzPh-RI^7G-w~|B3}^O
zcHq;}#`1=VzV8{p8#Q)o21#ya1L}L4Pa`druDaL}*fs9^_{2m%|8(H(U8Vg@t(+W%
zpVemn7Qo;33@Y?p`&lu>at}0bA+>Binm9;icf!m}cZTKYtLUzf-|jC7Zl}*uLP|^W
z_G>*jxHk(MeLMhoe+E)s+A@Ge;CQu{D`L9N3i45xe2vp+Bq9Kj684m8DyiKebq>0z
zZ8ntWsU+$jEvTxiB`Edd{q^QiE46OZyLM5bOyYbmYIKp!jCP&k*m)|$XCK@@Ie!ie
zXmGiDgWa|vRLYLk)z{vhj}hxm18YGB;Va=Vy#rHIYnL<2Uf908AK3=Z0;cGVeSxIr
z$hgu7!(##l9!wOu&jMpF2KX$JyVgYsRQV7e)_pB6&AN^Vs`X$Aw)es463`CZ7QUFO
zsa~N2;j}bLAGbg7z<}Ip^H(coZ)vv}fKHyb6PL|f>u<0NFYkH+?$zaYx)ak%=-qO}
zlpk65AHx<|f`&`oCV)c+4PH~|4QGY@#!;s2f&8Ov5Ynx4``_iEL9A=G9Zz3d3ix%S
z@79!|*^u#WRbf;U-WST<>ELA5iCnDNuZqNf?A1jaLFK!IcW8rM80~|riF(!v^58zc
ztiF|@32^9&@)-wkosW#hJJRev3HYF%f$s2@9mlemb}C{0SF#L2Ls`V?D!WVK%Yi(e
zMJl&lWnG}a)04b~g!I0T$_U{J(yi*WHMVg<wc1ldoxE;PL-l6FKXkJjMtaQ0+8J(P
zmN^fIdEM22z#Y7~XtQO?h5$=4U7ta#L<fBa&FPU|K-da5=io(eU7wA<=`eN2*et&q
z`f6m7ciMO9w$}3{7WQI3mj%ptF#vh>9%^BB2cFv6+~{F%Z5=s|pvxH0rob2><|Vx_
zn^Z-^1!6_obY13I*LrK6c4XhjcLL}=b>15?EibMrz8dU+`_CF7nQe%Ej!|i3xA`QQ
zsQwV$Tj^e6jDlb)*D?a-eo#GMuS*yb?{)oxDrefHA6XG<d~w;PNmYIQ#g+thN#5*=
z(ZsM1C&d(bpJLM!;EYk=(KsN_F)h>W!{YZT>v*tz_L`ODjS11uafkrnUgHW5ku?bL
zD6(&#yY9V;EE@5<1bk3*`F+5xYgPKkdYtYTT%+G_4}{yL$(}L*`L#ffKY)TMFr91F
zF$uwHD{(~j8R|=}=rSJk@UyUl5Zl&{E|!PRNuX|k0KMf|0IEn+?pRo*V;hWucMUq=
zgy-9$ha-D)1%&k~AMS)Bul3;16eo#`jjou@#}xR>1Gv49jc=wJ#}=!lUT#^lU$Ri>
z4r;F7`STlcM<+iK%ep3XF97ciXg@t=>G$H#72$PcWiqb^%$X38&V4?ebM1Tkh3{f#
zpnRWHVkCb1K;mT1FVx=+><r|0{PakqfvbPILD-#7=&5%Ed-ZeVUK|E5f8@)357ItT
zu6~=v+hal3X&(!1L#}@EihSEox(!et&w1pt#`-GOD!Ru`&N^GmGO+xH&Q%;43MzNm
z@5`z0r^<-6h7A?4jAP)Q8@~}v)Nzj+p0ho#;oDBk>W?X($Md!{FhLj40mpB3fYh{C
z7b1f7dOsL(&eXn9k?g?q`$a)IsU{=@Twh@R#R3(yMMoii?8EycIr}}P%>mIdK2~@1
z9clBrY!ONn8o5taSA}%(C<wg$x)nzkGQ_yX{>}O`^8HiAUhH@^YyK+L=XNeBwlZog
zt)2+{gG<W}XP}+Qe4GX~>xPiORmXa=@73oc_>J%l%T4*lw9%<G@y0lCB(DJk%#u`>
zMy6u7k5nx-abbf6`{r9oYhrlckfMM*fD`enC%u+-4QQNzMDx-0z&!l;<aWch!n7A3
zs7J0u+Vp%OpVg68>hAoPsE*t;mm~$gl9*JkEA?8YjEC}|ry$MwYCO0$nLzt$$%J!^
z3%vOYCyq{&w{3ZfBd<Kx%UpL}of-GD=VlIPX=ker))0lqV4=Q5(7{|$fp_$#%&|af
z%(~sG<XQtPHoqA9ZuJ|d9(7C(N^h8yvpHGcQD1J?IZnqCqI312xq<Ckcc%;&I{K!(
zE;h|vVb#v+##Ws^JmHf=Ss6n{)U>N0J15BJ0EjP?T3~A?rle=G1de0Hrl=npJyq2%
zJ*1y)+A)9n-G2M>vAB30>~=;&4e9HE_TSd>@w9vMC%a@j#Lu^hHY<0_97`|2RZDka
zsj@-m_t8nBHkuW>_I+-<eW(lH+UvfoG3K90%14RnjoD$NlAL8=Ii0IEot8aU*xYl9
z+qNYI70k)f0-yCAJz1(Ml1!oWM#fcD9%}E?Tzca+ALOjT?7Lfngo3-=J-1peg+4Sm
zCV@2aMu=q4?2aJl5*k@dDWHo?p?0ETyz?uj+2U@V^00Ok9Hp;+h8?f1_LX_r5F)hT
zOjVruWmVhE?Et4|6G(9?)jIhrO}RNNh)25K&Mk`Q2h23fXOE4$)}kpq2zoP}#_2@H
zrE#zMgoYaWpnx)TWQ261debi;l~g+UlU6nb7-bCAo_&+W*6*rUi$4<MEE*|B%3i*>
zL)C{st)ru#uS2ny8nCWR6xnL(!1<zNJl;ziS1qWY&F7zq4j<B1ZJo7mUNxQ>nl7~y
z-##YJyxB%*R12W+U2UTInh0etvfvlAh_{iF!s1Nwh|26n)L^V$Gcwc%n!XGx!!pgg
z?c6sj(WL=mzE}7nlzRH@t?+-l4P2pCmzj|{*Z8mkN+(Mf^?)8<jE(G%V<rN&P@K6?
z+p$Gm+H)5}8DUdk(b!ah_s3ny)53wT<Z#VF7QS}izT5!Oh|Ev;YD|O1jRp^gsz1&j
zBE>z;Y0SI(IW<yWCY5+sOB06Z32b}k7$X5_a-UjtF5YJ#OUOsK!ZJ0^dk)iGgCYm9
zpl%Za^<feBhja<)>s@w?P(HEt#s|eo<UPM-D;~G+1~$G4Co>f!n$=KEd7IA<+Fk17
z`&Jxr*GT7g39j~idWP>FQJR8zBPv?mw%(*1wNf6NImh#oXX@twqB@GsQZ=U~`5g#u
za7y*Rzj?4PeL&!ZfAU7ve>16Wyq$Z+WbR9%hyCa`Y_7iI?%~mxjtY9XLcUNwCuAy*
z+1HDJKjc9~-*y-0pQ3D*E7VSiM7SDE@&4}p5LI|35UOeX)bScnBx?*A?sU5@wm`=K
z>@M5SH8aX%$=mBc_6}7QR8Bj#l@kjAuO|`}AFt8WobpIo*mv&o>A>TuY>vA81l#%f
z+UV|9$uJcam7L~);~nM$N2@I8ajE=7*i{t<x(|7?FOGiV<?%KZxr=@7TiWzP#Td^*
zt%8oOCdd`X$$=*C8(uBv=kzGw#}`9Io{TK+2du{2v3R_eLj}!Rn>A<+H?604`EnEa
z8XxeDm7j#btlP>$9u?Jgl!s~m;%M?;B>sebcOow;?HpOSiDh{f%f*H~R-Yclt=nX%
zm?W^DwblFy#u|B)7GpaOAT1*A&IyGvxa+;(Db3Mu=F42=%BMVg#=|;huNN5Y#v6$J
zVz5wCVe^gU?AVU?c?4lZ`}x!|^x8+pY$bqs@i70#IuodIh+~a$)I@DX53RCnZ4Tm7
z66jj}!h2n(<=v#_EmG)2%d_L-4{vxuvxnL|s7^8up4(f@oK{u9L(<RC>Z-L4b}PT!
z7V2>l`PdZT===;PjBZrbw2+fh<o57Pa<bm4kFEZ}R~g$7Pa{eKkTApv5QYF3AM(qc
z3u^;D0h<2{ka4wR6djo;pZ(LK+1{=N%EsJGuOW0uR(CEyR-}F!>r<<R=F9aCi#L^_
z@Az`%kBV=~hI-6Sx1k(MzJ2p>_tYDU5(*JN=XL9pArsF8NVHH^9Q2{i2Rg9~-JNke
zAxy+IYrxxNPoga+>Mf+~X>#>r9m??sY$vLnD4{;YE+C$$Bxf0%><Ny|zXYH?CBV1X
z#TJ&1^&P!Dqs7Q!2MP|MmIC`=zqy>l4ts7TD!qkVsU02C#KHbfCZLal+^#m|;PYgC
z?sY5VO@GoLW3O#fr#Kj`9uLrDDRwo(2V1D#gLTG5fsm1>!DL9-L9H?nW-UAVs2Qd;
z8};qE=5~o8^u&!`p2EHt5L4f$-)XZ6FeSSwE~TN&a{JTt`sEHXIJukpaq9((RBh_^
zkZ_h*9r^--?glVIR1Vj`4k+!^vgGQo(urgx3=OV?uRVwsrTTJ5IASd^Uksg8h0Ts6
zPqRqE;F#$9O~Ir2a!a^_(kRbcj_(DuHYI}YvsUkkG8y)CZb>yr4Ht3T4)mMMKO~07
zmWQ;rDm};UAo*+qocpj-`bUMl^Wg+s^U;Qe-Ix_Yur9Bx!bG!k5#jWtW%jU$!F1Lt
zR&Ws02QoMW#PLmbUfluv!xp&as}n&6?|dThUH5Vc|Dd9ow{4LF)6JTwE2A@oO)*M2
zfEUBwOmuxVv|a|%+g6tbWI)$i>in&<MTvq(w=5xP<mxl6^gd|c!`{tCFXi*`Qs=n+
zq!X3i6ZmW!=UVD?=Ka*#X|-JyCF{l9ZTCs<$i2-$^ZiDqWwL8rDSJgcHQ>6jz929q
zLI~A%$Yy6L2H?3%Ob0meX?a>S1Nv6*FxkqQ8tQ8{(Ljnk3AsB@OSRmc>Xg>>zNVTO
zXy99JMDZYhIFgD-z%r04eNW>wb@v*Q)!C-59SOQ3JRfWq`1xEms35?L?FF0ISK@;b
z&pe%0^;sQnnVlW;jNToYb?&IBeXnwFVpYtk;nFm(Ag=wdcWiZZ|E~Ejnjgn2@~%zo
z`z8^DeRpD!Bi41|?pN~GeD1AT(P=4%dNS}+c3=r~xXLg}#Ibs&r6rSyQEDqQmR35)
zFg8o`z}iMqmncT?^JPagKrhD`Zv|&&fG=w_p^65n$mEIgQbARnlRC$grsBK$`Ohrs
zTXS?W1y4KEoy)tWDb;Lpwl)WJQr6emw_RbZBF>YJId+rZi7<5Hlc_#-ysOY@d>MP-
zUxsZ+8tQ^(xaAMJTRr36bK+l{<6d(b-Q)EU5m`xbw+pBTigBlp(d=Q?&C8Y?AJq+3
zU4t1@EKR-j8iAa+dWJ69Ul0TFN==;Q>Rxc#i(Pim+!fMq+;tbX?x>m%&??-K+^w8u
zVz5@+|HU)3<^a*Ez82wk?rL)7-FG##bfO(W`;kS0KF_|Pq#$kW6LA>^3mI9rmr@dR
zj`iGQln?oLp{kaGSx@44X1y%v6&*QIS0C{#>Q!3KiE5doUq2WsdCw{Yc#5YK9}0QQ
zOutQ3zgdy-8l?r$@=X6XjL}N*ODgxJwca9cl+XCEaXG}th<b;%a$06WrRf|+xgNnw
z^wR5OFO9gXs$9s7^cQ>imGgaoT1@y^@dI|ydaARtGnH+r%*;$xhdiMC04g;Vc6-fv
zS1xLx#fpZz_t*CP{}lgQh3rh-S&)~L)vnUXOL=+uEYhX&)1$SOiBm0JfgWW(qMXY<
z_FErLzf||491lT|9Mo}_#V{>l=Xa{VoadErOF%8={j|SSh2f2Ij}u?=W<g#I_EJ_d
z)ngCB-TKq3`|7$Dd?Ft|O}wtxpPo+t(>wH^bx11)>@8^4cqINm-~4%yJp(tsZqGg~
z8~XQ#S}}rqVBDoGy|bS1EP08fL=S~K=A_EmKi4D;5dR3S`;RkS5aPtY!$;&)G^rZj
z;<`R$xL84g`Bs&`_}QlP$N+tAUJ_B^FaE+m+O5Ap?V6N#HfI4buu5k@6?E**GJ+$=
zw8ns&<VHK-QL|(~JO794|5EQCdiHmf_LF&Bvh4k6`(G;iV;Bm5iTk)e9`kpJUZq~7
z2FHCqVcRO?_^;9aLz~kQelP$Z=Y5HPboEuf_}7!VraGvk{pPHPq=NsauEGF#M!1S!
zw?Y0zUw<6mpl1KnC&Yivvh$Db2=b7ltV^AN+If!$BIe#gWC;LBHUDeW|7hc<kG783
zu5b1x;h#}uME<*|`v&NH^4$Y*lq<Ts?7z_#kv|8~c!5Hec;4Z}x_$gRm|r_|{!z|p
zA#^6s)E9(I|Gaem?S+)=;DoX`e+nke`)&7LMf`sa|F2h0s_D7*nA*3h$qRLtI)qu=
zHLBV7)U#8KlG0i+&zJc!7eSHv1*6IQ#0F)K)LEmZsy=riRflm;-%-#k`k+*JluRv~
z@d{U>Rn^oQskW}_vh&ek>rB7xmrD1BFWZ*<u#)hXNj*6T=^evJDff-SNZ2XU7%RR`
ze<VHhQcbAk%egKqS52V$pyl@&;(?Y%+!T-BxUH30`j6Gag3jNsSNv08{pL6!mf@6C
zAI6pTbJrD$mx%q9c?m#GJwYn<fy@45luwsk<$YGlean$;Xc*kgwQYE$TKh)|4MzX3
z$F&-5we!Zw_mqhwbqZaBAri!L7hKwr!=*;c9IjBpY3$px@!dEfP|sn0HKb#bJS+|j
zYR^Q=d*P}+DFLP56eEE`fv^jii<OeRj}g?N$EaXg_0a5lu}?qcRbm<-fea<$3$Q&@
zZBmHF6F=Tsks|BhdM<2Zg*D0$h`Gm-X5L&A(`i-0!z+#TvejoM2z^b@ctXW>f+cBc
zDlp048y9@R^wlOIXl8umQJurwJMJ!?9$utk@|b1Co=`a3n!ecn1Caa6B>FuiR{PiE
zVZ#I>r}^WoB}JzWA8U^#Zj5`C#I_l7^8-E+<S$Qhhyy;gYm{oBE_F1I2V{&chCwom
zoWKB-P1tCJVC-J+?e<wgJrgsN5lDeILp~ZcbzrQRAZ#EH<cyLYWI&qgb6R;j_A}4j
zwoxI=eHud$v()-mA`~vDP(+&)^UBR_JrjZk5k%jy!eLjNS>(surY79kOQT~iC;C}N
zALMpVVuF>l@l1DlIqyiJHbHttCL{Gy!7Z&4&%F93a{~pb%h%0Awu3(>L%!h$VB+8K
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
D(*ymN

literal 0
HcmV?d00001

diff --git a/doc/ci/img/pipeline_schedules_new_form.png b/doc/ci/img/pipeline_schedules_new_form.png
new file mode 100644
index 0000000000000000000000000000000000000000..ea5394fa8a685e582f0c649edb6cd9309f4e2845
GIT binary patch
literal 49873
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
Gul^4(TZ9Jy

literal 0
HcmV?d00001

diff --git a/doc/ci/pipeline_schedules.md b/doc/ci/pipeline_schedules.md
new file mode 100644
index 0000000000000..0a9b0e7173f39
--- /dev/null
+++ b/doc/ci/pipeline_schedules.md
@@ -0,0 +1,40 @@
+# Pipeline Schedules
+
+> **Note**:
+- This feature was introduced in 9.1 as [Trigger Schedule][ce-105533]
+- In 9.2, the feature was [renamed to Pipeline Schedule][ce-10853]
+
+Pipeline schedules can be used to run pipelines only once, or for example every
+month on the 22nd for a certain branch.
+
+## Using Pipeline Schedules
+
+In order to schedule pipelines, navigate to your their pages **Pipelines âž” Schedules**
+and click the **New Schedule** button.
+
+![New Schedule Form](img/pipeline_schedules_new_form.png)
+
+After entering the form, hit **Save Schedule** for the changes to have effect.
+You can check a next execution date of the scheduled trigger, which is automatically calculated by a server.
+
+## Taking ownership
+
+![Schedules list](img/pipeline_schedules_list.png)
+
+Pipelines are executed as a user, which owns a schedule. This influences what
+projects and other resources the pipeline has access to. If a user does not own
+a pipeline, you can take ownership by clicking the **Take ownership** button.
+The next time a pipeline is scheduled, your credentials will be used.
+
+> **Notes**:
+- Those pipelines won't be executed precicely. Because schedules are handled by
+Sidekiq, which runs according to its interval. For exmaple, if you set a schedule to
+create a pipeline every minute (`* * * * *`) and the Sidekiq worker performs 00:00
+and 12:00 o'clock every day (`0 */12 * * *`), only 2 pipelines will be created per day.
+To change the Sidekiq worker's frequency, you have to edit the `trigger_schedule_worker_cron`
+value in your `gitlab.rb` and restart GitLab. The Sidekiq worker's configuration
+on GiLab.com is able to be looked up at [here](https://gitlab.com/gitlab-org/gitlab-ce/blob/master/config/gitlab.yml.example#L185).
+- Cron notation is parsed by [Rufus-Scheduler](https://github.com/jmettraux/rufus-scheduler).
+
+[ce-10533]: https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10533
+[ce-10853]: https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10853
diff --git a/doc/ci/triggers/README.md b/doc/ci/triggers/README.md
index 5f611314d099b..1251313cd142b 100644
--- a/doc/ci/triggers/README.md
+++ b/doc/ci/triggers/README.md
@@ -216,42 +216,4 @@ You can add the following webhook to another project in order to trigger a job:
 https://gitlab.example.com/api/v4/projects/9/ref/master/trigger/pipeline?token=TOKEN&variables[UPLOAD_TO_S3]=true
 ```
 
-### Using cron to trigger nightly jobs
-
-Whether you craft a script or just run cURL directly, you can trigger jobs
-in conjunction with cron. The example below triggers a job on the `master`
-branch of project with ID `9` every night at `00:30`:
-
-```bash
-30 0 * * * curl --request POST --form token=TOKEN --form ref=master https://gitlab.example.com/api/v4/projects/9/trigger/pipeline
-```
-
 [ci-229]: https://gitlab.com/gitlab-org/gitlab-ci/merge_requests/229
-
-## Using scheduled triggers
-
-> [Introduced][ci-10533] in GitLab CE 9.1 as experimental.
-
-In order to schedule a trigger, navigate to your project's **Settings âž” CI/CD Pipelines âž” Triggers** and edit an existing trigger token.
-
-![Triggers Schedule edit](img/trigger_schedule_edit.png)
-
-To set up a scheduled trigger:
-
-1. Check the **Schedule trigger (experimental)** checkbox
-1. Enter a cron value for the frequency of the trigger ([learn more about cron notation](http://www.nncron.ru/help/EN/working/cron-format.htm))
-1. Enter the timezone of the cron trigger ([see a list of timezones](https://en.wikipedia.org/wiki/List_of_tz_database_time_zones))
-1. Enter the branch or tag that the trigger will target
-1. Hit **Save trigger** for the changes to take effect
-
-![Triggers Schedule create](img/trigger_schedule_create.png)
-
-You can check a next execution date of the scheduled trigger, which is automatically calculated by a server.
-
-![Triggers Schedule create](img/trigger_schedule_updated_next_run_at.png)
-
-> **Notes**:
-- Those triggers won't be executed precicely. Because scheduled triggers are handled by Sidekiq, which runs according to its interval. For exmaple, if you set a trigger to be executed every minute (`* * * * *`) and the Sidekiq worker performs 00:00 and 12:00 o'clock every day (`0 */12 * * *`), then your trigger will be executed only 00:00 and 12:00 o'clock every day. To change the Sidekiq worker's frequency, you have to edit the `trigger_schedule_worker` value in `config/gitlab.yml` and restart GitLab. The Sidekiq worker's configuration on GiLab.com is able to be looked up at [here](https://gitlab.com/gitlab-org/gitlab-ce/blob/master/config/gitlab.yml.example#L185).
-- Cron notation is parsed by [Rufus-Scheduler](https://github.com/jmettraux/rufus-scheduler).
-
-[ci-10533]: https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10533
diff --git a/doc/ci/triggers/img/trigger_schedule_create.png b/doc/ci/triggers/img/trigger_schedule_create.png
deleted file mode 100644
index 3cfdc00b7a70cc4d5b9b134c87a2cf45d16a330c..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001

literal 34264
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

diff --git a/doc/ci/triggers/img/trigger_schedule_edit.png b/doc/ci/triggers/img/trigger_schedule_edit.png
deleted file mode 100644
index 647eac0a5d0f7be6527c8cb491204905ca2e0a79..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001

literal 18524
zcmd?RV|%5~5-&WlZQHhO+fF7nS8O{ITNB%wSTnJciEZ0Cnf>4UY@X)}JnO}}?z^h1
zs=KPXYxS@DS`mu!5^&Ji&|kiMfs>LHRsQk?B=Ymw5EA_JTc<rb;L8^vZ%YvoMJW*x
zLPaNgGfNxOFJCAVObiXFrD!RC8W|ZH{v4;Ff_4HZhlfWi8+P~ijQ8{t4j2v+=A`NB
zZek*C_WPrFcMSDF7BHLyj!ECVddwC#dB3Bx*Q7hClX17+1poo0tsI#_Ku93sv$K@5
zvS7X*eEH4&3rvhm@E(>>r{%}@F9ZBQU!zcg@F2f#0M*SBogxsrA*YQG0IQ=Q@4|kS
z`SKfB3b7eHJR9Cf$A9S;Rs@w`GsI8&aGW0h*?zI!)gI-=1pf{x@HVa@MjpZZ!Nd#<
zY*;4<#8F0f##Tm3#xxd8Mos2JX3muF>Ph~n+y@w{=%&TyNtB$xBJ66tTmmh5jigPI
z61^-8S_%@qKYOOWe1}IhM=TV82#_;J<%vTaHTc<MST@$vvl@gblQGm&>w^FiG(Z)c
zHo^-I?rRMmYI^{U9EH3EZ~i(Y9Bz<hW&>(;!^y;0yZ-X=(SQvC@<Iw8+Kuw@@zHts
z@zEU>a1XL`DhU6D6kxd-ZH<iZ<qI%~rK+ZjrkpIdiM=ho(N}w8Q+jt>htD{D`NHeY
z{rPBX>S9FbZfj%b%<axc{I?d|pU;2D48(+gYvN+fN31EQNGM|OWJ<_J&q~io%nwaS
zNXYB-)r?zNRQ&JapKpA`7A`Ih+zbo=0DvCALT~S6&cMXQ#l^tL%)rb{_t}EZ*~8Ao
z$eqs4ndC1e|Ij09>TKd<>EL2%Z%6n?uaU96s|z17@gGC~_w!djP2Da3Y01v{@5A~W
zAj2OH0~0+X!~YU<u{8UCi2dRGCHA**{bi2#k1}pWOLtQnEm2EbQ#<F+uJLoR^Zw1s
ze=+{q&wmlr{1-vSf0O(d&c8_h=?J%ilcnjWkN$XqpNW^@zi5AN&&%+~P5<Jyzhd$?
z>Ss{+p?MkpZ-DusafWF@zkCt=A|)!M>JD_44X3N7*8btwg&&6O95_UN?h;hDyOic~
zBfSH<hD<9_3`-khDM~JOF6sbtyOF0_yd#lMZ-vsu3Oc?N)vIZgVlh5g4Ywc7DKD%(
zb2yj|O@0%<e|7hv^Rk!avxz%_gea6V80LKc+pE)eQQx}8oNrRQV-kxy90gcd5E2rI
zh!8Z`9})5IFNTPRG^GFX{!gYRRCu065z+Af5c-0Ix<~#G4gZvY1VR8Nru5v5C-@&S
zQyT<<C=~LKH^hq}|1|#(;XDWwOh!Y0iF2{S!vGQA|Lo<TW{{Ci2>*)?0fdCg=kODO
z=pVZNWspKj=Rp61eo06eSO5Y4MD(fl|9ax@;>46AeE&leFg~cL;%yI+z<)~!3xbNZ
zLHrMTPta$l@i}+H{?o>%d&2W5AO6esC*2(pQD_<@?Dc<n_tWe6|BL?r;dO^|FfntM
zjEoHL4-OuXiVzhzN)K9?BZMm9x3ZqVaE3q5=?Rg|)4Hq5#rYU=bk6g!((t#e7Fqw9
zfXT&-;5v>cTg$R@Y^XjVbGZ+0<7ZFP+}(|*^I^_|4G;Es%FB>`mJ`8^aAaH5#t}>8
zIfW@*Q9{mu7mBS=-)+sm%y+@?D;cpB5$aZ8F{HdEEht!`r8!{?_;fW9QU`~>QHCPU
zWRWH_-`LZEN#OYjhVZZCz+eWVpq@2OcrZF)7MC~hFG3(C<}JNzqUW1cjdP;wb(&+}
zBX+9Y$kCr%&*!exZUCB~F^*5+Rrlj><fx{nyKIxHJiqfM<x`%az8)A}G!~jUR6p1Y
z<^YzrX*IBCxETQ$FNc4*5VKc$;s%sM*;jhNH^nl_dbwlwwa;=E4M*p%-l0CYV?iep
z>otyFecfdkRZKbMzqp>t0kQXy^V&WdVdP_))cYkdbF2z{pj(Lfx>gt7Mg<?CvTbO)
zz=VQ~#6bk^h-H)`T##|Z@=+yLB#C45dcV@i@?-YbZ*{+h(2-mLfd$^5b$^m04+ja;
zr6?y|vMNqabPVzhpl6AXJhWTbtMjp6W3OuTBZOu2?4tch!N=gvM73KhAd;gZ!995+
zX9e%N!jqPPuD2YJxjGrvyv=+wr{1M@`eS|e`Sm%*!;a{xSag>+D<~<`MRhNZoaE%r
zlcOaT_sPV0<oC`}(|}QPYQQ@vrz=7F2OWu0_qg@(N<UYvhZvN!bjwcb@Duzi<LotB
zNVg}`jXS#urGcgEv`0e@yFf-T2Y<801VpG67i5FSk<6>(Wir6tCx)Rr+i~5%M|<{i
zuchL~W;_x1f&0@xim*>c(2?rs#=RyXl)71^AYGPJkvZD_eI=2KoL-ecYY}=i48uQ_
zSXG$c8`Qf3$Qu$HRAbg{IqQlTv&rRdb2W`0osGS;y3pUBPihvsZbqB?PtCG&yYinD
zJS#!%Q4fyZ)6{2_FOD?7C5imWYQYvVMw9Ae6O6o4Zv^TTK8xK3P=@#XU&#l>52Fdy
z&oYJfvIg3*;0?~-Yuf3^B$XFq-Z>)7-aGgKUJkzPcux&$ixTAfMx$ZkuO-I`HFtwr
zlv*9Sapw0psI<pdtA5kvyWQ#q`cWeh9QZQE0$H&bEjGR7VRKA7xaq+d8;d(uo^elU
ztP+zC9g{>pZrti`3Lp>%bUOtY6l|O!TGhO9xXHn4I48QM=T|=bqq^S;PrfplQn^*@
z`X|ky?pCfwn9kmnEsoEzmhWNS)!d|ur+Ao(KV>$b+*OVF!l`st4Y4@Sd&xm2#oRbI
ztk(?R9j~Wl$DKh$zw~Kzc39<w;o%m6dzan$Vn@6-oil-!GOgAt6Y!*+kycD$3dE}{
ztU)bRqIs}f`z6FyCa>s#Fs8?h0T>FoYQ^kX;_AjRiC&wBn~%i?r&YuD%)Ic@L#_qp
zdZ)QL*51dmcv3sKJbu|`J{$n~jFmiNsBBi2xUJ)I_}+H&KWCsHwJwi{j9zU^uM^mq
zYnk4ivp2d#Rjgw-A6^|`p>W>e#B2=i>GBW0PaGQD7yuqr=)Rf!C?9WovwGX}It4lM
zzK^3LN!Sq1ON7jF7MjjbIDyUljg_$oKr<(LN>E##{E0LK;xLFi@ZPuPc<dWwv@@o;
zw%qnpw-8pzMDYhT7*Q6Hs^fQl?QJB9Uxj6o^!GOLt~I+R6Ny9psa+@12Zm4Nok&&z
zdm?`-;#3eMF3-`qQvYy?Quq{$!FBgDVapQ(S$K9rN&0#Jp+SrC3Fp>2M!q^Q+r^eY
z6$^oD<BMcgdMKwA`3XCn#I@hdtBROd{?4;h8KsBDTlid+2tTcanK5lMf%1WW(D9B4
ze7s*%<kb`#zDjQCM|?Lw?;*BiZ+^AiVBeVReTUhGZdeG5@zqd?JZFa6(N@!x>(cYN
z`W)A7Ntn6rxck+*=9pF?!Z-<=Ej7e2m((($JdB<9?HvDxN8SYET8!uwh=7~5h6k&!
zS*%f!N-VbYR(^l=l(bG)k-Ev+Ez%<Mv3acw%4vpT2_H*9?I=XYW#g}64~%d2mP!{F
zDTH_Tw5QA|@lWLd4Pd@~|3pj#GQt|wHHC!m4k?1Lq*rRRsOH0+p!S*HKXvnm;F%=T
z(+jRPewe|V)s##*G(u(4nQb0-t^J4sZcDwk9$PZbqPUBax9V?9xlDgwvG=;wV9<?A
zBN&xJsp>4}<RD>ijwKp6w!F=nL?-2J8}xE}b9`KPr_C-1TyPv~39-icEr3bB%H<Jp
z_)&DxHKY^hnyGHI1>M=L6pB?`F>Bu2m^&PI4A*~H@1OX5aZPa({}K6n1Bfh6(Vfq5
z$neQ+fkfuS`UoHzZA8}h?z9hT^Mq@AAD5!Js|e|W9YBwU>a(vsp21%xFr%EJmRI=$
zmd8M^+{+ID-|TQ{=;~@;DLhkiJK=p4DjW^Br8acsMG{+7bAB6McrIn<zk?)38o9n2
z!RNKwP~Svwq}RfR$Au33S-o#XfwvaPkAE+K=g8a8jW7CA&{y~1-`9A#r3l?zZ3){H
zS!(R~_Io?_vF?853-5DP)9fh56mve)Zhd~xpWDH?B4j_?f6n8Z*W?xwXu4x)C!f;A
zMtm4GR=rpa%4RsUo*ciaiXJSc6`VikTWTl-w#<l!;FaiQdZh>{r8mfv@SO!#omJRU
z=xf@*ck~(QO=M?`8>ojXpDbZ#6A-&;Nk<9I$;`RC<==>%%I8L2G<Tr9U7ntVjm52i
ziM@CSi{YKW?0T0~(`fPSImm24>R_!wb?7p$g4?h;{<!qg^H*9P<11eWVBdA2k@-zs
zU^U2VFLb1+BD_H6w!@D4>P1sG$Mq%CUNG!5B6D^rqoB#psW@oRpI<@zmUmLT??$84
zR|(Ft<_eITJ>=sV$qNe?(PFzeOIF{~hbi|-8>?*E>>A-h_dA<b=;S>42t+37E83gK
zJvkNny^U&(%(wm6J9tC+Z8q7=7>y1d9K4N!bI$VPp#)A~f|h-G5LYx#Y9?flYBb(y
zZIk^48?Bpz!(I-|t(@Poe4WDMX6G*JnCCZx)m!$KkwMVPc0bzwoS>{uGpv|3Cp<kv
zLGn*|+E%G<&!sIJjT}A4(EvmX2%zKr_5dDD^D&n}qZv8Rod!69D2Vi?y;lLVvGyY_
zy5TH|+roiY;nyYIhdxeOysa#rnqlX5jKz|$EcaR1Xg2&Ll4`ZMW$wA@VCko;dB+ao
zZy<lGQ2>g~W0tac9#_Ps%v;!0fp7lW^M1;VEC<cW>lcv@dR~EY9^u7h#-xjVav7ZX
ziR7a73g4SeO8BGjIrt`)@=O?gv%14RU}df0vzTL+Tc5^u>m1Isc1(}H_*m1MJN+<&
z3e0nm&BzH?i%TlBTjVch#|E(ta#0Wp+O+1=5?OOwLwLE#j6M*2jA6n>O{ukkPi+>z
znZtR0?M0IO+DZRfn*uDy2j4Y{ZN@jmmSS?_<!aF`f8=;n94*FqGIl3>`zthXT7IYO
zc?4JYL&Fq)>}6mogUgkyCRD?6iT8MobETonCj9jQ$zt`IUSIPb{P-AWTCyr{<u@X~
zZUo&Ns?*u(PYJDKD$6t%XWXGU#fEtoU7iDEV2I<A<(s1#zUPY6pz7v2e~;Oail0!4
zh!VCL>)%!qE2<Gt*q`Z1-`fLHOvyC)P59y0zZVP2Vr;%~xzIwV78wegMIbSPn>r6=
zCV<W@w(`cMg-Hp(0%B0~Y=7?cJ_xO{-e)g=VZlX0IgImhdfQOzwp`^go62L1eiLG8
zw)q{oXR7LiYn!JQ;-T~b|K0BvmZ&>e@CC{RMrgXd&7YZOJ{yjm|8^|wo5#YlUQPq{
zteu{}Bs;$~ZUhz=<blZ)*Hq|$A-r=|FQn_>C}!Yl9WDT_boF3Y@^2-_8H!aUs1)In
zMB*?r<Y-6%cfTUw>>)58{3!z9_~lnnc>bxiNOAu2yu**$cQHHS1}xVLWs4oZcM_^O
zLZ^|5B#<h_B>$xmCi<EK9gn#b#58jV?HBW#@45B}T!V2NIm-tfTGTf)y3o|;WwPd-
zUQRCYL-vf~8)8EH8aj2RJ7%oPc`~DtLSwz+HpO5Qp%t!@R2({Y@w|&eH&d1iFOYTx
zHzxdQP-||fGpIaByt)yh#bG@sYX)lBEhmlABiJU@Osd62S)`WVe>V_eR(Ok%W8`l~
zIwf_#6HAxAtv`=YpApO{o_W~(AUpZG^N{2A;<(s=A7U%IayD(>dCJjXEZbU|`?mHF
z>(SKVM^H$L?sOBxN<i0}rO{mQBcx|HHAtyPsP|a^3>09>_+#WFg?+c1!CL2s>IqQ|
z$D)OEVtpf$373(OsW`<~g91e0Goq;$z5s5X=rb=}ggmi!tjv~^a`%IPvn4~9sg&y2
zE@Q>UYE6#J0(d6Nx~|aadp(v|$>`jyz@*=nk}`4whZi9v*X1<2EP~q1Lw)ZP^kD-O
ze|CtR1fa152aq|-JQN#E4?{CpKRKQ!1~{R?-l}^v`(3E^_r03N6=%#!HjAf?APJCS
z4WZq}2`+@sh;WRH>8Ho$$}OL3PB0M)o=Dl~<}xT-Ja*I<hm3SmKBofh2LZ#r3Xq`Y
z&wyO=9w)weoNgvio&8@XoE>H85Iz=UHF;%4`>k8!_mg-1q1`s0Da5;dROa*kU>t8!
zJGN7d?>gkE=-~Cb2cU+_b>a$VVk`eBhN0P`NwHoQdHDqH^eQ+V&q|w{sgX@;Z1a0B
ze+}tfByhw4);J;raAPhjh#ke^A&lrK`1W=l9oIdqQf28`XW!LU%4!_7*pttSxv1o7
zO9m}N<HYMqd$-@_Iaz^c(Pz9f_p^%2SonEPXQqw|SSYl|;QX|Ow{{!POBVJ@VlRdo
z+XYt&YdqVCza5+GEwPUk@!a_>O%*mqcmmB)j7tzO$$74s&wDf&Tn`G}oT;AVrs@FO
z?SY5t(00f4vp-{}nj6@Ph0O1Eo7rp>JeXE9ntM20PKHjTnmzEeA?*ffu{5&n=odnf
zPU?e<ND(1&0(Cp<2T05V<-YYX#WVmi7caLsSkN==Be!gtjjn~%{Gv0&OeLxT)$GBJ
zp=5wH-lUCA#5}37YXZHntgGlC98MehDv@V85+n{ju!$LD6Q&Pd)r$mxoNHbvY3b)K
zZ7r>*ucNSceQH+DAhbhV*;<mr%?cYLrlR&3qpmdbTj11Wv$LE2xRQjitt-}Ls@f`~
z?1l3zehvg~9=RBd2N?Jb71Diw$72*o8sG+r$BGAnp0%4LczQZxGyh>i8}TH}TG58j
z61U+ZcE5XGuA|9RU_6FaJekn;H#x7Q(PBpAnb-a`Xy;_y);@_&6SE`tRjTFZN>g~&
z=4CyycIVEB+@{Zx%QNQm0z>*%8f?~^azHXUp_2;PnZo7LHBv3tek?MM8ok*rt|Q=c
zL)6K?V`xuA(X4iTS?AfgShWSeLoh^SnubD#dYL?`r6EVMm2-r?j(74~pWFd%zYo&Z
zNt7-){UeGt_{z%}22OoEP<5xX$nodXp2b|ZP5}re`Y~5;Z-X<rBh=U!apidA)sF?g
zJfoZuIQ1j~ZF#Mx6zV?l&&qDl867SRy`v<)L=88oUaWj*G=UyLW&@Gg<hD8CMfh-V
zU+~oK&)W$(&tjW)Rj4}G`7PqxulZ$eF@Q}<mzj6kiwllKv+w>_sDAy)=Fe))V@(pD
zE0dp4u2$B?j-Z9m88sx=czKDne|mGuLfu9P>Wz4)9YYL(n}K7-P7kOaGUE#?2x~e+
z(-ki8w7f0GC~m_KY`^*<!1Y)glNH*8aMG|!mLAoJw=~4B8)J$!G}b~yIwYGO7HI{*
z)s94G0lMYDuj^m+NFe!J9gibpB!-7+-Ri@y?q~ASHtjO)VmCRVA#0D9bnYwA$nCdq
z3)o7`?U_mU+-N)g6{1-b9FVE3__`nToG}oOnNdv7LMtFk!~_nLvl5Jn&|4T1f${Q6
z!x~RFaihDu<2@gwt#8;0;infM+TS?iHt&x5!94h5z?FM0tvR~F0CJ`kNqz!V@4*r}
zf+`N9uy${>)ypAQAni`|%_^b!T@lDpu3)CR_;B7zmW3RIRkPWlD|e}f-KSC?AaYh+
zn}B7aQyqTI$Y*=U`x;xczOiNl8Nrk`+%8>Le4?Fh{(TLlJ5YQl;+2)|z?JsIEK-|F
zx3Ge1d_7)9OT9`g!%hC|JhLWUc&OBI1Xtwdr_{wM+YmS{tR|p`^?9Lin*Sw)sAP3+
zX?8M9cY{lX;3dth$9JW$+tY|2-+Q?6hjN+tcJHur^!-XZown0r6})Sr6`u@yGl;V@
zQR-WX0K7M!I5iq#`RiaA6Xw`m<W01r$$KY*#R8K43$_7j<y-i5zn)rECg?m<oTW;%
zSAJ1U`CW?=STEg<Nm7u6tg%7ma9y&n@u7<khE;Nwr5-@tLmg-kDgYkw&^ffa>XJe}
z%i_f8z(8f65EV383urN<Ws=Gn*|d7#caGcebXO8t(%)d37i2|pH!A}`M&|Ml*VqiM
zoE)D_gjwMOB|U|M6VH_pjT*wdP!p6|7Z8sAw#Y6ugDTvcIXm<FCt8)!JN>iR1{`o3
ze$n0_;090eP*}FRaJ|Io*m<wT-8vOzs9j&yi<a1Erm^Ow1-=!mGp6`!7wFa-oL?TD
z0x>W7K0XBqcAekXy_59ib0vU5z+o>Xv*WVt_Em>h5`coJV~^cCwCYHCKDQx|0XOo3
zP~rTbf^_!fA9u{7w&Q@zttxyUTmlyMWe3UtwYxdIvkNuu&hL{1HjrU@G{nbfzo$JH
zO^$$a16Jd*9lM{bqoCEy8;1o)W<kma4bUe~-hN{}h>QZ?fAh4~vUys2gLDyyF@7*1
z<wE|m^MMSIC(dIBwcISGHToD@WX(c|C`lY4n*Il3+d2i=I7b<o=bk@bn<VjHFtkGo
zs5W$@UPt=4mYMnw7@$A1>fhi|!9^5Ap}LcFnaDrJ{*51CCnpFxXs5^bjqCq!Vuzpb
zYN15j^Zy%EXZ{*4asHB(f5S=gzo6?p4HU%LS$(}XW%vK~o%JW0-)u%~{_m0a6PC`5
zPh`8Z{kiD!zoW7636$H6i%$G|kp4u>lMSf5Bj?_V`@fB~hYGqQ%oxq*o@0|AFeT%w
z;v6J=Bc=O~B=AQ(6ay74T#BOBiKPB_UOQjFd1cj!l=dXNdZPs@52Rw?e`gUr>*t8{
zE!e2{kbwWQ<1X?3X*q6)d7v2o!P;C!;`DxGjK$}RWHFz}>JZf)SS?W?00jdx`Wa6|
zSKat_lUq}?eX_=|dHZX}*6Z|ph0hDZz+mC|U=aWnpd5W}@r>xchy*C1k@!!ekU$(j
z&TqM<F$KuT$*st^xY1oc9TS+>w@0%MoVbd>g<-XaWN~IdN%8K-JIt_6c&7T}4F~s`
zzXyoS5uf1{j$MrEq9PsafTT|UQ?;%;XcBo{MTn@F5>yejUb1hVyd`pnvgF(i6rEhM
zn>h~>Do<wVUq5jF<xoMe1W|E$pn8r}o26=o4`)Zl)NWZVLv+%6J1eby1Y3;8^e?32
z4{A0J<$W{G!RdC2DWAm|pi&}l*><*ALjVqo=`R!nzNNGAtj@s@@Kxcj`RWM_G);LQ
zto@A_8t6~13D!{LfetWqJpxn6XM@r=8)4lWU)QxDF`x3Zp4N}DqLd|~;wBU9X=MFb
zdjG62v51H<#uA58n*T^uJwZX+(%OSe@mHTIDcSZzW!S@6jE1n-p{?2BsbNV2OEd^Q
z)L?rXpUYnwa;?_`h;r_yfu~Z)fBDJ*2>DQ1s4l-6`{jKr_-rM=_M<0u)0Zxs8?SUt
z4F-ZjCWzsM|FmC3+Z}SVq!5bDHP^y17#w-ne4f%fA4y<bR#0;Em61zA1tr#mP*^xn
zhhhQ(fCA{c^Q>yJU&I=I*PdhNd|U~wB@iVVlbMqvSlmuzzz%rNf8^H+0c{I>5#Zbt
zRWV`d_w>Z$Wd&s~jGhpYvsj=@S<QKd6aT50dD5cqwy1DsXs8GoSmnXB9K77^8z{b@
zWs!z=PoslTe6)jo`Xm5hgHg9`UpvQhbHq6(+f}7k;iByYB+sQ-7pVg@{E7;-*u0~E
zg<euVTR6<bIX6CXi!Rv-c4|&=vVb1V$aCUXnjUynwJfAZhu!^^r}gzEJklj~{MRcR
z!Gm<=ywBldWdu@VX$`s*n=M2C#k~aK$vNdivf#4Om%H=Wshh)x?hn3ouM57(L^|K!
zHtf8&QzEUmGcr253Aw^~4@io_#`7F-X@0vt-{CQffDT2jE+c_3cIu`G@(Du3_vE)=
z;d3m+Z;p1{s}w-0c^0BIgClB7rxX5upvS2LOeY%A0Eb`d2Y+}FCv?wU;m4z{+7<jr
z<Jvp$<VaR5KO$H)KWZCQYodj$9fhy1aK2M~Hi-GqKz5FDe0UCg>Q3Bijyq8ZN2%R}
ze${P3N6CFM!ni-7D{egeymn#)7*$+qslr52QPZyb(Y(BqA-&MyT;O!2`ZdWt3E4S!
zBXq7Vl+mFaYOrD0=$HB=CVKLP;7n6Mf`w-Z;tsyW8+~$H&AGT0DU4$|P{?#kUlhXW
zdZzSM@_B=Fbj0}OSnzXl1HRWqi}gV^frjDScN+wOy7VH=tK4Z%+#kwsr?j$@HT92D
z!$WIY;fBN$p4ZupJx=337P&3>Cl|A`j=8puF7-%gqmh(l6fUuO?v3aTL$aYGw)k3P
zH0l{JdFzPb1bJ?H^yHHaEZg(lGr7JvT8>7Z(sy4yJ1A5Jc8yj>zRG{)RAMrNox}UR
ztaUzJ_@emwQ<T;t3Q$cD8}^`4VByO<Lq%CW13m)}QueU$<edBTdNbKX@UY=0ARdjm
z)n$lFXV8TU0gv0GVc<g+XiURx?kwC*5oP0C_nv>~4}9)bf|)!%E!o|#u+lKFUs)qO
zT>+n)Xp3Icg}`GpGyJQW<?bH(t6G)#DPxbl*3x1M-5Xw?nHkYCAI1((+`&;KGtpb-
z4bAD{3t27;5#;d-K?xONxnGggkH!r`I&+H2tf2rD_#f`A@DdYQJ-vwrg{MpUsM&Q!
ztc&y`?PB-#R)LKMklN!f*ohVE<dBmpO%KOki*h_&?{aVYDm3)7YGNFhX>}bqpS@*r
zaRr>+2^^xA1MN23zs4A!wcE%^j(NS?p00IZbhLB^lST$(U76zQ(<}xPlaY#>8Ts7V
zjZZdT18-L8?pBjj`s<?jmIRAJhpt#X96R#nLNK1yBP88VbTnA@?<l4k?pzlIQ^`rH
znxCeepn75$6hcMRwG3Wo;0Mv@>OkBjXT+zy--^@uc>(N2de?fs<?_Y)gu2y~Jp(f0
zTT8t&w+?J~*T8ql)q4O`yeNFKTNr6kEr8RWSgQxW>-o@4=NYIylBm17m1A>>`iiWH
z!g8}y$|{$z#qIV*TtVeLHDThAC)*C!^^;s#vHAL=(#l2a`Kw-SIZ0V<=`H2?mrT@B
zrQnF}>IG&SoSf5yZaf}0la<j*miEWX(Z&mQ0nYJ!Y0*8ElGbw5_V`j_m}dEWP(^#l
zp12J6&-v^vPX*@0%yQg)2(<npp;m5XSo7(OS)OPw&8DC#n9E1-d3YPi@~g_YHMW&_
zjKaR59p?StLR)Vclw_nsPS7BbHxLqa&QJpZjUpd+)lMkkUiZJv6SBv<@&~(-kq2GD
z1I$i$zN^~m6;4k+N6khoW2EL?wrOv?CksBvB9)g_vg`HdE;!X!0UJz}=dDeo-IkT|
zt)5e6#lU%&tLE2f-3b6C?d=+=rl1;a5OsE20s0O&4V|4?ghOEmcYFICJwBo=C}%K+
z<#PwTBoMY65+x|<13%0!5%52@egbJ9FD-p%5VM);CcOKmlu=e5DS}I}Y!=<$%G413
zz91(m#~dAn7skBvd#{UFLQg^Pxa=hTAr`g!{n>gIu;CLP71dvo>$_Y9y+(10e%sN2
z>@&n${`QbrkB5nw=5(lwvAYTrnblO4W*{~`Bu*MP-$dRVPf1&@9XCITvu~Gs_We<i
zZem+?32t^c-Wo+Q!%|}J6`B=G7&R;U1k37cbsPBNDi@4cjavUv`#l&0l%Voh3_Z$B
zd|e%?WI#su^w^|BY+Oa{*Zr@${hZ7>WujI6FNFE~ZLn6W%1Y175XTN-VGTfoWP-gF
zJhjjHp!zqk7muwO*cn>duZiaS1;(k!zg!MZF(NA)fnRS^Am2dE!|Y$<RPlMXSi!Zz
z5XzOmZ=1~~&7g-uIhLccim{H4^lIO=6OXkg;#Q|eubNX&8IFf03e23cTEdB6Dn4{u
zx|J6VV-^zkTf7~1;p5sQv~u>B2P2qDTgCL7x_KVglDF=U$4Z1Fjrn>Zi!i#8<g%f`
zJX+K5V=qJLxYI)C<RQRQ%k?gi*cE^L;-QvBFOG)UYB|qvQzKj~ZP2>ml3cT=qn)*0
z4J&sIS@y|iX^A$*3b|>(Mm5cFtp`;jXhCno>xg8k2)=uEDs!9ERDt}y4t%BZM(rAq
z5T)qQo$qM+M#0+*S0v`_m~MoOi=7xE#_)lPB>sWubQbdiM-yDFWCYA(A?)msLbEox
z!1N0BtjoT%AV<@3`k`77nF4bakc;oV!DFrWAUC~_&B{&ZiZ96no40<K#Qpv>EkWSQ
z`%NED1af9Rcpw9G5j5TaSLhaLd0{toW*bAeA&Y}#mYif6+QidlCo>WESpJryW?CE;
z5<jzAPLd3c<J_{4mcO4-ZjY8uP+&TSKbieXM@R?~3o&@Ji~Y<XJ-O@du0->GQo#Dd
zkH;4iB9`ZJES0VF(UTzb6KS{_L`q{PMYa^F>HofYVWK$I^3@qfZBe6qq^fUMf|@sX
z^=q*%pbMiB_Wm0}S&6>#w4SDH6VL6<q(Izf5g>qrgUje~Uk@1++7QpJxS0R=cw@@?
z-3?{V?<w>l@XYnR-YVIRp0E`1U7KRQ<*rPg-}xHm)FeqtgNJGr)hV7EYuV7(kI{Je
zID=cyYfygK-64|;5+8S@-{)DD4Q!%MX(zrj8aurRj4};1vX5()$tt}%JCF_-YOXS=
z$_uYa4p&uWs%%X$fO!b<14K^{Z2WXcHX~%a%8k~u{>yzQR+@Jl^X`o|uLZg6Q`1jR
zF1%@0qiY9t9{mDvIwIyJGQ_rO8s~ATS*6ExJTQeC!M<AUuN%%Z)X`efE6J=!1rA5r
zFj6SJay5>IaTZp-x0-zgNu_GhiyOa71}|*N2WckjQCq}F`Vh&>@o|EVZeWZQXnP`y
zpi1>pVsr=?3A||^%&-->AM&?bWEidYX}?l#m9Nv`av><!8dJYCtLMc7&sC;lQEF?z
zg;9lHYl~D}_qsCSL?-ldnwjnw*kqPV_1{JH=)wyp{_1ssuNJD>MbIPMd+zr%ea1X9
zy2o(~qBg+7Ui-}yG`rdaJ*z<_M-hU*Q)wpdh0xU3zx~SvN%%uM{OGDs7aK0wQ_UCL
ztaDTJdPa;?OMHK)c43Z~WkRV=Md_!jdzAaez_+tXDHC1S0y1YSf|OMeTbF$qNld&2
zX=fbC&ZmbRQ4)%9W0Zc2W?k`O8;Gn^O4<)lr4~_~zGk@ThawWoMlH{U{${z12#4GG
zA->b%B0ROK^oa}F*Ez(>vxlzW&d{=^L~ZH$K-tAw4=D0t2EloRdd17H%vJ<xBQN1>
z#c(<9qZZm<1<t1IJ2T6&-vFMT+|bZV%})eKNCCDh&H44pYf9Tckn`q6m{v>&tFe>l
z7AS<xOwkLTSsqv=&f3W+mrrx(9i|30_>|NGmJz5EGPR<?WmYt1VUleg8ls`vw#UUg
z?8}Qy`$elRG&?v!u&Vl)c6qXTv`;PHHdhS>UpnIxo%Ch49P6v3a>+AZ%y~_2AH++K
z&7GBQW0m-myT=O9P3uH2mLQVt*`zqrJ!+mx>%=DQuU6#X=V=%NAIqZxe4e)qAmD|V
zL%Qwnj^bQn5$X=yQK;iy;l#q+jK=fVmRxpKPWj%+M(_xX9_koc>$?bSU`yCLx%$H@
zB4x6KHmht<wU<LU==muyf1gJ(vnU$zn>6;wtMVlW2#;TjWIGh>Hxc_NXC}7E`0DKW
zc4eHo>7XrGC62T$9oW^c*UxxOO7Y#Tw_gl=qmkS0d{~d_4T2yc5~6Z1eYdm`RZ37+
z%9h|DObgs(y^Zn0_o4qFde+S>#<T7Y6T1Pl``Ruo-K0rPh3?_<902K2YF#$`N}G@~
z**5<^LmC?Nl)Dxg?JDgx6mXQRlz~q1@FbmjKUKz_u2mY0rWb8p{v1>6s;k$QdSc#6
ze)cZr{&M7!u{V!tWFx8hBcnUj@VZs0b%-B2k$@glSvtY6GN{s1sH#n7c%>;So|Ovd
zXw=O_EYecCjBF4AVxgSPxCM>Sm30sizFCJmw_#lfiPFcH&#mT{u&^$#axSH*mEd6R
zDhL~zMt`WTZEb-pD8>&lI(5@>nY4O11>vJLRtdS>aHV@|Myu7Ys0%<vaYr02&;*VO
zR0|qs;tM`d;oU*nlLu%JR!3uDF&R<}TK=a3zYz&AWNvhzHCp0?w6YH*ZF~D^DZpBm
zB=a2JU82+V7KA0K#qays!i}{~QG1+JDB6!@_xk(s%7xsvfvp<Ur)-jxFQ_s)m1fb7
zLDYkg_jOv!56rG$n<A?3GT<KNxN)3Dte^#szGeiA?cuN$vw68z<*O4f%*SdR5Ff>o
zj_EakL~$C`8p<WJn^X>BaV@sy_*b%ZG@bk&z4ekY%{M~>Q(6zenJwMg2NgNVI>NH8
zSa%_?ougx$W~&@751D1+a-}<UESvLPDfQX^IxIUG(1d*5B*YET*i!th%!45#B5flp
zO;KC1!Z3z#QVL<VLOZoxy-G)pMN?k7FH$?Z$RiG7243$f9fIy;jdufTL#0@@_k68G
zO!Ic|8~HAqeOW|8ndT@}JetD(v~D-#tO-mE|8#^8gTr{fk#-q*&(PFBcdJtF&-#n;
zq1^%4q)xR$+xH@}LNonlbmE;YA4&K2FIpPx#=k&Nlzv3f*&0$7P!w6M_Z1W%$&5li
zuqNdEni>^RQu`KD{`PEtr`6Ghn0yLDZ+-OpOpHf4btU7L%R3%LDH$&1{@1TmIMY(r
zXXJx?@X3139gRdGlckLmR>G4AU{V8>pfbRW?D}yD4t~W0X?eD6^2sn-ga$ljU~AL|
zT0aE*AR%;guMdQ=_f&H0-8PLQRFknZBP@XfRM+H*&-3NpD67=l=>QV1>*3@w4@QR(
zQU|+?ch0agm8fMb-XQB!ONOG?*Y8zO;@aRL_*J6!qbX7__dyNnS9Z{N1!5|9bCj~#
zD&dE?+VNvMx4V5>m^blE>DJ9Fb6T34wI+7*tf7r*qz^x3>|crQ=5VavvXZo*BFf6j
z9wz`@K@_>($cuV_Y;xLiww2jpIUWj2a9qoq4Ts{C9p-kH-0qr>ixS1l`C-lm0No<x
zyGn<f=S$)8!_zb8!CInio)Rltnt_=OwTKYIGZu90oXc5<2^a@ERKFkEq?nc!`;ent
zes-V?9mM=5#~jDnmKj?GABIxsy+}4p)M(#2r>p32@7*^?>8ZT^h`_|RmFU6($ln#E
zTR5ZR!gf&ethzsU@f||V<RkJHhZW<g9j!;O*=c)`;oJ4KZPKjMeL;xOaC=`JlgF&B
zryZt%hK-uc#4%jd(B&{ZMVRHZ6a5*u`m=MQP9I@|&ol!lW0UTg-i}T*BgcThZ-^7!
z6R>8=jt|}*3PZPQX|MNJ#|T$t#H>zzX~NM5F+Y{+Ps3{=x(!wRz9F3dlkrJbI?KOc
z!11R~IW*J=EFC3k89*h?Cp;^(T$jwKZ;3eCgj<b31+wI5AwJ%Mv}karn;8!7a7~Kt
zc#IaV8|~$=Y>xH2@s7!rC;u~?XlZ%3AYM~v`&7T2&KRS29yJoriCe#@R(6RvO;m>L
z_%y6?+mhp|lp2pRL2r15Xw_a8`1!GP)xJ5en4I|)%c`x$O@A*NL{+$p1&I8LubqEg
zCA#ydjB5-X8_h`WIHTB1ASQwBeCE+)Np6%Zft02}1Ed?-7)kGDPRvN$&K7fep&M__
zE7>{eXu!Ldb%IQc<B&Z`ZNcFFXC+ccAwch4mhD?<`E`x1kTPv4m@qvwHR;E58@{a5
z>E(X76c@$Nh1p;}IW&??(-d;BGaPrJ8H}NAQK(RV@FV!{;p7h6D<%d<c5gcl&Ss;$
z=DYOoIcg2Q;RvF2hLw?7*NV{AwORqTWDs2$f}-&sQwmi_2jZI<j-`**AQTg)S$Q(z
zroK$u_A=tat?+ZkX$U&I<IAG*R}0#tr7y`TgvpH)it2>bZfAw~x{3VN2co>hW==Qc
z^3SAKS?m5fvcZ}oVeDfNnd?-utTz>HyF#`uh0FT4m%`e{s82uT9yqJUisT&<u}8d@
zCyL+7@Lp4af|?eMv$nX%NZ&6DXd1`%Ls!FN#_yJ!F5p|!jGi~A%1kt`C(qnJGW`ti
z^TN-SN^i1YD8|nUFAM_aUUit>-kY*SWt({gZpYdvFk_@vdA8e*rYW^qudv!_@i|Jq
zJBLVOSsgecC%1f^&7EW6go5I@q%VmtTb^4@1so=h|3LN=!4zNn<?HLmiwwJi$iv~Q
zGa0in;6Xunx0@Z61()SdMXK-nwD-6fgur_9A{K>C#K3@HX=#bc@9n;skZqj{p++Da
zaCJ%1c-<ZYylda8wBvfna`<}jiXZ+s%f#z#3LJLUL$}b&yEr^Z1~R&O(?+zTT-U;#
z!ZPX22j00NF}z(rx!{B`i|r#=vz?c25@i&~UBP9}7XpNzY^Sre+)KZmk5-Gftv!HY
zP04B5=I>K$4hs@iYLtY0aL9S0dt1w_E+=dvhen;96u__QVe=iz3=I3M4p+BT3>j_U
zIHQ@$C`wy4g_Dm3I-fOHrHlz1!5qhoLsq*Gv6BQgaadMj-XIf^&5jTNrKU~bbD&74
z$O#raVyG)(RNtOs7O^!>Ssxda0!^T<l<&Paki4&`s6hVQ)yCS}$a#K^E<lo!sfu|}
zI+ODf9EK1qs?zXNbNokWw9HU}V+yA`=3sAr(?nwhE+qPzNDRp658q+kRF6vRc`z0o
zHM!->Nj#(wjwY$tH=dk3#ivTDTg^vofO#!K)psn*!ZzPY-h8MB_(Nk-g&|Qw7e=`d
zHa}JmUt)wSp^?+{sQp~EAaic$U9}HGJ;wtTwVmDzRl>B4)TO%f-sB5UXM_Ns7buZi
z+mS=|WklpARV1gNWf$@J;u4iaAS+%ZwhZ#&wCJ+AY7R#dqSW8aV5_k5lwbMmkt+EL
zFJJe9OdS{@ZqN^E@Y;VoR{~Pw;PFa-3sh+RESAf3c)MHNS=o4Z3)}8Q1fdYsIiIFk
zYSxrIxJn@_D$YtIzt7Tq;2S_kWQ%><rY`(o2^t7Cs$B@3i|B=ns?!9=#0Hcs#PY4w
z;%A-M<%WNXJkWp*#=rT^z3e%j^Je^Vf4S#7+8GCI9>C9z8Dv__1&i`!%|tq7kX}ko
zBu)~%oq>tMK@aW_3G|tQ++vk<p#mpc9{G29ictwE+d37>s=iRE6>Nn%Nz74n-w@2v
zD5pK0I^Q&UHvq<$74vL=ro3pgLJw(Qn0}x{6OAB;J;<Z6ow<d+<+?sQ^90Us@s<nR
zlyA_6-th71A8cXAv;}1_&yu}`na)h`Qf<&dO7b;*Ar=53HH41n4~3fz7Hy(eYWqAD
zumm=4K}H6loqf7qcvz(v1As9nNZ+k<Mp>86^T^9FPTSnG)ZP~ou@=eZK^Xyu5E#xH
z!{?bGH)>G%@8g&1oYYN-p}_l9WLB`NG!%m(rh?B((zsJ1V8dNegcBRJ(Yt48S(d6R
z|9BzFkDyA>6%lWUkG|E)5>ho5p-UP2HMER9$(-hxwvlUN?%SsI%g&DU+g)y%WY-G?
zK1=e8zRHP7B;PpC6Fnc<{uxi{K*o-PVuwkK9d#jb_fNze`Rm6iiVV$s{f*k~1lsRT
zp?-Y`w1LDSYp@x#`^qa8t}l&P3fcW#aMT2CIo2L?p^UJ3_T?f#)QLEIU(4$|^ltd|
zZmgzu!5~H+spI;%t$vjvHR*ii9b}ERR!lauw9_p#E-(AJM?nE}dlkzK)<|?)O3Ak%
zuH0I&FVZ$s&;Px7FmO1un|Xt2b!?bzsfpyN>ubIAygS(jI}nG8`yAd4re%o--lEa9
zK+wMMNb;V!$wI^PdEEO+%-unYoh5oP^o=^m!($VtNoVO$Ha52!AR|3@P{tv=d_x;o
z{jP>frAKX96ydNKB6q{oZ{|etVN-A%JSrYhl4lOMS?z*9k-q{4u;!X--C3aBfQgSE
zD-s5ut%lp#+h7%ik3yBDs`O^tGqH9DhC7fu9InHeKcH_#At!9zR*lF67Vb1z)QGiu
z@m(P^|4rA1Yx*bNUXtQkb~ZE58^XE%^9TJNx0#(qp&Q5B_O4RpyA{VeuX0?7+=b<G
zpMc*DSB}Lq$@(_;V6tw*qo222sHa{ixu7yk*#cDtdF5K$5507bl}pgd5-p1&&u06k
z`zUexXejX&!TUAV4t131s7EWxs$UK@fpe(qd#3`~S;0SxlBIjIa{>(p5bbz}A}rDP
zIcd>7ZY-qwVhWN}&QeCj2f+jdkEy;$ZVv|et4Xdln8!7H#9Zf%$t_lD1u=P#r?vJ#
zWxZ5K?%N13NQ*f>yU;?>h+Wfdgca_)%I1iS<1IFbS-g?kzLyREde;K683b#6!rqLt
zuCMbKR$1@UtTIs#iqkC8!fF65!uf{e`8^aT-hmDltYP<M=}5nd`$$;UrxlWck2$16
z9Dj$fNOlExkYB;|oPA&-4Q93Bi%KrEcyFhY#cFwXmyqQbwnc@FypS$k`KlAFL4dRE
zZU^$+k9PVdZpgiUL8u0_mDBL{=%|(49BH_3G;YC8bmSTY#4qWJW1TeUBCU33cO|zP
z%t`A7F!W04v|{h7sBm6TDoMqgigdm$F5LZt%I@T}&||V-rZwuskq9&gUDUEZJ>Q^5
z;~h9!hbTdmL|_o@3HsQCyQdo!u4L{2dt$4CEc6a6(C_qVNO45tO^<T1Gl$bZRI3=i
zcO+OZ^|0Z}EF^CqIJ)6ho4%E@*|8EWaFuFsP0`AxI~I$o^i+w@BcaViZ2H9rtF|%&
z6?wpOWTZc);n@qI!VKp|Ha&+Qn9<OWr_CJm6*YP~x6#<a<cFC`?FV^Z3)_Jsj(M5e
zP+F}M3S@ljPU}JzZ;IY*YXzytTY|~~+Pcbv)rAx)>gy-W_kki%pL2|%Bb2afeyvUo
zdJ}x@LpiYwHgiO6H2+OO?AvdbN@YQKhaR5$Xi&NDIjxn}#)%qr-ig&a&Qllcq0+G`
zZcugX2HD6H{NQ>x>Wt*IhsgRGVH>iHRb<M3AwXd;n7=NCUCXY77DubnMsH57h5pTS
z=1|e9rp+wMc5HP_d`;(lIltV%@2>hA3g8{GmBOjg!_Yd9y(;B`)i%S(s@O90Ej9y|
z8)kaO!g-&5d;wK=m{rF>wBe@S)igX$3t)8>V-pd;b&<#qw){P=xuM~z9D#*jV#OJ7
zF4`0!F0YLz=VW<{2Vo_vgWS?>=f>)PAu!f<2(HV_82;SHznsw@7e~;ie;{x*o}S3*
za~y4ESI!#o!Fw0uR-LlAzFS?|qVGc12cch3WvfbFNlovV1A6-K7C8$4xW4`L9Gvc9
zaMU#iTDW(#X|YYZVW<JB1fT~#*Nv*Ln0Kau;lQ4hpA{Lf)eXb9doK`AU0@3eYxf&f
z+64Wc18hFO;)BoSB&w5Gy53*OUitL0@lj2LL{#?%7;3{nO>c?UsaNEl>}89XRpK?6
ztmG^DV!e-f@|!dI6KY5R&p~N|QyJSHqw~f2wJ)=8N!?mdj|#(FV)9H1O*^tSgdru6
z@Et)oRNT2r2*{}o-jJHZgi+6X9~yq-&SPhDx=i!qdr;=;SCeh(JrXo9nTGiBJDaYo
zmt<sqpH_O&)r7FN6+2Tba_fU_77QXE0RSsY4nOr<i~<kS6%cgm=$N6mspwOU?tcBW
zB~OpC0}9spoTX<zVaK+bvmVZ3jU(SdmF^^0o2W<lFtOe3H=YeX<H*V1l65pXCS|vP
z5wdX_v30yq^cbFSyImSAawnhN0pwJKB=x<v;DmUcePdU6#!+o^fHO$G$I!BzRc^U-
zV`<wE3(oAgNlv7XF^Rg0BIxYzA%Irw3l_k%$goh?X5XrdXJ&5y3KT#+<||WmAbCC2
zWtH<})NK!@dGg$8k~QY($p_FOn^Un8Bkt$##GWvi>=O;LoIzn|(>EmWwq+ib-aq<U
zJ<k<U<`o!ZY9N1fQAr=~2{V8YI{0mvc<dF+;7Zx}J7RidY7U#6n@D`S311e1!ag{e
zbazFEW|#7wM*4`XJb;ZsrF9pauL{(9Ht24J`8py)DH0fBL+9sOU{q3W$BkmVR>*0m
zng5q%Bf>j7af*$D6n`!yWJBsY7U;%@kysg<JTp@4n9Q~tl9hEWsK~c43++}gnl_?Y
z5P<+6_i!557JsY$ajw(ERwR{VW+{!6=2xx;T7C*2e<IcjH56MJUm06@S;{Jr6blr*
zAy$J-D$xN&!2wm-t`@|iCn?CcdTbQGA*e#T^Q82-h#d+?ryfhX5-aD71aFD$-OVk|
zbIzEp)mF3nsp*Wivi4#(zE0=MhbuI>AOlX_s9bogvMNiLYc`zE8G+BxK@5N2Mx$XO
zbxI2t<BRZ;Yp9xIt5I>++NN0>Kbcg7qi5|fZ=H9|sr1e3g?5Q^3j=LT%A~7M!&dhJ
zc_R+T^eC}Obg!QV=pkHOFU9j3T1vi|IFB{l3NSk}qdb<uCWC_N6&!!4&75~C_^3dG
zXe&{#PImf?;xjSjs#UK`vafhD@H~s!KYSi7pAla^u`JHRXCO0s+hmqS^f0~k91{c_
znt}9y*457&&Fn<I*s&wBn!mpB^Q6~1E?>jEz5w&|%2#C8`HSfk=Tz?fIiG!{Dt{~^
zf%N9~3wkHF-_s>_>7Wz^$Gws@%kvYZ49tbTA9Ia%_SO2;_&nrMOvg|Z&T=E#x^-Vl
zV!)hEo-ZoEz;%>npD+5I)d0GFH=hOmZ7nzDcEu{}Y<%Rx@5AfI!@5^hV8x<I>7uc_
zvsI_DH!larmE)RD8q7Ox(?;8zuY^|PhE=v}=uCKbTW03?lA;vKb-iO&D-*eS*2`1Q
zVo78hjU~zH+#|(CPeg+kbzKfRij_+2$dm5u8D)>19eyDIYXsH@m73OZu4^c}=%Dl$
zH&1_+1S24$>kbHl-HF4pNnFNR!rDP@PPJ68<bi)XXP2?M`JH(L*+}-eG&?6*C6V6)
zaFT*J@_8S10RpQcS_+kI+R+1zT9~t}XhZ_#-g<j&fq-swyR<@(odWr8%5C=wjY$pR
zfVkJNsATDlzx#0;=7%*;a8OCCMa__+H6osO>bBr9rhUgF^;r`Z^<r)|udv)s>SF*M
zs{N%;B*A58|1Xi#M>~P#Cj#!kQTmR`7qn^_=K-&DoEg>iIwTr$k9(Hmd7+jW<KiAC
zhEuPSG?Zu=U9LHD%(h;gT93FJ=`-#wV{2a!>54^?<#M|3Tvk^kD0QNl;Yvz}iJe!+
zuFC7nRWxW^it04V3Z1X~kR|cobvyZmis6&ZE;}~fs<~}u%N|H4(wIcN@P^J>Hez!>
zUJo=oJzc}b<#dWJK5r)t@%uC~bOF+pz;C|CUGRS(hjj^fB);`I_Ct)WrY;ar!~t2!
zadVLJG(98R8QD%0KZpP1^x2G>Z9R76XleF=aBH##8|=9z?fSfEc4!Lv3KjTX{lKhy
zMqOPPi{mo=gUCp^kqFeumQF68Vglg~^?Di2D@t&6P}8=jXhGR}+0Qk-hL_qiz<O4P
zvN?;@U0zzIzLKigfUUPE%T%#CcawsGdk%l-EaUJb8j3_iynTNY#|i|%^&Wdma$!8s
zBXgDgE`&L=SPT^~(x`eCJkojnIIQ4{OTk+t_u8KT((1X{(q_10aQPM`(s;+Em$(B%
znH;;$fY+wMKp1`SgHl=H0M{{<d{Ye4nymEc=hQ?tPt)4W{9XVE#CG?`OE(WKX?d&m
zA7^16fKGk+0_y+gzW_#Zoy<un194*S<d*#TQ)6d9yu9lTKJp+-S_JkTo2v*8gcVOW
z)B~O5{#>A%LIc%4cUoqa!zl`sq2H>^XmvxWy|WjkXR~OVBqy>LXU%Ft(^OU-%vLmT
zvIbtk>)<hD12E>~Og6<Oct|ReZx^fJ078@tXN3Elxr1~@V%Nb$HF&M=9C<#E!|ZFi
zEX{57kY&^5VtA1vDK0PEyC+jygNlnpp5f;f36v#Ofwu*?PF`)<HuIY6F4Flq`+l1j
zDg7`P<gCVJ6%8J^Ur?2}z*s*y>(W@sJXpt+KQ7}~XkTQmSes5(uN&sPUn?=d+o)=w
z!T#q}O~C^s2XEb&`bg-xQ;iPMFrs(6t{1u<4GqwYsO3+PmkT=@%KJe#Jo+2R;QnH+
zr}_CtmI2Z2xBS*5ZLy|+7^lw_v2t=_NcTM26Ko^5biJds_X?`Z>;G(xy<463mG7U}
z_G@;rgBFV&s8o318hzbs@%5uOv!?EwZZj*=O1pQF<ci7rTo>Q)R8W-ysdC6y5wNk5
zTFKA4A#DS5!J9jVH@}Ak%69CS&K~;uW_!Ic!`b#(4|!fS*8%tBhR132KAQidisx%o
z&zpt++v?5*?>K2G(%j76t+sLcedC}Su#xVZ9WOK8IlbV^D}ig)W-eS+zcnu~Q^J3a
z1>@IO-CvvYo)-CKN$f96VkkG~efL%RdWmDI2Y8uUW1ItUacf+_$siYyc?|+P3OHsk
zJ#GzF((Ht+iz{(wYJ7YDmTa+JC1fQocy%3k@$7>pwF~XirE~4>?p$BXAu_`&e5w43
z*SG5xK1;25f9~`0OCg!%+k_oAZtnD|L0>;u!qvE7fx_84`&PHQPMv4;Is7I2D?PbV
zeLl<PT_F)6PK&&Pm)^bUS7!S*XOYl+=n_4+)#?5Rj`uBU=a)Zrak2Z)eGyw*!>?N`
z^$c-(G(pkXM)=D__sf^<<Gqhw-yMJ8<>cdMgP(ti&J>zDq2i_N(Yf>bIJ)Ncg(!(I
zp*rfp8qObrht65;D;8X!{$%6iQ@YER&t(Z(gt?eAMgzDAbx%N6&?(F#9$u(0eX8p5
zW734LO@=Lhyuiug^ZA@#hu5MEd*E8dZS&{0pgX~VBW40iit%jjO;Jk&P+bRFq4-T0
zxI*#wO-~O<T)_+mM%)2bPhdRa*5<hJ?;DjR=)vgpph-oc=@Dz>+8_-KBY}I-Iu_ih
zD-#MrTM*5tSl}om;P|m=O_(Nl-8#yO=>>mT{@GVwTxPOnccLQ$5O})!xvX<aXaWE^
C1Es0}

diff --git a/doc/ci/triggers/img/trigger_schedule_updated_next_run_at.png b/doc/ci/triggers/img/trigger_schedule_updated_next_run_at.png
deleted file mode 100644
index 71d08d04c374c6fb75284b735e6bf18706066333..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001

literal 21896
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

diff --git a/doc/user/project/settings/import_export.md b/doc/user/project/settings/import_export.md
index 7a4f9f408f190..58d2fd76c6102 100644
--- a/doc/user/project/settings/import_export.md
+++ b/doc/user/project/settings/import_export.md
@@ -27,7 +27,8 @@ with all their related data and be moved into a new GitLab instance.
 
 | GitLab version | Import/Export version |
 | -------- | -------- |
-| 8.17.0 to current  | 0.1.6    |
+| 9.2.0 to current | 0.1.7 |
+| 8.17.0   | 0.1.6    |
 | 8.13.0   | 0.1.5    |
 | 8.12.0   | 0.1.4    |
 | 8.10.3   | 0.1.3    |
diff --git a/lib/gitlab/import_export.rb b/lib/gitlab/import_export.rb
index 8b327cfc22617..27d5a9198b69d 100644
--- a/lib/gitlab/import_export.rb
+++ b/lib/gitlab/import_export.rb
@@ -3,7 +3,7 @@ module ImportExport
     extend self
 
     # For every version update, the version history in import_export.md has to be kept up to date.
-    VERSION = '0.1.6'.freeze
+    VERSION = '0.1.7'.freeze
     FILENAME_LIMIT = 50
 
     def export_path(relative_path:)
diff --git a/lib/gitlab/import_export/import_export.yml b/lib/gitlab/import_export/import_export.yml
index 5f757f99fb301..89088ee8762a6 100644
--- a/lib/gitlab/import_export/import_export.yml
+++ b/lib/gitlab/import_export/import_export.yml
@@ -39,8 +39,8 @@ project_tree:
       - :author
       - :events
     - :statuses
-  - triggers:
-    - :trigger_schedule
+  - :triggers
+  - :pipeline_schedules
   - :services
   - :hooks
   - protected_branches:
@@ -116,4 +116,4 @@ methods:
   merge_requests:
     - :diff_head_sha
   project:
-    - :description_html
\ No newline at end of file
+    - :description_html
diff --git a/lib/gitlab/import_export/relation_factory.rb b/lib/gitlab/import_export/relation_factory.rb
index 956763fa39927..19e23a4715f86 100644
--- a/lib/gitlab/import_export/relation_factory.rb
+++ b/lib/gitlab/import_export/relation_factory.rb
@@ -5,7 +5,7 @@ class RelationFactory
                     pipelines: 'Ci::Pipeline',
                     statuses: 'commit_status',
                     triggers: 'Ci::Trigger',
-                    trigger_schedule: 'Ci::TriggerSchedule',
+                    pipeline_schedules: 'Ci::PipelineSchedule',
                     builds: 'Ci::Build',
                     hooks: 'ProjectHook',
                     merge_access_levels: 'ProtectedBranch::MergeAccessLevel',
diff --git a/lib/gitlab/usage_data.rb b/lib/gitlab/usage_data.rb
index 6aca6db312300..14d8e925d0e48 100644
--- a/lib/gitlab/usage_data.rb
+++ b/lib/gitlab/usage_data.rb
@@ -23,6 +23,7 @@ def system_usage_data
             ci_pipelines: ::Ci::Pipeline.count,
             ci_runners: ::Ci::Runner.count,
             ci_triggers: ::Ci::Trigger.count,
+            ci_pipeline_schedules: ::Ci::PipelineSchedule.count,
             deploy_keys: DeployKey.count,
             deployments: Deployment.count,
             environments: Environment.count,
diff --git a/spec/controllers/projects/pipeline_schedules_controller_spec.rb b/spec/controllers/projects/pipeline_schedules_controller_spec.rb
new file mode 100644
index 0000000000000..f8f95dd9bc873
--- /dev/null
+++ b/spec/controllers/projects/pipeline_schedules_controller_spec.rb
@@ -0,0 +1,87 @@
+require 'spec_helper'
+
+describe Projects::PipelineSchedulesController do
+  set(:project) { create(:empty_project, :public) }
+  let!(:pipeline_schedule) { create(:ci_pipeline_schedule, project: project) }
+
+  describe 'GET #index' do
+    let(:scope) { nil }
+    let!(:inactive_pipeline_schedule) do
+      create(:ci_pipeline_schedule, :inactive, project: project)
+    end
+
+    it 'renders the index view' do
+      visit_pipelines_schedules
+
+      expect(response).to have_http_status(:ok)
+      expect(response).to render_template(:index)
+    end
+
+    context 'when the scope is set to active' do
+      let(:scope) { 'active' }
+
+      before do
+        visit_pipelines_schedules
+      end
+
+      it 'only shows active pipeline schedules' do
+        expect(response).to have_http_status(:ok)
+        expect(assigns(:schedules)).to include(pipeline_schedule)
+        expect(assigns(:schedules)).not_to include(inactive_pipeline_schedule)
+      end
+    end
+
+    def visit_pipelines_schedules
+      get :index, namespace_id: project.namespace.to_param, project_id: project, scope: scope
+    end
+  end
+
+  describe 'GET edit' do
+    let(:user) { create(:user) }
+
+    before do
+      project.add_master(user)
+
+      sign_in(user)
+    end
+
+    it 'loads the pipeline schedule' do
+      get :edit, namespace_id: project.namespace.to_param, project_id: project, id: pipeline_schedule.id
+
+      expect(response).to have_http_status(:ok)
+      expect(assigns(:schedule)).to eq(pipeline_schedule)
+    end
+  end
+
+  describe 'DELETE #destroy' do
+    set(:user) { create(:user) }
+
+    context 'when a developer makes the request' do
+      before do
+        project.add_developer(user)
+        sign_in(user)
+
+        delete :destroy, namespace_id: project.namespace.to_param, project_id: project, id: pipeline_schedule.id
+      end
+
+      it 'does not delete the pipeline schedule' do
+        expect(response).not_to have_http_status(:ok)
+      end
+    end
+
+    context 'when a master makes the request' do
+      before do
+        project.add_master(user)
+        sign_in(user)
+      end
+
+      it 'destroys the pipeline schedule' do
+        expect do
+          delete :destroy, namespace_id: project.namespace.to_param, project_id: project, id: pipeline_schedule.id
+        end.to change { project.pipeline_schedules.count }.by(-1)
+
+        expect(response).to have_http_status(302)
+      end
+    end
+  end
+end
diff --git a/spec/factories/ci/trigger_schedules.rb b/spec/factories/ci/pipeline_schedule.rb
similarity index 70%
rename from spec/factories/ci/trigger_schedules.rb
rename to spec/factories/ci/pipeline_schedule.rb
index 2390706fa415d..a716da46ac69c 100644
--- a/spec/factories/ci/trigger_schedules.rb
+++ b/spec/factories/ci/pipeline_schedule.rb
@@ -1,14 +1,11 @@
 FactoryGirl.define do
-  factory :ci_trigger_schedule, class: Ci::TriggerSchedule do
-    trigger factory: :ci_trigger_for_trigger_schedule
+  factory :ci_pipeline_schedule, class: Ci::PipelineSchedule do
     cron '0 1 * * *'
     cron_timezone Gitlab::Ci::CronParser::VALID_SYNTAX_SAMPLE_TIME_ZONE
     ref 'master'
     active true
-
-    after(:build) do |trigger_schedule, evaluator|
-      trigger_schedule.project ||= trigger_schedule.trigger.project
-    end
+    description "pipeline schedule"
+    project factory: :empty_project
 
     trait :nightly do
       cron '0 1 * * *'
@@ -24,5 +21,9 @@
       cron '0 1 22 * *'
       cron_timezone Gitlab::Ci::CronParser::VALID_SYNTAX_SAMPLE_TIME_ZONE
     end
+
+    trait :inactive do
+      active false
+    end
   end
 end
diff --git a/spec/features/projects/import_export/test_project_export.tar.gz b/spec/features/projects/import_export/test_project_export.tar.gz
index 399c1d478c5bf7e6e71905710c0c80e25f430a9f..4efd5a26a823ea7a93d6e1c1ce379a05b6171d47 100644
GIT binary patch
delta 210765
zcmV(pK=8lRzbS^yDSsb}2mnZD4_N>M<hgZJ)bG|eOiN04ihv?D9fBYY(jX|!%rHZD
zgCa;tcOy!PbcY}*C7{xs(jh6Jgg)c%ob#M>o^#*Nde^(w{l~lBS&KCT?7goYpX;;t
z{*Jf=!T<OR006+DP`<z3VSjrEfFOT+|NDy%2nGVcFc=sL1b^@WfFK|YNWcgEpLdOb
zz{AZQ<%)O37i)!ewM9ApcXD_!9Pa;O=kIa+@Av<Yyu>A(U7c*P7<X~BhohxE_J86G
zd_G_h$bVgbAP`@35Ex&702B;`0buxl0Dw>c1WdpO_@8j*|9t)by#6=&JVAU`Htu|X
zD?cR`3WFfwXn!;m2LPjx02CAmLV|G^m?ablgCh`dFaQY!BalcwS1is=!WxURbd$K{
zj&*atMG1wVz$hRP0Y?MT2p9ql!XTjds|*K2FlZ1Q1H-`pa1;iI{#R9bCk)o!UQEx)
z&BqZVh5$n;L0}*XfCHhRID{n(j6WI#g5qEhC<F&UL4Pb!c=xa<G!~5eS1U?3j&2xh
zteBjujS~)s#d-X(g@jWgAwUQcip0SY7y#4~0LEf~01y}p#UM~vEEIyZ1i>sVVHm)_
z+S0am@^Hn9>Df49ot-Sj;1DDw4243#@SKFjp)65IG@hLB0fB*F6c~+zgYejf;!}l0
z0sqwuK7XP=9?7HrG6JK70ze=H6oCPP&{z}*Z3%(HARssj4S)b(SOgFQ!9(8?g@ymC
zjfbvI9?sTc4_$qnt-YM^b|6R~B^--G0})V5OArWW3B^G1UI8EwI1*|J1EJvv{3iea
z1BU#YGgt>_cONmdE6NdLO$k6l&=@2Nf<!}MAb%|05*CiZAh8HA1PVl8Q7{Au0S6(`
z|5jlf7Uho53MC#F_;_(xEC>j;L?XbJC<Fonf#Z-E90Y@bVNm!_FqkFgztA<uI630%
zZ7}YX5Cjf@#zN4RKmZ14iN#>CFdPaC1>uk=EDZlg92|~@0dSCiGh*ZY{|cHodz8C7
z)_+mV(#F+|5{d-l0SG|B(O5J90ER-4I1Chr2LT>r5ID*bjln`tP!RTC4cIu~`Oy<A
zf%3w-IXPe{0a!}}K2R7OFNgnt&leaAM<elg1;WudC<<R-01|`vH%K|4T<t8Kyc{WE
z5I6#i&o2y#gIS_cFc=7f!dDQ5#UOxKD1RDh3CCeEU?A|{mEGL2u6QV;5th(DKtWg{
z{`>-i!GLf)5D*}|4jKc+g7I+1{@dmHb1lVyl=!faus>-=B5+V34h6MD;;#V^3&TM0
zkVK-uP!t>hL|Oi)Du5CLL_(kl7!C&kBj7M35&#FH(0FhF;27|q%wqvqC<^%Rw}0_(
zx<E=41PisqVbM4U2#W_75(WjqP#C=DKp+|e$LA4%g<#SDl`e=9hD5+|cvu4=7(7<6
za1aD$i60Np5HJP{#3Jz*5)4M5f&YarloAO?0W5Jy7(Qqa3V%EvW?&#13;@D#5PaMa
zyw3<62m<~OT@a<Er6mL(D-ec4f`5Tf1Rl&tC>9ODLGkw;hk!sJc&xyXAjE&Df+=xG
zBm#v&-~omNK_ECF6pO?F@zV?vg+CSx!vh3@hQpEnnPn&?3<|}U4ugT9kU#_&goi%_
z14mfmjiI4PJY_(k7(7Z*sQ*xfQKBIrG(Mvk?4SE=iNWHK5Inzu@V?_u!+!??w?yD$
z0{`c!a6Av-u$CAQ(h>{+AprP-ApUSG5D!f#<c~VOVpx3X|1(wuB^CrmqEQG8{x%{2
zXgps)pm02+;28*tMIm7z1fC5LI2`mpRFRZ_a)H2C6^yU_pE_X?AQTeMK$ai?1_pxR
znFt0(fpC96)g5K!CgBMb2Y-kIC=n<S3Wi04kWe%l2nV3?Z3TqS6CTA#tR(=8=VdGk
z2Zo^kT@$bS|NPpiD9S2OQfkY}tKmB%5&owqnbkY%W!$F+#Ikh3`^QXTJg0p9>P}yH
zU$7A^NUag(Th;f9?rP1wynl4+s9@fX8GocGJFZAFw4;I~ajiO;3V)fPLup#a$4R-l
zPo%$4eCZuu^xWbUNUE-WyW6X1iouliB`kx1C6~qY(RcF&h6a1}gOB-3a-CeIuubC`
zr6-=<HecQebZ_`(u}*nlWKQf%8y&aPosRUyrLxI<`j3AQ_1MOSqAL$x>)6%3!<d&`
zmXt1Fe=@01Y5gSs<bVGeL>(LE)1vtkSX&?HUkrxateBO{8~c@UEpWc;L*1!Q$l=8A
z$E!P{cKX|8ZGDtyJ$F2&_O5ac)RsRF!9fHrYl(jnpsi;60YEpu!=%@AkehEUU|jt=
zDfGUV=^6o6G#VZFESXuw(Y(2qb@S+{&-;XQS-3bt;$`BPuYU$NC|7YHUw+5-%Wz|R
zk4TmFdeQyD!5(<<-TD+{<!Ss)5&hZMOk~An0=z4)7U}B9m|gSlm+M(SDTLULnASeX
zXc}JS%6V<KwL#qrcq3dUs5d=C_3a?q2_jA`bK&!Z)INXyRy|ZoRj^=c3Ygsan=+F2
zGi#{$=&Ik_&wt0;e$rXi=^r{WC`TUR-YOY2{Ce<Wbe@*FH$2o&)jz~V>lyA>(~(iI
zw-(%NncnWk=h=5McP@($0?s)f$GJtN{k`@iKz!|eB3NkM;*vlACQ&dUamt=+c2D53
zve@WVXYpEGFdSw*czmL-FF)ByEp&_XE+>Ou**iTplYd>dvUr%q27~1O`)VE0hu+0Q
zcfP)5;Lzwm@7X6!zfN(mlOKx;Z%)(bd*&1Qpsrp+S5c_$#(vZh)9~vR1EG~6ELvjo
z;l;al>DUjD(mL(=F=Ep-##c5bF;8BtAJXS5N>pTW>}lVK_{1G%^wS)B*uNHZgC#?k
z_EXMaVSm9QcRzNkZ5_F<9!Ep1_cDjwW<CZYCuKeokCGWa)D5fc^nU3raq085j`9$c
z<r89D6yiNAeJQ`3+8O<D@r8|hLCRmnSH+{BX?{|;wQ(~<!ItNK7hfY|FxP$mM^_Ih
zM}o%3<K`A(Hc3rd+09B6?vJG9dwdalwCSyID}O{jGm#v|$@@}CS5*;)DX0qkJkn_P
zV?p^<lq^P3!Ck<$&(H=XOez8W^)ra&Zd%;S*g1jn+mzzW>6`I=)(=MBSl+a1tRH=d
zdKC&vDbTm)^zGz)l0$X<IGtw&A{7Lfb8$*aJ!lu)u*Zara9oKDE)?-loJU}!WY1hO
zCVw6@%d^{Lk9^)5l4YMInGYC%4FgYoqphA_z1Xl5aS2>DI56C@D<}Y-8i?1nx5Pzp
z*FSsq@Q41g<_o78A$PLOzk(llKn!G>-w-KpObdB<lue>gD(6*xN$@W5T?36f2%0E&
zai*#GtbNVrgH1BIw%x1RTUIZo?>IfBbbnPS9E=-EAI`AIy)LTxxCrusN0_w4SMX^*
zqW^We>Gq?@MD?KUf|!VNnKl@upq)FHk`wi<GH$;eQMKBgn1*0tb{22Puy>uFS*{<|
zN~R3H2g85Jv1Vy1`>G#*r|$n~W>j`gPOGaXY~a~nHr&?v_2qW9#oNatgiWU6YJYc!
z*6Y^O(|?$Z2z9jvnaohn2M;ujQXG0}C_i`TSgBj9ZFjf(S^C}MO#wyK0ISiGneYwi
ziNwFf|Iw-XrH6mgb$t9$!<0lf=oA;PD06!GaAStbFK(YRJW@95^8^M-%j_`vN`?73
zI`Q9Cr<OL_c*B0Lna<wQxho3?aeu=}pdSd8)K9Gr_ewuNhw-2&89<f7Aqj93@`xx#
z0AY8@kL`xKEw%CijL#{{#cRyvSHp!WM=xZ4T{eUvuFCq});%@?sJ*FgY8%?)YP>1R
zBJ4XM7r%n$E@=R}ca~r0k6@jK{M%6d8^CGNkAqSlrG^Ll+Q<01heXHu#(z7z<^+Zj
z)9nE0I6zDcIX*YX?~v=rm6;O~lJo9kSNJ-JF6DWBV0<7UYpR>Sqa!X{&d5Lhd36p2
z!5)IWW1H0SUytYjeVXGN9~&E*>*Jdq<o}51zCGPXKt7_bK1e+{7#$@h7Ck5?rYp4F
zQ<IVQID=Cw%lZZl?zS~2fq%ZM3vy#)9hY)De~nf6HYW!$0fwKoW|P$7-|Qo%rMe-4
z!&6gIpZJk@kEcQ70)pdw-H>U%al|KtaCdt<2s%Axk?SZ`6PWX}%U{HX59?U^+QGuo
zqP=@9>s4l^VQyTCdH!pAYZ8K55q8^c(m<iVAJP0*mWKQKM5p^i$A8BVP_a*4LxWwS
z)17>R!(&4TVdU@_qPuIVb85so*OMqSuqdSH-SGb<M>hyERM;ZY{s3e^_If8D-TmRG
zS{(~h)Bunmu!jd5AgdbTK*!k9azZJ-sO9g(-+y7rmm@?Fjl38HX69?i-E|RS;U1<O
zt#nCLekA+egAsX;f`80`Mm=em2$SJjyW6hJ`{HYRg}-@bhMrjL(QfobIrz7CGJMh@
z9$0>o3rTR9pfmCyAO`H#r7Wyf_@wFSu9STb`2b8g)Jv84HK{BR_s6+N4a!ac6WJM3
zvAuxL{QT(bepj7$xWH4{?wJuKI<1Nmaz`?p#f)g~Qz7N+)PM1P3ls^&cR$xh$iC%1
zt7@dX3oS6@(Xc`t0lgW!aq`^gw<&S<ymmRO9U*SyiUn59Q8m&9vMf(|Zu#t7IV7*d
zp4IJIBCRpp^iqu9S;(}pT%mS$N0X)7Cnrx~WZT4T9<obYL;-C0vb6sQaFK&&#^;2m
zhyTem!(W-s!GCA^kr7VOwhEtQLRNxvB<0%`BH%ydd25QQ@X$5Lz5hUVwtc8QAW}rE
zhQIEy?DS9o6+uTmh3^WH**}HD1GoNrVL=M_6`}yDe@e%PO#ipioz@i3*N6g;|B#l5
ztSX8>><Ujo^BMgY;-g^{b9iy<e~QQFgslCy;$v+T+keYM6915vNsPz)J;AT^hH{21
zl#I;8#3r4&sGG7sl+1*n;*f%Fl_-Gnf19q!3GC>iFhJk$e=IvopyJ|EfHlZx5y4+H
zXF`I$A&QiBq5#A{q-i4q3*x)GC{n)i%{=}uJXyF)$+1QR_^00?1%b=|z2B)6>udi^
zxI|=n&VRe<|6Y2WVmq7p&qSf2NerY`I*a$cR<li1)E}FuDiHj*>c0CpV^{dB=79PG
zp|wlPxFv$AlKp9as)l}QC6EwfSpu*>IHQIlA(Ur4o;f`(H%~UPvAf@m*pGU~+sX|W
z)Z-J$#h|B>PF7K>Cw<e8y6uAD>b3s$`RG6hKYvZ|#|*Y4FM(mUBC=J1uPratLY|hc
zP&>J_rc|um?HsLro;o@u&Hqg|JzLg;@%(jEsPeR$``zfW!{6oI**oEZwppX>^?D{v
zHdHHf{pK?-qP?HpyPrc<NY*e?XyPSeb}__@yn1}+*@BPF!BQ(>_%&<lINCq$Lk-Uk
zOn(GtSzo_taNMS)3<s8*u$lTizb3_ekyB(HRoGzVj7%IH3t_LdjLL{gPynF^RQ=eS
zn8%a#holo|OgX#MskE2}#~Wr06B+ErCfF@`kw4PpHj=9P1uetOhDsI&h^&l`NR%_q
zkNi*B>u)|KJ72GEi%tp?N^!YHIh+TcxPLpPB3SVFTbkY4Ge4KRCa;69^-{Wh9QC#B
zqD6G8hQ~2@XC=o-I^qXh1;1=`^TX@6{cDrxN%j)>O*Jd|AH5+^WqMl>I44c{<heVT
z);le}pIWMNgWkh}Ur2!a<bGaVAaxTPS*B3uRL)G%kx-nvtHIgS%^4LYIdur?yMMp4
zW*+yar1$%hx?=*Vl#6nlPhj4XpW|k~wT`@h-&5Txzrl5vF$?NEENka6;=!`~+i)9`
zBEfN^{`}1OXW+vhQ)VPF_4@^sy8^&t!UR=|1s<#qfUeYJpZd$Ni1ve9ey~0h!z-dR
zL1Vg9{KS$Rr=`{fpzGcbxw^hmbbrm|<8KIGZsE!6Lel+2%Na8J<~LiE{=Ry{v`bl`
zXA4d|>K_%_CEpBjKsIS|*6u2BVq>o-$ATzbJ0H(yu0Ky-Cz^?OHr<e7*9>K|VURoL
zChw%=5rnH>>`6$F`$*g*9l*LrP)wYdUDLfSX$K*#uzQ(I<A^l@Wl?=Tcz@q<Wt8gT
z$0ek`-&?6p+n!t<;nE{ESAADQzwjaDRjjR<FfEDn3YU^Ku&4RtZf)Fp^?k`@Bij0~
z^&zQl^1_E^Lmz&A`(C0Q%pp?C(rDQvbRJv~#=P-x<Pp@4VI-KiOwzfWH`{+3^weI`
zJ+F;0&((!9l?I=Lfj`7gcYlBRsV0zBTj*ILiGsU6>mKFatxsbt_9D=hwiEI_9@^HX
zh=7<)1RbLl!}S?2L8e(ZbZVrSQbNv>pRSV`wtOiujS15)XQZP2o(*$M>n_A*S}OD%
zd1qm}vUJEGd%3=2tERWoG@CKD0@I*wHT0P@%sMZGSV^I7(WX@Ymw(>25hVoE;ud;3
z{8nNcEJON|1MgaVn`Re)y65b{*>&7DLGe+q1}DW0n1#hximb5~C6k$(8Xipzx9(9?
zRW5a>Ts@UBcGVmV==A<l4p%5Oo}`WH=lsy)d0TQa(IRDKvluLx4OSV5D3j-rT+tT(
zqFb+$S8rKhFypuA`G59M*<3q2y=y5+`?JkXJiYPWy`qKh6BZ@WhYpIy*(oHv={WEC
zB^R;o+Ize=L-bZse~>U#jn;n)4)D{-y))0|y8ioJv6rSYS*rCZ<5;S&rP-I&ccFj{
z(rpzx)I`{}_AsEmg5betKq%vWO{`$S4)OaAIx@Z&IlI4B*MGHJf~l7g65)X>NOPx{
zx@m63mYWo~)=LUU$JL`$Z%NSMWPR-30jF(T&_>!Hsu&osAqLU~EH@XY`L##Y`bP1w
zHRSf*W;wM@O;%T-aqM`vmQ6OIW1<kH)_3@OiI)>>T`TM&z)7Sfq>C#{sIzYw9Z{t0
z6IBaG({MHrs(&aJ<wgms^e1SgnPxvpG0Tw!B_=7Aq$xB;tyFhhot|7=xwZLs?wA-?
z6wdcn*UHnnwbeG58m-?`DVtx7@9UuQRX7}a7t`^3PO*>uoP)vZEzLr##{oEJ3>+yV
z570eKVfZs)eA4K`PZ;~<U%6#f;xUpkDRjBXRu^s)M1Qk7Ur8NkY6;+bko(H^^rAD@
z5z&x58_xT>BHl6A$@17suix>G(v*zyw<xmuCde;Rbrd?%7ptWB7%s^<&Q+!>pVsa<
zMBmb3DFhCamF(#Ce9QTAQRZ1|>lY_p_g8w;qk3M&by9m-XBfk~=EuY*J?b<4?|8Q~
zjfA_q*MG-2e!YvNdDj{cH|v|5mOVU=sNCSL%~!G!_P7|ZYhE|*xG}VI1G+2XHx%;i
zZiM-vqGdzIU+^6Gv+66JJ-__mzLp|~HcSvq)@>3EDIeh-!Z5L4MhFfsR&Y@odKq|`
zZCqDhpp*5UKbt!b2=o=2g?TH!`&Lk!v2{Orsei+J=c6qw3yfeFjhIlqg&a2<NZ~gs
zCFgkW67!R<^z`g!wijWYzCoBK>AXj$8pNnFASrJ(K|~@=@XczYz4K>WXe?1hQN-f~
ziA46EJmWX;*I8<Bl7_glXF2i>3U63lF0%aW{Px@e4sIs@q{Rh#GmfygNINF_WT}gV
zU4L`s>+x$m8OeSsp{)#FMF>0|Ky-#q>WcLCrX6RPX%-H|VXB&s#g#EDC*r5ci79f@
z+<jI>ueqO{Y4y&pn!Ezdi(dwPAif|aC<<vx_zR!IHu%I43_a6u`5oe1t(8I|=+12D
zHTi;ndWupI>>hGj@!Rw@r=Yw5c=fDfCx4S15i_a)E_u5)eDlsVwd2(*uHLsW#7Gx7
z6Y4N>w~Hwpfp!-c6f9)KT$fc&>abh7k@zWLINwc1jB~E_S;e#84HdlJWd+~XR-6q}
zPU{PsUd>+IkT@Yv>5J{S=;)wQQaBzm@=d+ftU7q$bI0@)iVYmOzda+detkbyXMa8Y
zui3nrrt^=>3wn*p5%$xE%5pFC_nuh2Xje@_DB>(gH#ZFfqWkCo#NQIlUZxdz621%^
z(%lKt37{Cd6P5%7#Z_f9*yMGQGoI-Sy^1$ZesW}>9ARdIv$WE&dY(kR#&R=0pddR;
z)$wd&<7B~+_ML>Oj8`g-JM<SqK7T#+*I0fISAP9O=8|cpYhkYL*TRC5<lnnQF9cGn
z%0XU_Sj>lev=U$a$!)ak-#~Y_+vK%uir~O=-KO2vmUe3#S7et#TrY|5`6jjD)t8{(
ztuT7NYU#X8uxo#e<zPHe#g!$Iw_7M2BzhzL8f==GtXJ=K`ZN~aZPP*jQGZ$U2A{_y
zmvdB_HNbn@i1YH!-ho@^#_)izzf7EC?(aHB*5X-K>V*s85S{~6Ufr9~vbX)5ddIiI
zI(7Gm)Y{fZf@|yfuSrFg2RO5?R+|TwWU!i)X$$==xA(4h0;umV-zz*Iz3(}4`B;t|
zmd57$T)>0mdHvf)s=`7m3xB`!{K80blIl&lfx5_X+L&TN<J&WFU9d;^?php|=4EE7
zOUHXRrr<}YI2fmuVp@VaU_w6_zTuaereSs>Eiq)5y;R+~T+{vZUW?n)2>%VAxk(=>
zA`fogh(Q)lHu_I*R^;`6J^tjwwluToDeOa*94qJdzUz{JpF+X=6@R`DVh1$-+RpG^
zE)l_)LJ1i#CQTEJ2jJRh-HZpO-oDMwn7Nkst9ByvVHO3N^s+qxOj)eARpk3%If)bN
z_enGfj4vO!t*|hw+>}41KpRxETIcd#qqt!sQP=O;7XGs=+GDL@-O7LZ$Kku)#+PAV
z3zb>zkcT`;gk{O#OMkv7I{P-PZS=44uPZNSnkSCrp<h$0MUv0?7ijb{L*JEmNpI1~
zlV|+R(|>gAc&gipznK*NII5lyE(X)E(PPbC5I6a&W7pFizSj62ry(bym-^CqqF<Ap
z_eM^8QVWf4Pdm=U+SY`R1RE4Vk*JcOqUvD6#OHFWBe06SMSq{VVPZ05kyG=s>2S13
zxy_V^H0+Sg4(XPYJZcn}a!GJK;9CQA*>GD1P`aqaPONP#V5>E0{_Pm+p;T0DHSO3<
zS>IFveA(kQv{nJsq5D6XmAJ@)EkHKXZ^&<)_JoW&%a`=4&e;stax<K@w=KF8h>5t&
zaPiL`u1tC&=znDa6P&K~x!)#+$b^lsH#tHK40v9qcwm9IUAh!*1{H~f$>tuf7L-Pn
z&sxm9wdYLoWP8eIYJJ{6dBYJ)zUjbR-)|S|f-zR8DJq#z4!P}MU&n-Sg9}czxCu?2
zR7fwgT|8<WP+LVU9ur<^q=ouTSqs`>^GlZ>q}+VqzJHm%CF63aV53#ZmlMRC+5+L`
zj!DXRl3)4Uf0e?3c%8qB;p?L)x(kKC!tdA9N4dN;NgTF*b5c`@cDYS%@Q$i^3q&w!
z@e%2*`_B=3kq{?4B!v;DRQ2GBe5*`S3sUtu-CTHNEJKAqUt>T&YCXmxq7yb9#K)IR
zkjEFf7=P*Y*^~2q2I)^b!r$5k@9!|qKB@LGojP_w+nN+)ji&Na{-(?lKD>51(E-fF
z(?N2qnk{~WRXDq=xKAu^Y)&QDry*A|9z$bbzk15w!0WJ1P$NRQkoTaO-1&u|N`&S*
zvQI%bFFN^o!HaV3qcl^2`t^MAXOuUZ6{3<Nn12*lpO_&;$9qD`v>sAL^4797dEC1T
z*>-4NCwNuy{c6^FmiI}mMB4jq`@%G<AMF&wQS}bRM=<NR25*g7C-Q!OD7eD4G`EA4
zHoebl`u(r(dr)TH0)Elheg*f=DoYqh%(zb%wtxI+5GZViT@^N-e(0@wzgzYoumQ!!
z6Mqp9(DY-n_Y;rnN8w~?|M9lH^70BN_Mj*HH?YMo_`~Ca6VhHlG%OA26q6{KZiJJ&
z%H5*0Rv9Sfc~JbCF5(444ciS_hVi5aHk|{)Yl<NPKrDZ}H(6iZmmQXjja;WlLLL>p
zhLf9CPp`EJzH+|&LHx|pN;e5(^s4Pc_<su~!jHvWALtqAh_uetzwgRK4`h^zfPNh5
zG+`xqx?rJ!7(b&%L*q7NfboQ<qG^j<a}zq#n86m483y-{>XD*tOUrLz9e+d+z2Wg?
zQi`*Rd%|P7gEj5!>1d1VoSXjdz#flGLUM~`<A;-w`96s8*}Qf>?W0lK$?Ewne}CQl
z9oweK^7uUDb|}w^sb<I#oBN$YDg`E5DuSDaHIBPeQbS}itM6K2v3Er(1KgBL-V<&0
zU8JKDIjS~wv`jgvI*P+k<evNA>|>P)7F@nJ8pJe<54=(nHL$Ifev`l(e{~@pp<E~h
zJ5}~WX0Vj|oC+}6AN#yGTt}M`Re!e(5bRywVLWE-^a(na8VG%TkM?uIvDTr#f>1&?
zs#oSqhHD1PhiwZ;gAFp0Uoks**^)VygP>64rsV0Zih0-P4Bj84^J#p#nxjLGCXRQw
z=_Q)W9aG(ATvQgZ@RH~XF4^%Xn+C_60nBZUqCES_bAE-DXUgRss%l<-aDTVd@iV@*
zf<XO}hNYz807q8Vrp5~zaoD39a}J$*H0^MC_PYl~foti{5(x&`wkpJeYRg9wSndg^
zb}|q==P!ADq+7V9CiVVTWXcljb~v9^I^qx7RNXS~(<$vAi>#+5^^a5riKvp2bXv>y
z%>Wt*LlxTtLgVAV@7(MSGk<Y?P!L;jzO%ROn?+BA9goA_>)_B$WE-fC=euDLClvlX
z|6U_Mhp+*d+dYlGKPTk@u)kJdAW40MNu!g5(@5^CsQyEFfyde&ALy_F67Q|qYcmsS
zj>vA#)s?oe2A>@mpHuXn$sAZMELHQ|ayTZY1nGLHGLzEB#0ArN4}U2?Py7uL-#=;{
zTpm?6jM@1Ka3*d%F<G4ZX8v-KtDPA2bb7>Ub3|=P(*WT@{3v+;D_gW8y?+&u^T9cD
z`_@WRuN_ZxV6)w9Q<zre()*oPw-S5r{0i(58z3a9oI`|TGbSK#Q#<ft_{(7RnwOG$
zCbS_SzXIPzX@Tr_M}KF%taj_KVj6rKtS9<--ppo_y=DYs7ZP%nJfD<Kc#RLq$=91@
z8A+T6Mb=a&q|VfOz5g_jY*zhN_C#ROAxj_O2Q{N1(CTp)7VbngVF$LqRVF8g1b*sk
zYdX;t&D5(<9H%^V;Bbsl>?$7iaV<4`GwoAn4Dcf5GyT%?OMgs9`@tjYDK#do&)q~$
zARDHM{!IVJyg#G0=K+!8rsHyN#m>qYCB9Z8C)^>V&Q29COB9cH2G0&&8i(W!M8NOt
zaHmG`RW{{wo=Xj0+dj5tZj)4Jxr{F<3>gnD<R-0TcG?J--tI2SuzYLp`>1e#-t<(I
zr-hpH<N<ErTz}haN)U1ZFN_=Mkh4*`X(d6bbiv)ezn}DWNsqX<;kOXsl$Y>23F!G~
zX#4IB_aMXMdg73*@9(c!7_TSgYdNIc>u%V1M$dFf#I?`wkntDAskxOP=t9meDaAHe
zn2pJU6>ay%0^|=aZnV`fCbU}wy@<o~w*9*ME^>O%)_>Obr~a$G^3h&vVL}G}?(Cfx
zqHzfpyO^mDU56$gYed)h+p_?>_e)Sl6J$&6-+<Q*C6a}M#ZR9{t*G<Ktn}>w<6XCa
zQTg5C>u(|6MU;7wS_eHG1>;w@HK=|uwV^hC(OybcJa*Z0*1sa<qCM|Ax$(R`R6KYh
zNSaCMmw(pEk&sg)m(){zrduD_9}7@batPHpx8xkc8`=%F>$Ok7D%%76Tbk@zEEGh%
zZ;2(z17*z8T3JqLslM}0-YfNih$qzPZm>uN<^eS>=DtZAq)UwH+-(RTjld4(0iw=*
z(i8@UIiN!n^3aGx`e{i;CA(@FuCKC_g4PUr_kTjdDtX5?c_=RQhx>?Dq`X}A+?*~A
z9{KrE0V%(y1wX(hd>;>uViVg~BlNq~{L6jB6@+*_JJTAQVm4M<L|PpBYW^cp%x>g#
zo~l=np{bT?{)EV3b#v44<km^BYPMbfwZ3PmBxaS|=6zjoA5|$Tg;B`{Cyo86t^8aQ
z<$o2gv)ial-Mavs=kS^{%pz}Mtl`@et+2^xaUGOr^K7|XP(vpX<7xbkscD%^@YNQH
znu}kE%`-${gSR(|l$lWMr9Q;7GoUcM|8vMiy`8sHfMa1}w-?nD1Z(S#kFB5m07`FP
zL}Ibj;578%Ett5q32EM%@1jspr3u1{?SH)Ivl+I($L-a2!ewo{R@Dh_sDGPecFUzP
zd8szJHt;FKwh?g$bxo;${G_f_k*MDTnODEn_{+@C3OQ0!{$e+^KfSbf|5zB$Zgu=0
z-0Bk^Y7=6gzUD7kz_GRK@EqwbkB0rdmwH^C8_xf3qLx>ASgIOP3Sd_)R*BcV`F|jZ
z&yr!BX`;EQRIl{*l2SjQBqujoE}CDC{US=?wl3tUfSOK#K>wIZG+P2YpT^VAY6uM<
z(<7GbJ~4^oJ+2An%O2_F$_wIWJ;KW$0ez#q+DVlH7@h9BTv0>TB)&6`#+IMivMzp&
z=sj5b(DUY=x$LCpvZG@KwNo4PnSUF>Rb{Dnc8i>(or&?|gmzB{l6`9g2R})hy6VrR
z=|=>&t~{@7&J`w4kEDv0sH7%O%6yrF{U2ThhWfA*lk+X6bxZ+z5~nJAIeRk~>LviL
zy|vs|d`H%qTVZMq=@iC#d%pw6%Z8IPZ}|-@tW)cg1{CHgvo-APVDxs2nSWEPa82fS
zLh25NyJ=WBCQRLSHvKrmLmDk;tEj%%9WC^yPj2f`E-e=!i^smz*Ms#62<rtsnpoYu
zEEOUf`in=yfd@FOv()3cg!0UM(em!ACS;zVQV9#;RKZTqtQmFGV|by;-pB?q=e`{M
zz)Qi}2p+Q9#$_s*I~UFAZhzmoUVfE#jok!R)tg;g@{>!>R1`AxaF`_ieYtc+XRs4_
z)A72;>)w~YEj~kMmoao-XYWnTo{xQ<kJ&C#P$5_XUVoHY0`z>cdmg>yN2?2m-gtPI
zyE3s0Jn3_nK_XR+N19g%)vDKDR{5gf;jz}hK7oJi_hZ5*{p}_izkh{@EC6IBp>L3O
z33N)>?9Q}FiX7e%B8%P?QA-IQrk0`86Z0_&e?l8Hzx>~6?O)MG^kKBj{?6?eU#Uo`
zT~~m_jqQAVl&_@iPvg`yi@p~$`;S*_|7w8tXwz%=@r$ntvLCwiDJv*|%_4l{R#lRE
z?nfXD!tXPWHQMJW<bOZz0mWFkvPA@A6$4+#MTVCS%K5X9<|UG+2QDqVsSP!Xuk)CY
zmw9Wg5ysR&bIJ0obL-&RFO>+sfkJw*R{5A$hn-Q33fDCscU4>DV-zEQ>ow;;$Pf*b
z;Dt>_pYRxkl76eJdc=K}|9hPv+hyLe_DP76PXjfMRUs71H-D8QEU8HDKm7bWKG~em
zBx*LgkeyNNLk`4giAsI8-qwsA$2IQ%TDXjnfN(Iz7xHNQC%;bg&tdo4m{k4bj3(3_
zp451~BJcW~a<*Tx6&oM)io?LLTIL1A+|O$_0!Yy<dDgdT`%=+OJlE{^qlQ>Z^en!9
z%Vji(X;ZUDU4Op3*i5b)@qO6cP&u&}%hI7t02OqS7jW%+w#oUX)qXBazRsavLr5eg
zXCo+G@yk<5FRz9@L~8e|Gk$|b>~eoTHD;NiU3t;UW^$2;=Su2FtFcRraQb{Ji`@Qa
zlUGkFyX<I?xt@k<@G)~T5q*$^q$q7vLETSple2u2wSS{{3bjQ$+q#ickGteE7d7i%
zIqzvgM!eupWNO|&nSywIq|eh69W7DP(RY*}m?=QeBmIFoVRtHovY&0ej~*(9nM;=$
z%b%Xp3B@<-QXexl;o72UE^19maI%|2<_iQ*^umV%wb(cPzB%y|L_%&eQga$vs8n;o
z38oBnzkmJ+VK`zqt@xd*azNE@$XSr~A3!Sry`S%YLv*{e@%7TpO$RozG*eVk>eoWr
zX5?yu-kqmg#pWkFi}M<a!ehY-FX`pz<*M%M6!#S=F&mDobR4M5lJ7iBB5z!ZNTE@W
z%X+KO?vn5#LSXB*DswPY=K+)H%bdsgOv{gl2!9<jvmQB{RlY&rlUk;#_;IsPgQiWe
zZF%+Z+OIy!!Ox7Hk^6M2VAbp!I*QH$FP;;S-L-^#mq{KFH~a)b`~tWwi#~01fjKw3
zKfyH6vcAF(ZDqDS1GhGlzV)R0rXe_$gI^-(g>uQ;23<TbYmW%ZP)#VP{8G=fi+}kt
z+kd)-ThXgGqdEG=fkN;52}t5dsn8suHrP-M*`DJ91^C0#UZ=|mc@;$TTEm(e2K(M-
z3RWPZI*B!@_4<TM?IAgMs<8T_N+Zv1N5KFqiS}<<9;_{!TvJVa+9IR0YrVH83DsVy
zJW8N8v{SS`dKJ#Rc%7iQqX#i;xpwb-RDZ>kEGDU7puLsio=dZs&M08)%*fgJ{q?hb
zt$Jf#!Rdkb2Q2`aFPVft-jD{>7Tb8K*L}2(r-;}w?bg|C(TYUgc|_g3j|)@$Cb<8>
zr#9PjINs5Nw8g+z07SrRd-v#F-0ntdsT%F0lkzf0qmCp-Sqbh(2b&s<lFy2Hn16$8
zO7tgdkw`^o^JGi=J?8eSz+H^Kbbr^Q(u?v36$3?f%Bq4?xpl3-b~`TxrLw<d4$-<&
zECyxJ1*&4*X-_p7rHFr+BwEuQ`|O3t{GMebh<(H~;b3x*e{Yl{Ky2VaMwCxR%*wAV
z^I!CrL=q4VWcFV;Q^n(qEp-1%<bOOZ30i$q9jlAt>~cZTF}LJ5<pdhQa+ud9dp~mh
zX`lN)#u~<rY3(QiT{}hC=uK0YD6~}W4v#%%zzXC)Pk3`%A431pC^vwtE*}kG(T<AC
zMRBb>=D#f%k>CB8@fJua0k*tGVQ})0c(e1UDa2FNP;LJpAfu@)EUWUmzkizaviF}&
zR4}(2Q9iJ*9p|n(;bXe7{DReoM`{!EJpOmBxLtJ&?qDlVz4s(#$m?n2>0AN5Bhd!@
znzPzW%)~g@y(Dtqm5QmkRiv{7vEGZv+0@%_HYd#YcMj)oW+><FhkBU0A5g4ELFd7}
z_cW_$A?35w_X)O|88bssMt|wc8T&Out*5LX-de>-j`9b-ku6aCxeQtJehKZ#KfR86
zc>X@&q2KP@E>pYcVOmXevt2&@I-74BpCC<3mWbvcLrB7I%!OC8(;)q+gJGE|#?@MU
zEi+-eZnsG&fP%^}wZh7>0MYQs-&e-zrgZIC^<ZMP-Qx}&ch31?QGaq+hx1Uu$@tzQ
zlxKU(4VF5xC6RS*`RY(Tz-(b!BI6URQ>e~$R^>e#-Mga`CxfqN*)zo%&iyb8w6;s%
zCY2@o+y<VlI>&!L7B~qrW$TSMQx|i)xsX!Tgn$~{2E^+{is{k1nd;B8!z&!8hA*dz
zf)U)lgM@6RdG(FlMSt8!DbnL#60~BMZMSOEb#)#jJz+Z<?~@*I`|_#SV*DGdBIq9T
zfgH!zyi=m30o8s+{2p3Ui{A{$RSeB3)SF&$mlzZ28O2=>fPT7@AT@4t9=!kMH(Kxq
zDSszl#6KyGN$KoL#5a8`>BySS_YQs$HMV$o-+hcb_KhN4Tz_h;Iq$8SKB1C_dq-nW
zpLp!J(-df@_fe(*6K<#ICo3_$LYI;YceC9r)*K5~s?HL4eVx<D6Od0CanmduSb&PK
z6n&7>UKkO;4~b_GP(mm036r80xl716HcIWPC8dHr#vjJs{ATn+>}T*wFYk}Ummf;m
zddGiWV;Ta}u76kYb+VeR`W)nc36J?z=uwJu=vms#b5DG%w#mR+E>=DEJlg5kI)S6j
zC{F(;rRsEZuyHWZC7k@4%U~>i4+VCvXW)Bhmh|X7^J9B=Rc(DmD&>PI508s}moIwr
z*Ek;05Yx2}f-$C}--SnC>`ey(sntjC%nH;uF;<UBjDOEF`K92@ce|uMGv}oW*4psa
zV;px|uJd(HP$%mEOI-#Ib#AWdjyoE}7b2w?E^%%xVvBSAHrwH8Q_p>w{XV>h;F4aB
z?({DZdoOUh#7>4_%BI6#p{J|s%{HiGHDDW#Av-rsrO8`wpCaW_-!+*Sx!3KSkJ^}T
zTUHYt<bS`C@||Zk*=nr};k2IZzWDU<j6U|FsPNOn{Ws~WGt3ZgpSX@8#XO0YcXg~G
z<6A7*gk3@B(mt5T6MlhrwJV-m#u*jeKQyCnPOmgqTO_0lBn@7D{p@JtH@N5X^4|5t
zCsT2WG_Sv1(*zifK@_gVaGkoeoiwjp(5W68%zsSDdfB)YmKur`h-mo?X6V%=z40t2
z0jDVZ^de}(pf&=vXa)-o;S>Sg{lz&$k3T>}tf|`XVgxmm^fO{|FLv5~koyz9r{!3F
z1^WZEPYHOSg`8b6wWvpMggDZr_2(0Hx|1mQD+F>A(G0%6&G%M|SLp9REAji>;<5eR
zA%9yw>!YyQfo=@R^&0+x0$qLq`#XmWYl1w(AyMF95@*es%drmvtS-J5-48z>2cu<`
zo^)0{Ts=Tle+1lpE&4%<^kbbwCW~!vu~Ug;RHEA`EEcpc`9tg?xCRfkJcN+`J+;gg
zuRav}al-riqLzQ=wA{qpFTWE_UBsJ;1Anx5^R_Lq@?nS}b9&=xvg}`(#Ek>9<+hq%
z;<C!Gr&lB<AxJt1KTWaA9Ym(oVm!_%mt4;sBXCCUef2im*QGnE_rJ^^Kaqa9Ei+VV
zLq$MXYI0AXS<#e*QpXfiLKAjhheQ5WD3dGmiKgz0PbXh|6jz*m1QfeMq{kfwu77tY
zfo})cql-IMPHc_#Ce|Dq*-r+P8s~M`GxhG|8->kjbdBZ673whPEwa30o~?kckGH<&
z4q#H<k0f+aF+5$ns;BMj7^@2jo)pe&xpf@<M9o{eT?Q&sr}V}D%e1IP#1826^uqQ9
zfxX8%%Vb~iPG0QIDr1G*(w$#&H-9sn(HbAhwte=n^c<hFhd#Ov7>AB0L`CIAV2X(v
z-}&;qX})}Hdnlc(jM~jO_$KC(5y|6$K1ktysef;RcwFUXX3&$kF!JU;$y#}v0)~JP
zZN&%iO0yvh4kU@o>|eH?&rpczSe?3<(`9`dMA|8UJR_y|v>!8d1kGVQ;(x^Vr~)Jk
zMm{dpkTsQEm65a`+$eJ-_@VF8UsD)5QuyKOr@1X2UyxO%J1KvAeCwEbpXvVDF>{T5
z2->2(N5MfK;2nVhkV(Q8Xa=d54(?bmT%HJi%FQe<HrsNH*;9K|=)@jG-krtAqbFdO
zbW>l?vi#1I2e}O45;mouJbxar{q^*g;f=7Ws7?51w&i{k<va0$<sToEiasoU&8AcQ
zmMhNq)woz~|9hundb2-IZ|`BAUTYhoMKu2oTpUeU?G`N>aMCt#S2W!|4xuKcP-RHl
z92>6r4H0)35!YJ&;FkvT#CK8N(CepBAAptkrSp~H?3%|3xr@hf)qiFJ$psr8Tlsiw
zB`VHLzCWj4vTbto{q%5iMc$N^`$>S|B78pYwr0t>7o=pC={~`6%geM-RJks3=EbKD
zj=pZdc62m+peF11wNds@I%KFHxS%SUwlU?CE$aje@2s6D;g9bTyNBMqpN9?P*X4dL
zk3{o5cuH?n(*1O!jDI<syvjywzjLd}i}rA5#w`dztvn_@olan(8U$SU*`K-9h^Wg{
zXE5*A(-HR7FF(U=Ts!QR8do;-U5`ntp{euQD*xEzHxol>LnsxM^Jb_l+~Miiz1~Bv
z{d8~5KHvF~v)%W<(ygg{LMqZZkT$R%8sZ8*>)9$n=6z@afPdPn>Y6yg<I_9VF%D`a
zZiZbtm2-+g^qfkWgJIDH+_X+g`DT!+p)LO0mY6f^UkQiw_dggHD)P=pT6BbdYpN|b
zO}#+>6f!f>N{uda$8feB9lY+Z_LDDJQQa(Kp4J*kf&8Rgn%pt@+!01rZ*`?XREf)P
ztD|Okz)s3HIDZ;39F}n$=KbW7m9s<QWR|@<<Ns`yNd;0IKI(!+0RY79>-=}BiJLfQ
zbu3;huHpwX(dv(PlAEPGc+KbUzP~}jAb`rytja-sZ13wEi_*;>NW!_^!Kf+}CFD^D
z-E~m;m{TsCk+XJtT>DML;}rL3i{w6t7FcwLtYGHhh<^alh=_~<O|3)}eD``{ZFz<*
zf?=W4=?9}`sLX+ti|Vvb!lP7PZ8<Vscli_r1Orxwg!VjbX;r37WIGp2%JnR1tx-aR
zt&wo<eWM6?CiJt@^`#hySV{k_&6LS@=1Yhd@uSd}QuMc~soSP(-2N;8R<Z7}lTsYS
z@Zw@Os(;>tKA(AMip~|L-}e)Xh4BkO*~>YQhMhrLdEF_0^K9`OL!X$Pt*S2T$J<p}
zN$;F>t&d^%f*;kFV!xkD(VER`3=zDL*!%pAWS48r+x*tX+A#DzcF-It^pfiViYcLM
z_x-T7@LM&(X>)kE?une3#`sFOG66sCV)9prLVv4%sF@`89z$}b&Fps^=@Yo_h3C_6
zcJ4e;P48!ww7R_~{6h0rL%KYL16*`lZ?>IYb&q?OQ@i<b+9<3xGES6v8IycGR^i_K
zIwD@5YCQNUFj_QHyye}k6vJl>gby3ejW=41Yn;P6xjDssUd?>{nC-L?c@OQ*$Zlfp
zUVmrdmD$Rsu%78(TW0XtXrXc_TLz@wGG$wLR6e%GoTop|HFQHOW;KQS9k0`3@$rK2
z?%A-ja~=05b#b$-r$3@*-tJ9&!#B$?X~XAQG+D`4e>Kb6ck=z8N#3wAQn!aP3wEqT
zjJS9{J7>MwA<-+%b)5A#pw-|PO{KGA5q}&WXh%$q$@xV6+_I!358Zn)WI7A9r2bFe
zzl*f5eo&BF_~*NUyAIJnOGAFN5fd_oN)Gr{AaMDEA|D+~7z2*z26c!c8=dw!^s{ka
z9O;0Bx9AUl6$)9lpE{LG$iV|q!54w;<dCxn*Y^DxJDm^uDYYJ;_DFLU51^ULY=67c
z@U5)qXG>oY$BFJNEYq|qH4Fpc44)32<$z42X5W*o51(<crn)3{IiKr9L_mm*`ku#x
zQcvHT+QtKoeo{ks)roHHWyrdui~<XKDJWeB92#hD;D4olR6!taKG|Y+Y+D;=%w$7c
zcra6sJ%+t%GpOUWHW&j25D57}AAedJ8uoy6Pn(*38{dnvHbuA(=P{x;a&PI+`F@@o
zM2UFIEsPqctT96R!#EfTy32Z0C81#bd%3T=`zl-??C<#5RBfmk`#gBGJl`!EZ)E~u
zNDq~2yVzsVDLGPSsvxbj&o=t#RaiQ|zcu&x^i)9(exlP5tJV*{Emjfhr++ZV`EJ|5
zs?N0zSp1DlL@1NCFpqdCuqu^Zqq=pAq3zLlxc?i~seLcITc#<>$G^NKcJo?glWVZm
zbwP6{xW++v)NBE(8j#>&F>$qNW=?7AU7McG<Cj7XqBL5%;gX1nEthS_Ce`#V$>!M)
zY2&v(KY`T?RDP@Cktb;RUVro>58j1&>OTL(yY_pL8n=0NeoI_fO3L2XlZ#H(>5~_7
z7!_Z`<Xa|3t8*#}&HDuY&&<ilpV+?UC}~qA*cE5vQ`MboFMYv~utZL9E8LAsgG%mA
z$HA$D=5zChPkC)p|M#xR)Vu#ALRttb{7DFqtg4s-aC~78THt8v@PEvq*B;X1#kuwg
z7Cu}!Jga}!a^_?w7m!^{WDsm4Zdn{mn$=IKB=Dsx>1}SgtXN1!#+|ehZH;~w6<u)}
z3IP#zrephJG&#Ly$U`--z?Lgz_+948&pF0^7SKxZ+OjR;U7c>*Q>nOm!o>q8gwL-)
znFy-f4b>j;#X!ETgn!a4ZJl=V2s&3{?z=`uuX>pEhlhxwWI8KfzN3=rCGMDWu|loR
zzRDs77IF6Xvo0mUzbsGQc9T*j@p2)$l}q0G`0bwc&t&5D*N%u!+MCE5m<_V1jvR-Q
z^^lD){cqPYDw~!Fu5C_yj+!0>M5bfxS#vySkEWx<?5%PHiGMy_>8ho9`>tHwomshL
z1Iw#eo8Nx_dgl0R2jN(Pn@m{xW&ZKi#TTkkeI~=hiGp{`YgSKdKRUr1Zc7@z)zm3o
zS*kQKt56nfKy#!=5%1HAO&3l|YQfW$U<Cr(?<?DBrWIKFqxg-V?pmy;FYBdRBpJN<
zWxv*|V%e9qNPkA7xRUXBk?~gc5qeX(aNOvzO|OOI_WaPtbJOQuW=)R4BPXlfwepfR
z$8ThLXRpE|oIuz1g2nWGV$1a7#40gtH6a_Lwsjn&IlT5qwjVBrIPPit<a=Z(4JyV*
zEY>n=*~{oja1GT1M0rCzQzH@=RS#Xb&m?POy-;6Llz)y)gl5yy=94Y2n|!G8w#-TQ
zdoxRM@<MZ2<eJ$)2DPsAqC@k<qI!E7`x~Sn+Yr?q@?O%~+Y~%CQoMz+sn4ozo>4WN
zaRn#+qit%_g`Ax|n#1ZQxvt#+;OUGwh)?>yqNP_LRqj!1-%>?vqM>D?{+*WX^Wux|
zMdk9wa(@<sg>G5S&w~Y`=;AB8)F=}KvRohBaZZa|y&b_oIMygA7}6gVyWuvd79T(g
z)XjhQ+*_EXJ53E?7Q>KF{w0QBZq#jDuBFc=GZxV0F>VkG&f&gHdD6MHLVKCcjA+wX
zDgHrd^}ds;`Z<Cf^g^CoYRfGs;kQ8QxM8Z5^M8>}2Fl2;5!&Evm2T(p>>z59>;9`+
zQ?q1=A<>JnkLAwxjvCh&-U(;8a$V-Gc=ioc5VT6q<F`z%xr~eACJ$$#FG&;gR0Gml
z!?cjH8S484Zj&#d`r~Hc@i7y>ibXG{Vx#F$n}H5dtrp$ur**bPo@*5ab`ktqtG!CD
zlYc4VKR@f`W#5WD4r$4x;-_}KxT6%~|K7)kA$(zKKgd8ZaMS+r6At@YYOUv^QxxoI
zt@_=8&zuBvP|UmAxjPosC+ed-Cvg%N3mo6gd_UBf<=&y08?{wVQ+~gpTzC@HF;ZG*
zaa#B@YJtFn|AP!Da{Z}Kn>o$yTS6nlw|~}>e(^(P%_Cew)wKey<7%9B`$QOazhY*&
zIMNq$DI5WNa)&N#&8i#O+Zx3?%GEY=DQZ$Jjs0Qq6oNw5uImnKGPxo<4nGb8Dg;F&
zDxC68(}>xo>V458kkO?3_LXsV0|B8oxDV%cwQY|3e~J3ZH<EMb!Z#H6Xwy9#d4D8M
zkZtb@@pQ&|+;r^~5HNARC`V9AWJU5-;06Ujvy_M`sA$BUAzI>hg5m)c`2zdyKfiSP
zvti<=GbVPQwl|=n18oxSYhAaR$(pK|IY`Ea&HrV={PGUJV2bs`@hRw*GKzi_fUc^K
zCaJ^e`du80Gg8>Ql9&~f?gtqS`hTX;i>k2?cRfsvk1LrH$c^$am9>%Xqniil>5P&D
zh+pM6<`@sR7I>w-e&e{)OL)e8nf<JD7!NYThc8m{O3VH4>+nd?d;mcEC=&Pr0<W{)
z`=#ZosOMgb-!3t+_nr~TO9WT#i<OT1B7c1%oLj2TgZhQ?GzatijsQt~7k}_hR_atq
zSYl<ZgZH-*ZBj4N3{ywuG!*Yt2WcG)ooSp69tY|F4pj0mCLrgF?n!qrtk-*-%}pwD
z#$lMEe+?r6@Z=X%!|i(49hCVnk`LVL^&P(bY3V78^2mk45P?*#(9+kj^tRjv-Nm7^
zq3))>JH@^@{falIr(?&5;eWHj7q6$h2lxOiK-0fX-kdhIa3xLOkV%&c@Ugk8RDFyd
zUj3{-aXP%BQZIu1XrKOe+a>nRapzYSILaU3(&<@i*c`H`Jfq91=}Wehwc-9yCaXQt
zj#W%6^)Od0%U9@;69q5qupjnadHIK(%guE4wiJfKI8~odnd)~3HptrFus?ri?T+3u
z?VP@I)Qz(rm`yiV>Vge^6El|Z+G}|a|EW1E&>OtgGAx*Q`^(}Jv(}e{5BGaAWJ(@*
zWM^8v6Z^{KMLA2Nk|Aon>+*H(U5TI3Fi?^;Jz2A2Co-fsPQ9S=?247Z+!e})-!WH0
zE*cmDD8yXOXwWI+NGz#>XxM+apk#ZoBai?n#`gJ@tz7oV#c^9L<0TOR0gvO~cFvF0
zc*KSN^-QB**QjURAb~O{*PEHic?Jl5JqWua{j~9~DQ!)(4ScmRrZtNQZKdm%zJ-oY
zM)NP*Y`PIj^k<0%2V2OJD|Doy74Cm)a8b=+a{+x>z9I60p`))?pp$=yA^3LMIUz4)
zsi=hj!;%-I%;+sEyF(36_EnQ@$s>+Y&P$f3oga7bq{if*qSc8fwZg2yLpC?I4tg{_
z5x#R;U%u)Q^l%+uK>Q;2z=^_#VD_7$(a~Rho2T(JnzW;np?<4u-mZO+;YALLQ}lfe
zBLrpmL-A}@*}$Jvn{$7Rrh=W);X_&RC0Q?^ptiLII9M6s|5|lih=3@~!78Wtc5%w>
z&?czqRcm{MZaVlwSFeqdr=T<s@1?HKt@Y#Q89vqD*4Kq0r}q5I14QkNn8!tdmzK5t
zUvDjcVb|Rq|2k5(bI&OrdTUrH`u%yaWciQV8CNwj`4`u}nXP|5`CdhA0TrVJIZepg
zD@_7~i9)euUt5x}H~$wk?-=B3?`?~=d$n!bwr%6rwry**ZM%E5ZQHhObG6-l-glo<
z=YQ+g-c|SO{g$VaCp9xOGRH_|CfnJY_Z{=E)dhf=G8c5K+VQILEISu_S$XJE1av`)
zA}xQ3hnk|?vkZThC;}n@#bF5&@T>;Mv{t=C+{}+Qb;ZkSSUml!kSA=UOkNWOf*Qn>
z%UF#L3s(xNnY?wL6nW&Qqmz^sPV=b3<i_y@r4zTf&cwqTNIN$%?2GF8mFiQV3)z0o
zg!B_Lu_lSisx&hV+oCGH3XM&z$2y~>pqbgULR%l*?(l!n+pNyYit}=Z%#&xlzFQ13
zjQ&zMGn_Abav*8cer*g?z|t`jkbG*gBWuksEXc3T-ZrF95Z)y$wuJwjrThQyOrhd|
zzdI8pfgmK|{=7QJ5L4kMXow@HwFp^!|DHLi_`X+kmUdnm;}yr7Q2*gVB8#a&PK*>*
z1FMj1C#Qc=r9usmXz#mI1+rb4&lKW6bP}2g2{Z|&2v-J<NL2odJFyfvi6)=;l0gka
z*jXpCD0HY=U*pxsYMoW%iBEaC&v6e$F)Q#_!gHeHu*gDwmx#e4u&e@PwjU{s6CT#3
z%-~U=nP#)0R+{tMg9T+1AJzeg1jpMeJNV=uoP~e7`7SRpAg-nXDX41(fJX13$+bZV
z|BXs;ZLRo*-t$TSN20}J?zLIQ^5G7G8r%Lb=fLaNBd}U_APWLjHfv6~__8y&@@n(c
zI}S^(;*1W(b{tpwYE)_;L`G3DL==pec(YvCt1GPMnMZ`4-U78OZMS}V<^gbGTY0VJ
zY;Av$#4i(^q6iB13zBg69MCi>MW<83+B_GtnnywS@QLSdfas;z0lJ~4MJt<f$j38D
zTs0P!s~)44uHzfxnPT)E&Nl;&yNR5V4?2cE2G81E*S0{eD*BN}mhn+jx57V4Balot
zdCxnswLFti=B=Tv>BeeQ+Ss;B+IMEd%us({UB=0x4>oJU2q8g-Q4+>ESs2?@Rj-z4
zt*ZU8e2LZf*9PR@oo!lJ5_U)o9Hpv<G};|<5)Ug1D^&ikjCs~1jhso1>YK<4Zc^Tr
zx}C<37p0L8jnTQCE$$*-x28Ty^WIF<Int-ee-l2YY5Z%ylpgy_Axg|Y<v{XLgrR?E
zGrbULd+RFb4L!?$ffH6tWF|T`?#4ZgPWFaF`2jz?<FIAG{e!kCCU_7c#E|`5l_~Ho
zAW-@wOk=}Cv?~qZG!vvC<NaK{M7l2j<B!A_=O`q}>jZKq3zY_OGF5JHzUTFAZ7#B=
z{%EaNj@jn^b`!fdQ1zdx>Kf-G4BUT#(82e?#qQTG*jBCB{P;P}(y@MdZYr_W@4eU3
z>^5Fz^WMG{H2qCp@x3W1k8H82?tkxlhVJvJJGv2NKT_QJqe9m9&C^-RTNM+L!3ldm
ziQ{JXq?x}adf#aXLSDn+Jey=+f*S3Hj8G9wW$4HhRu_s8Dm|wd$}-J}q4Iz9YfB}+
zPDik8QJ;E-h3Truk0wn*VEl-9e%^75@h~)e!{%#Q@hHO~W!DLz9?ZaO#tcJ7yfL#5
zK>uq@{lB@_{`)zlZQH>*e<UTN9sgqCZozvIj!26{NPRr6xHx<~!2P~cfQO4-Oa)1N
zx<YC`*M%jAzrYN-po8OgP!NBwx0oD4oZvhm$0mDqX-ImV973zXS?%h}DN&w$-(w%N
zTvMv~-BpxuFJqU-;<8~GG*S=*N-aucr*o>aQ&-#DoX28V!0wUF!pg|V!lX1OGfO)z
zJ3GlBZQ9m5;rC8;$PKPki)DS{Brw}e^Sfjm@eefxlN1M+N-Z--m=J%J-A})+*r~5>
zjcE76qvz68QoR#H;>6Zx@<R5KQRs{M=B!P*CV~B{dBv{nFv4VreGeBb;N)um?DXd1
z@OB1q3wy(<b8F=Ja>f2$Y|)o}+G1fBPUYwp1NrD275m`)v?eWUz@e^y&d2;HT8v;a
z?Q?8eW2APOPy<0FGcSKTGpP_KTVE$5t;*WaW+EPMf7P8{*8YcJei9GYywUu#KIa>{
zVvUZamXnJFuv$*ZB0xvSR*hOTynvgB{<5^yG?lH|T`>g`+S5&yaN=~1Gi?;Z#-03e
zj4P0{8bJMU=i>GR<7WD6O*M(Se)c##*8ZcZN&=smp@jTRFy()h+v`1~?VanLt26ME
z_30$6c-Fn~eY%^MugQ7z%Qz?Ql{=prH1IP=Hz-J`;yDey41ZtB>-qP(geJ`RVaBBM
zi{pT*=}{Vb@dbqWNiq6S=~-r$#=(O`d}U!2!`gxVC0Qpw!_4+AryOZ@-^pG|<`7JP
zqFA3I7<imB@tS`ii_b|2>Tw#n$yq2W0No_{6f`AE1N%UQ-~=1>>il+;u`$Ot>L6mB
z1EUcO=jXHLZfE9=xG(}3k+SH7?6lP6q?koj?`vJPj=j6Y_HA%^qmtZGJQzGvIN9H*
zAbtN&Klx?&AzJFLb}d-xZkt)(imM17bza^q;=SLZ6UKkGl)1hA2OjOobXR$k;!oEf
zEDh|;)Ws;|q(&^|181Mi?~T!VUm8}&W9ZYGXc*nliW*MP^af1omo6`Mb89Mlt9SC_
zv5AZi^$D9DD1dslX8#`J?&nxo_!31gn%9JwmH-s}!reSL&}s=_wHW<>W0ZppASok9
zOGPKW%(j0-Q%-4k_WoVCpDi}=N%F9W|NG|GZtv~iu<Ot==$Jm}VR^^OWsaQXI890|
zZ>IB=Nw-yXo_uJQdzl4hntX_R`|{H!hOEAR=BTVjo0zW#YhmD#vJ1i!Pd=XaIWh}J
z+9(U<w~ix>5>FiK72xHJjC4v9^iyq(bgjUY>#Kj)AGd+?s`V!pbaNBAmQGs~Z3iGL
z9VJUgM>C^bJtbrj&fj@gz2dV$7%=J3Dd070-<w95Pyj%pmJSGexwMa#^&(M5M8Cai
zY<1WBy34{w#FQdiT-GK&$F<Ma7O>iD#`L7su~4ZK!iQ`LkXtvFEBCY^CNzyxBv#cD
zyc>T@x5EyyW|g{Yt<~v3d<g6Y5M)3cdm;QkeqsUw35ibJ!VL{!@Y8YUTm1GYmUbSJ
zQ@Oqtz^K|Z1xcNml9N@SGjm|e*-NfV5K%aicz+T5=75io-TuHVAgoOPL#!dl`_P!w
zcCk%DoEkRM0mWO>0hKT%MPtUPLP_~>DgJ+_;AXsBETM6r<7a%CNzlX@C8|N?e)0kH
zTYQXk=9~NUww6my?qnI~<jg&lS}++kg&e?4*?W(s)9_+8&}k4-SgO)$_uWac-|{CQ
zAR{6|*i+n0%F0p5$cb9Swjb0qOjiO#fGwA2mMtjBx8o(o7l=aIAg~DJ|HdjtHpYLs
zNy_nQL(Y9Z`++3ixP^OTcJn5=w)LaUyEi<blL$nh0_guVXJM>oWo(M(U}B+MnWUC9
z-{agh`hA!uD`<Hg7V>Pavp`yE+iw3$DJv;2w>B#|DLuPdJw9z_kBv@;BRgS`)_wl1
z3@!3GLz{wLE4r$Im!y#tt;#8Ghhu-F7=88RD;vl6Qe9Z*UZDg-A)k8E`6+U80n0@7
za?S0GKZjeuP+xj3fXwhzi#ZwCY?z<31pNP6CP_;<HZChqJ3ckLI7w~xx1nkKM_BRI
z>8h%=)_9_aLgrrb5=U5-{J&m_075(eXYCdjmoN`O^2};B1SGLE|GkgkYsr63dy|Ic
z00m-TCRnWakC(=zXkup-CZy%)=O<}r%=ei7X8Lpt3+Br?zvKcgT4Qq9%n5*iNO2J~
zYX9>SCAD0|5*>Z{l+>~u!?di_@}!($ovzo>O-)?$$z|$=$B^4?xLL0r5d=_jA96<x
z@c#i_(GYQHB^Ac=+74=7GSh##MK;$aR{xeoJtt6!4~)?NWA6ToXXNGR3_RFh=C9;1
zYPr1Kt!rrFIvmfW6#j|=ed{9H7g7KJk(K}#xtUq18A%oDDVlwAtsG<vI`qDUnpO4U
zVNSi-houkPil+RzGc|twHL$>0R?1}0?YQ%Iwt1;fV<IQROQZy4$ohX03cZNx0uln|
z>hU=$_I?`gJ{*<1Ta$|-JnmZpD{tYopcTx7W4r5YoR~#czbykUkaCO6f6{!Do8x(N
z+j-pyC{zLen?uU}2M+0f8y?3h@0f`If1~O+F@s%4^7B0%czpgr>fY*iWcMB+*gUVz
zmmbs!3gr9x!Zd4Xv{!%PW%@WVVmCg_+hyC?<kB}`{ddNfx{f}Rr@CB1Cn^Q9pjS#(
zo9KSGX?ZV`PKVbCs)QpeYM`<3vaXrnId%=)dtavPJ>smwu65Xzi5x#XU<vQoYuyy^
zoL^nr_67}iWpJdRU*vJHg35@fh<GU(IkAgK`@||N>VI&y*oS|R*XZnTr8xeXR9r?g
zFFHA5RK4+hvYA;?-dTE-n@C7ud8tia??U;9W*P(g4SU`a5Rl3h{AeDNqniAXb&3uP
zVSs>%p$&@vgTh}<t}b`3{{`+`ZSHvq!{ReP!^LKE(>$;Y${b&Kps`fg!~c1qdi-&t
zck}ehD;JiMxln)Jg+2W&GcE{4Oc{S`Z_gkT6!RK?v$QNPhrR2caZa__JCddA@-d2|
zJ2mY=9-IbZ+#y&b^vDMN_l-^95|IB6>kg+KkEe&m4_E}N1l94!9wUc;0X5J-ur~<f
z<xu~z>i_T|jkM9<*(x<{Lv!xm6{in;bTW3`+7s``@kD>303kmiD||`0oSfs!oUCUa
zZEsy#bo$fXQ^a&;$chq5Hoj(%qEPz*dAj*-Q;fKYw&R&jXHT`*v={_cIZZ1iM>9V?
zA2U5Ie#vv~&~^#pnrpUmziz5oukx`w_1@@L;89J!wExzzWk!1l*JiI|CaoXWW}`DV
z=hyEy^+11vim*ym5dW{{G42ltk?yd+3|>!PpQ0z4ACE9wi${KQy{Oxxc^a4)0f9kB
zfM>vtHi%sefeM@Rxyn<GDNU`IM(T7U5o;%>=V&ErWa~~j*l3NUtS++<qebna$)?F{
zeEc<=MgwANrEV(+`@bUMKNd>P>9~Cu?Uc^kA1;4qwaV&`0Gu8q;)q25A#y7vCdrue
zk1kV$n^m7!_Ko`|ipkUMq?3*+NFbsGLZwph|A-Q4*%^Aea0Y2Q`f&#7fBS87p*emy
z1_keLz=HI9oZ{RmYh-zABNBDAGcwbYveGlsW7a4RxQiiK_v7C{5lzef1oWmH>gtW!
z0wI5lk%<?<{s$4xP_YRAc55ox){}TWx9eGxeaSAZI@|WQ;w0o<3I&u<aqM3r)ObA@
zbzNn-j#IB-mDplk-uZU!@6bg&pV}IF{3q@~?GJ3=S5Y6=s&Q2DKblMsS4)3*BZ6xV
z!=m+r9AxC>3;}NOk?MI}R^;{^m*v7u?2Lbu+@$?H@dtN*WelzBpMHL^m6nfpc@5%z
z%_xJ1Cl)LeNr%dXP&^pH@@XB5V(2^gw*4+FMe1b&(sWRA`tLF7G1KB>`jn=AxBFv<
zqL%#fX87{>$siJ55%?Ln#2cS^CjJ#Sg8q*1yEc3CzWi``S}e(RR&W`$xTI9ge@TDm
zN}yi<beey7+uJxz&g*G#=1ejR{QnzBH?c)73L*bPa$^jV(~JLF^yvL{G85sn`|g9W
zU7<_$Z96@Kp&5$ECRTo4jz-9a@u6_F>e;B^__DKRnxywt$HKi}1`JS1)l$vQ$iv9X
zNJ~Xf&U_KAd78SKWe3?L-#X<5;8uS>wkHw*X)ZEX=l}YjMM%6sp84F5^TWa@tqK^b
z^o0>^>yZnl@Y2aq>FFi;F)6wECE8Iji-#P{DxCQd<D{+&zg0+q@5xF;6go*I?d%w>
z6nN!U;Y*BT`EawRz^VA|dkSLOXVUpF3XtzBLq*YlqSx5C!tAQtgp};e3eA6*jQQV8
z^ctL*F~ii(GoK}Bv5%>06bw3cQDx2C1kFTfwRVYHoI~X(>j%G?n4TvZBKkLq#V87(
zv2P-qdIXPbJ_V_NuQzvnP$gg!yn+bq0aH54F_{cH(o9@hA~|5kn9WB&U_)i!{&5dP
zfJHr<5d2fkWPzF8t0lkJ%&~t%+hIZ`NN~zb4x!T)4Y-p~$(aMzvg3(HYQ2#_J_I$$
zGDaTgaXajqRK-|fWCFQ4-c5GysgTXDa=v?JwGe(80zD->?=7|LX2MdW!^lvKOb;ey
zSc%P%TAxw7)oBUaiz|H?uVC&1PMcG*G4p5i$o?36&2}7c27*xsRc?RzX8?d$-G2uV
zwCfzHmoIr(pC<Ihr1D3*lvM2M*(D3&V7k6nV<w4UAGWow<I<E6xYsF|lX)bwI5HNN
z2)a#uM2`4WERL~>3oV^D>RmbfQ@8&&n1VPW)&>kWYi{$O<3zV&#BH$w<qPS@GQ+ie
zJa_&3)sB7-67~>2oy~u=_3PSFSNcmSIlcFMzej4WY1KoNPvGKdnbsaek8IbNY3N`^
z*x=b`sz~=~-M>;pB9#l_^H#t6MmZ>9G`G5gr%agjF&cHh(k&NEvu7LBhwYFr)Ebq=
z;D;)+R6zSs8ml3Dg&8+JKq#!(0TkRCN%erC2+3rrkHtu;>$`u>mxz5}=AXFzK&(`b
z!<OHSujH612=GUva_vBHv%1Qr{{IBZOJk>uJrE#pP~dVif7E}!wfZ>?!GR!WL7aM$
zU_7#?buR62CEUJy6_hl<HKEDFibTC_i6w=ll4Z4<+HfZ($bW_Bo*&3&_&Td|l+V|8
ziR&>aB;M*=EQo(7uYvtSB*e*BTz2I<<=Jgl-Sy@8@N@ecCb0ESf#sXcY&JW&_|AB>
z`OCm=W2dXpRmFp2rQ~l3BZlusmM*-R-3L!=li8`r@lCc^XUm(i>ht8oS`-~#heHZm
zlq2TG2y8W1>(_+d#;(LAq)JB?j-4+PJ#{N<We*;Yy_<g@HLupsnp%pr+BZInrc7-u
zjW#Q$wQbbG^G@5y=K}oA2whbFe#&Bp0~&W*U$B?8wDk_gN!K-=o|D?U2cQ{7g|!&V
z%X<2vgdOvEW$`G|tle&&PP{9{v;7|#JCoU8<Ar5tZif*<pfjJY;8ESs{m8A~`eL@u
zE^=F^Vby<*jW<%-<B5agZJ+&8Slnw>HUC^~JB7#X1wid?op#0zdb4}pz5;)FUGZh#
z#mB|00;???6jONEB#2DWd~U(4>c8!--sG{q(J|!D4S|Ozxb=?(xuiuO+1?AsaMg}|
zlv&k=6?pyIG1gx^taXJa-xrPVee<=kwE>s;V2FPPXkOQBH{_#_CSXtT^|M(SoWr}l
z)z|-dHS}}!jrn=y6zKjuKKguo_Y%LIDB0*4kvE9@toJm&wOMvM>-_^aWVa-yQdM=c
zJclu<&AZjFvpo$#(Ud^ZaB1$kx_GbOO7_*;LE~_N*By&*nOAR|s@+2}&pfQlgxWo2
z)L4IEIp63PAa_+^WGUku_mtkOT}|_8pZ;}1QCN3X;E1eO)ioRr=yaNtTHIkroMViF
z$Z15(*2h)|6jp&X4>nZjYJDo8p6WQ8BwJBFkQJTIEJ#fiXz%gwqL8B9JeH){>wO%u
zizJgPV-;Jy_MTU)oz;-{IqaB$AMv}<$=iS7wZ|wbA()L$0{^>`p2NXc^o(XQBm$c&
zQLd$Weeb3(hQKht5ZT~1ea93L`I}vr^b$hgHL0wvyk6(C-3@QNn15g^fOL?&`!(Sx
z_8a8yBk9o&9itp~u^X<(CQhPhszR7J3I5{^47-1mCK=mRaaGfq{p54cJc{t7KiPjJ
zNR%U)V4=JF>sN`rHA_|X<r^X<vqsi@^XJvXc!u$8C6K6v2%(XkKUzA@BmfZq3Kq@7
z%lS%v6?Ap=rU6$o_+FKdNd;a_du<r91H|ll4VH<eeY*-q0uMR-``zsdaKb=GQ^G|O
zB4s{6!_~XRU4$YDWXo?B>u0eL6;ppcm|S6G$V7&n?EP>)ER%bMm-+`rAQNv&4?5K+
zODLf)?XG1&lXhd4*-@;)t3a#ki34n4BZ_b#tLOUliZG{;V%$-i##ku0t_GWCvI^?n
zz{`e_x|TW<e2F%r+*EtN`S_P=?KTFNfEPxc<fb%GIDqcGg)+?rh;1prgjs)OuLMt0
zeAwsj?d^OfQ(kOT%J{iUHx&mWrO+HB)Z_m6UJD-F?1-eT(blQ{Q7~lUGM=tUI|H%8
zj}l<k^UEH(SFcH9p9RY<@8wzR1^xCNo`g|)FZgsAqD`%X_*%jrUyTC^x^+n^<&U$S
z?*?j7+z!1&*=d8VJzgqg2&sR_o|H#19_OGSUqiudVJ!}%#Yv3WG+}%%eJJqHm=Nd~
zq36kHWo1-Z9%lEVUnF23JA3GD{p}o|@cINItVyO4a%N&JJ@|Ni99FOSde2an`$0LZ
zsS7R^nD&MNR_GFs(8XACj+$GPodi@Cjds-L?s&N7?Vv;V{gOsh41j+i*rl^!*Oz`g
z0k0S1Z}E8fh6@TY2Jdg~H^^n^^Q(|fO~aWD<QSjnYzcrsLTKiz9VV5n9qz11CkB_Y
zT?^B{h0e!{R<L})R`x>(R`^=|id%fNLtm$|674H?$OLxiW5LX=S%6M~VICms1$to8
z%QMg(9zoIqHocZP876<As|7i|mMGmRCcP=}_Sxh`L?dSEruSsLn3-h7v>^U1uNrW6
zjl>vXT50<|lxQ&mVYJNwyOjP(Cr-xVWK6$8Vz>^TJjG7KL{}UVy`NlEiIH)$`$C6k
zWN0XWTesR|06w|nI{3ZTua<@mE27UcrKcTTNf@zCUC_R3h3bETOuY4qvjzq+QYgBP
zCWu-k?3`BjP6Y(JL{f^*QybM|)!k0%1qAru%~Ap`(>9h##J&GS&O+tl=ON#iHd;kn
zj0mrGj;VCi!h6R2K$wio5f`W-3fdd?|84(SqSuu2$OW0RqiyrdJ8wm<Z}aGO0!nFO
zT(d$K9k?J|RcwD>=Di+P+m1>BSl)q&8G+&C-4Bm*%G>)8e4fU(IL&B+XkTJ$SkFjj
z$=oH3dPPg02t$<*k2~>`OU2=lNj3`^<^dV05Jo7zeB06%@$Z!o2OEQwEGoyx!}T>a
zEgib6iiS7q*3o9p=-_SX<h=Z<yB}Tg2ElZ=0&W6i*q(ooXw%c~Q4WCirxAT(Z=`a9
z9dh7Ug%5_`NvD7LTA_i9gVB~IasIAqPVG)L6c-FG!{7+2s=GZ8eQ~f>-B+FU6iF$P
zFXPdMEWo%!y@9EvD@~Sv@yYuJxBYr$aP+-mW(p<KMqe}JEEo0fZ_7&Uik=oovKGZp
z*yH$enNEM6RP(uFJe>GJzFu5LN`=S_`QiD4=f%!8wshpW>(vX1EqjEyXu=;}&9Dbv
zt6Z8^N?|{<EI=39hH}}`qCJ%Z1a19J{75Ywut<f-#sC(l2f^F%D@-HtZ=Vd<?=5Za
zLiz}5s;dNivhsP~{Ut%S8dP`O8fekxHX`<p#m#?H{KUzP#yXB*dRCS<U<C-pfgG7J
zClQ=Szu?`r49r7{l~;I7bqvuGflZsmkQ(7U8c4LAfXD&T5rT0yBuJ&s_iT7kxhnr_
zDV0S2FWv$0ZPaIa4K+qGa@CBir{stvmEXQ*AX8C+;l@*CSpkfWmUq?Cqdutg_~bou
zFw}oCdh$1jU(I;y*J{Kw{9rKh65=_}YHvywW!S@T8D#=Ckq-jwWNTjL<9SBR?$w_J
ztZ^cw?@pXaPNa9yb*urK)%e+`Po8%!Osh`)jse)eXQ#0G8rp`Dira-c%sR$NYZVyZ
z=uSD}LxdEgbZhM%t&5%JuT|IlWx_qjVe@~8txe93_x3Oa?>q`U2@2N*zIfyHp=({r
z`YZD}12*PUC$Iw489h={Gq4z#4*k2txy$;{kUcFt^DGz<Ds5Coh04dH#`e<Qyma7T
zku(#_Kp8g!!n@M(<c>bpx=Ox?anJVTj9Y-(wI^?gEI4x>XD;;>Ui5Hrd$xGqpFe+s
zPPJ=Qg5#spgv0c;Wv?8bPuE`-F%aD4zE7jh^B7ddgxMyhP6e1LUO?Dxce8cQm|ck&
zTFnMn#qkiHVV7Sd4{?2e26wqX3Vog>1aM_bydQNl=@YWGUwC*6wP8G;TOIuvT1MW-
z2J%I85A((mcYpr!c6qct5yK@Jj!%EEqd&$8D@=_0L~F~lpAp8bov?}rvVt8GDF!V<
zD4lU!{ca4up_bVg&Ts7;fuF>p+eoh@m_0f9d^}2Pf4dBeOFOaQ|MB4Fpy=&l%A4`y
zXR#O1$&8At-O1}_$4Dda<M#Cp2}>(|-~8y!)So>u|ED4|sa3r+?d)dBr^|oC-Shs0
zHPa?Qg3pI|oC-cejOFpHPKHm)o3!jmr5dIU)c>8g=obePH9CM?M3QKta>)ugB5AiU
z7KGoh__jLukI|OteeUhr+cIIA13au}2JSD)t>9kD3}r%Ja^OAG`~DI{i%3uf^g!|S
zHIh3p3b)@xzlb3=GIAOj#2kNNuw$-qS>wfoEA#0HBXY{B<BONqoagouUR0C0GAE1!
z)41Qq^%aaTKcfp@_WDlK6tFfY%XmJ8iLa3jYt~;r%WYTS1n-ib9DdkmODU&-C54$1
zk7n1*Do15W2_$^!!hPt%qzeB!k+ho?nKgMY+Yw@^4;QrAy81<TwX=WnzOuD0QVeX4
z#s(pCdb+kO2N9yt*EP)6;9GB`4F(0}>GMf1pW-%-B<y(rJ~bnyPn}N#u_YxXWNpQT
zytEB9-PKqqVgsL%)#MB}y^b`<7L8-lm}`%>g67V71t0$6&9^A;B0qA$o)y-K+)J+$
zI6sTSbLGoy-<GW9nFoJ!Y0jl`zoO`6(=;4}9GOb<E4oqgW1!v{;P>=hSQ;j{#P}B9
zP8^wS-Gy1*cNrK_`(?e>Hzl#c_U9+Xq&&y!ks?Gm!y9?vJP)mVyB^C98VV=jojQyu
zQEKpra8FzC(~Zyi(2DU;jhzuJve`Y`@nHT03)5sL2ONxa>H2>V*4z3<cT-iC7JV=5
zfLdMMfqpCGQb4(qxCL1xaW1y54EXA-1#AJy{{p{zB!YaSy#B3FBZnFu0oq0bJ+=5G
zoI6hdZs0$h8oxH6uLfqm?e+4E%L)4a0#&2A8SFwzB>eYd^beUt{9(re#hsX0vmxQU
za;jE#?ZPaS5ruzz1ca<%)R9TVCaR*+A@s$@q<@A$R+pw&(t=yO6#Ly|q>*+*H`eek
zaUo%dObEhq)y4IqJ%{M_(pDpEfNlh>KG&ajm6s@z8Cmk@rhWmT&a3-WWw0~YNECdB
zzS97~k?dXB7&V8%c@xWZ5f4?RlcQb{Zg720{h_S9q5yyH*vwSKOpsx;=fFWu<eq*U
zoHhxYPK1JT)G6*?J2>p7S9Ae9<~VTvEovESGm%U+KEyu;XPZts8)2R-V_@&8*<B|L
z>jm|;MY-VFHIx@Bcj9z9B-y|lg_@M|L6fY|ZdUd~amJ*9MEGU4=Gue>E|4-f9F$X`
zL1tTUe8hh@MPt+SNZcj<G^~}#f-d(?Cb1DA<A~;R4!)MYP1VtDFxeqa6M-I4vD2sI
zS%n)Z++amp>(Ym!isqgaH&MGA0IaZC#)Ki83cV6N*n6$D_?Vhi391aYvx`jP^&oI+
z+N&P}lEA@fh)*W1ve6;qn*)Mhm?MAx8qv5n{&9acxpGK;%@4|xJ7$D|;a&Gme1F{#
zO~OpOr6V7^Yr-YMuP3RGbHA{`l-^py=P!`9qh$7QDv6|n)d8`dCpdms{WY}vOzkP-
zUrM3pOTj%reM7vPIqaOE5qa-{{Ar>=Rz~|~d6V@eE1Mr23GxnF$Uvv`=P<Z}(Ft`J
ze6WAVxstb$K+6hbmm(;hFA5ENJPT^J2m~@T&iSR_&1gs5+=eYiG^D5)jL_oN>prb|
zBdHEJS(M74^j7oHbt1JOLUx8k>r_gNjas#rF+!)uj09b)`n@2D5&GE2SOZpYvGqEh
z7!$i#NnNB1)&M?>qfU4(vg%>S%PK_P3I>0pqG%3ZR-YET4c@B<=Y)wwF$>%_v@tVl
zu{UcUY(fryn<A~lHnq;QwV#uL_S|qiHdt41Px&SOxoBJ_4Xm<-NYn-x&0-6R0*+^>
zSe(wZ>XYBLiGxHvo08>ubW|Lf89~pybf2cSF^SXntLPGlmS}kNP(3zXZ*$N68Ge6W
zDa%|Zc*#0c?%v!-{3ry;Rzw|FYMd8Y0Tar!IGx5sbuwiyL2-b>`<E5n=nd}&0;fAS
zl>XLQK!JZT7^5+)FHMSCGu<X9XDktiz%lB2P*uaJGPRdQ5x1y7>%hL29rD5c?uh9A
zPj6ki{B1&dJlft*FO>ax9pt8=WetC7(M0TcMAtY?Ha>(osA;TVBD`^B5A+-<ZT?MY
z%#a~jAPFzY3Ya00Zf@bD!9}*(WFqLIAYUJ8QNnPLPJeaPL#iOCu<{JCL9xrTX=k*9
z<NP~ts`q!=OEMA-rI@@Rct||JSStQbtGt&~!Wd0`ET3W1stRXj-eE9sT0wsnWR({%
zv3EEW`|>ibAQurE>qITHibi_-rz8}_6{sOWir$uM9@(%LOe^az4*FQXkP=<Vg3}m-
zF`8kORMt{qcW9RH;rr1b)BK5qK*tSReN~u33u{Kq1&7VBHVA6Pr$^Oxg{}J00%#t)
z1~rndK`<dpv~g9>+QfVLaW8)pKvOmdjR>{W&`0gtJ8)G>m?#m~$TN}P37taHNl<!r
z#|NSQY}DYGtBxGl*O2J<uZ#vw&+K~ivh68Q5;`G=ElT+~Dud3$y$;6iFylOYoyUVc
zx-w+;v3Q%Xr^x~s^2m)D>qn^?<7K;W9mcoVVfb?0DMCa~TJv5dd2D|fJnwO0#!^}e
zV7eM87>S{sAXK*UxWsHtH~c(2Jp9jmD_03Xj%ktbD0J6HWcj?$twd!qaWDWsBHVG&
zeTbuL{Vn4He~Q7b-A;&TYI$l(2riO5MbN|o^fYf#Fo1R#)c}7=HCjh$&8XxxQ*8U?
zxvu$D{i5f9(C}QEctU@P8pgJVj%y@C-eBnTyl=s?E#wN4mGFFMHbZ?HUvqNk@U)dA
zhh(ddi(c#b_D#1G#B;-g`cqlTg((dGzL7~k>(p0e0dry4QxkpO_&_|*J>(nLJ_Lh*
zi=({n7NWw0OUoUHWn<E>s`(c(ozam(9pw>iO-ZFk+F_^@=)!-aZRU5VmE3FnNPe4v
zB4tyl285CleWC@c#l~p9Z+?lp7+ZyDj1e7nu5<EJTf-{0Xw;P_k9T(!r6@#O3*f-*
zR%0qQ&QdPgPLxzd+M*aecvJd4zA7FeV~6)?ztrvP#({lWz&zy8WT}lmyBx`xIGxdQ
za=EraRJnKg%+P;4jpC~gzu6?09Q^`|^FX9Ap+4@-zFA3ZR`bYT&@V6sE8-Vc)fii|
z1)4kn9VeHN5AdTfvE}AR>jj<GMX*^aGrN|V4Q5Sz68+3;A~R6&oBGuV%(15jaF0T~
zHXk9&UjSCX8rGLe_|9NkJAZV|%_u_ZCq$&Awo#P|o1=e{>SNd_NJ#Kguw1}`zUS?J
zSyDf^)ZF$uZzyGjvtOQUR8r+b9;azXmdRD(lwF)U-4PF6fok{X)5W0&dHWZ?vO`?Z
zyLdt=2?UxohVV~WA-EV{;FCr`o3Iv!@4P+8vokEc2W5u#fVm^n%7Je0z)tM;cX5jJ
zk7_N~q6B}?X@^XJ`i4C8Sy*9INWK=k95oBdL5fvU5PC4OVfMw!W>P+kd*XELOMFTF
z?j<XkOw;kIi8`@GBl4?2f<Pg(e4|_La75_dTIVXYn8sw%CWCeO-*FI&J=T8{8g7NJ
zg}I1Q{NJXsesVtBB=RJ%Xw@qHuQ7%rLDJMB$WwptEK^x-W`??i!gP1-#AOZjV1;hs
z$%oTCcX%)@FgOv{gLi~(aVA_Bg64*W=y}MxH|~}1V=>tpIfblRMlD(36_W@%6m3}@
zfPLz9RKz5SJDuxF<-Lm;8Tw7T;1OwX2leVinW73l_{P^WUJuO=piC$eM9LXcnIE#t
zaoc~XFT^w@9qiWK+^{<)8c@t`2^s?vYFQ<Biel=$TJUs$dve7KSl9bCrNwvCBiagv
z8%X7+NN<IXpt<%{#UTPaHBo)*t-Xxyyt{N9MpD~mCo1wvhUqIZn1?8kFI(o52$eh`
zAMq}y6sP)ejzK1HI8(3_3L@&1QenU^HkE%>x5BcR_Gch%0d{eci)8p{&fSZjT_Z4q
zf2?WK*$v$rUV%K2hb%i08PX>FRxL8RISrC~Wr9JKrwD(_;=?=@Wj{i$ipQkePLMzv
zRe5LVYlti+-h#E{<Za0Pc&b&xV`0Ul@l#7FE}seO6C|+Pt02s0yQ8D{uC&y2k3oMF
zD|Uq?$oY@}nHf2)Eg+VDr_^;xeyc@U{y!JA>Ecg5)NfZv3)4t60CZeioq<L)aK&5G
zRiA?b8q=U`7itj`m{*iB*|OpBBpHc_jRmcsnP(~<4wl})p4+&O02R`@J%cc>-(7Ob
zgCd8Nv)j<Zd9QMfviYOsrcfN=ut$IUJk9f)WXPRw3{~YY2EBBu?E(-kZTU-hm_iln
zlok&4ANI4&9_1%cm6O;YArZSj%=OUwE~cP_iJVZe<DriMHV)6yiLm2fCd@dM=ue5^
zxZ-<m1Zx?foU>H%f9ql9Cyvih(x(>Pnh*O_N~_%Hs&|v<EX1LB9}-me{|bL!2Z`fQ
z?NIfWo)ubbt(m;+2OCN4stPt7zne-=*!G=p0_8NToW@&a9~H#0qv`~ledEhkpx53#
z;85cZfpD!P{(4P>$HQz&@hMq>G1QSZeKqfW<nJODsPlozp6G9mmqhw?rx-+k8F5pL
zQrv)$?mnaCh6#Cp`oh9>BBOtuNziMAD;E%$zn7dqcj0D}?#5q$oS`iYjMAP@xx9_M
zo40oSZ5Ws<Q%tU{lbNTTNgKkPCFF+v+_o>!Y}n=^%I+=@6$L@MZN(I#mAWtLbPr9+
zq@TS_!$|9FWib+s6-Xnp9^W&5HXf>-4On66rXc-?rg3Qy)}XRXndX0MHv9mrw0$Ty
z4A&x>A?UzoG*mlnk1F;GQA6vkOP82``ebhfVOYp9y;@vRC?GVJ(6SkMO`F~dojYo;
z9*6miP*fSvFnsW>lh$B@BQ%uhq_Dmqx)taX$N=+@QjNXQ$(F>x#<Xw|)Ul6=$$Uvm
zzT7{*2mgF|mzkho(!YO^Ll8s|)5&L4m>6TL=@FAR;5d~H<h`vNs>lgPzgNbriI$+p
zL2cugVrgDEQK9I@<Tlhnc88G{;Kz%do1)=Nim{$xPjtAjQ&nTiyrEXs&Ln$$2fdjd
zEI}}uDX{alyP1A7F8P%1L`}4w-zp#C2NK}|B8#sLsED>*=~;hiX-?ox5l}r+H`rr_
z#)9Zxc!(BEk~GsA+v`zgGWIFI_ESM~K(Z!M&Wu>Qd3LlOu7>+%stt0j+k)*asjO&?
zBbd_anaI$ed>ss>Q!5D~_bQ*OkBJj8_y0Jd+?zr>846R!Mp1d>JKzwOemy>tU^q#;
zk*3>eje_@zxto8N*THeVGH=}Pp$=EY`n?PgSOW#pTbQ_?woR9xU8=O(MVgJ^tyTi(
z`+OLQkc5<By)-29#6pQrgt)roGp5qa6k-0(OhBEj+^k?c%*w)TxAcaizXJk`#X8MB
z+3|r~PsnVag6>s6l`EyR8I&1+@udw9*Ym0vl?8@XIt+ggjZ0%4!LQk%hw(8}Bp649
zJYX&j2nnjyBGpSDE)QgA3*&2dtx|0#c#Y|Y=Md5UIT53VfAs283mh+AgQvQ_WP6uS
zYcUpz=-7>3MkZgJOzu}&WV+uh#zp{F;d(|z9K%1}#R=biP51bc<tjN2a%5;_XbSE}
zC=D{tV6%U9U>yF3wny>^HA<!GS{faI_X`ZAIP6$|&>4Q)g;X@8kROg^OWU&;g}k*Q
zaptz?51=mGwBFgtk~0x`jV+)D;n_!M=4@%hUph~DV7+yKkM5yP1vX-#Qg;0LTjZxe
z^idO&m+A2^ibB{l^z&VokH{`bM}_mdTGmJ&gFk=WSOCEai5(z~Hn(&WK>mUKcC2s}
ze5|K*J-q=;xld-l;1|Iml$2VF=7SbUFzM>9H7XU7@fdi5G)4`od?+F1Qd*o#rAq3!
zz6Db1jdN@eh2r4qLX8B-EeZ~q0L}FnxgB|@a;<1Q4SWXs=*L=Ei!p;?!-9dD=wXlV
zB{+Zc?#hJV^I-ZgETFvj8Y}iDyUiwpi9rbT9yH4VmXZg)K+^C_p61sb9P^8L7YVb=
z%7&22ntqzFG!nS~w19PqYNb`;%|kUyn$R4FBp)pqwZ4Y{OrM0v_++rF`j|DR)np9i
zt%bl;_CL=Gds!j%Ow|#<c#px5o*B9v9BqFU&~cdUtI_nxfDHI}ydJnI;}BhTLU9p9
z072s@yTE6ZJ8-dO6mEk_IUY9Y$w%KoiBfDx%Re<VF9RmAe%FF1k>(Z~;zp7)%G$P<
zoTeEqtHdPWOQ?kv=tjL#?t5sXxzi$@z7<EQ8>3Zh16aUSj?)my{rKp?AyR`cmDhjE
z@@eXFy=j#FhlM{V_3}(0C{Mw0ueB~Du^}@z*6lTB%MSF_Fnjn$8rg2&V3$ejYKCAS
zldp*Q;3(eCx=uD*M_O^Z*5UouLMcR6k+k4!W{!7U>>?J_?urJ(8s>N;gB~s|vBurB
z4sf`&u$rlE+KM5xsIGDHHNw%>8=`-v*e8K&9Z>J{kn=XbN0!UCYAt2ONt5L8XmKRD
zo^8_%lx1k1u@9&S7?A4Wm-n#8V)N+st6!fZs`C#I#sHr>^UuooS&j7e+=loRzJH0y
z(ONvLTgYo%rkUZDF|KX!Kc!cx2maXJ6+iB$V=yrz_u{IcWw|XaI-X}rwEce&W2e9<
z(g#z45n6kf110v<7CpT~!vWNS+2tV3<eE!^=$LLr{t!saJk?@K!g^DzYL@KQSRA9v
zXhc$^!}XD2JY0_DU&Q=$XNn4u>jlfw+Z<AS=>Od1x$g&xsVR&Cbj1X>c)hVE-`@iF
zW)k+-a{Ne%{r(uo%X&s%-Y9=wB*Fl!zcJH9b=F#n6rfER&U8fy{C%7(Kvjc!W6*|)
zU;Q=+btL!4Ov++rU9+4EJvU-XTA_-+bxc|aU?tqtSFWY&NKc@+d8X}#d9tOXbvnpY
z%)lJORG9Xib#S3dH%?9$-RFMrjcXzP1o9>s&PWS3jY+E4s_$!$6AgdpBC3a8XRN|3
zISOdvw%9)^5z8Du@QrJ6qIHNDaT(F_=B0DHhP+^#nKH<DA?!#;*34ZiK(+Sq=l@2+
zKR+M>h7U1OrcV$)TONAYUcKcGefOCD0UJPSRfXmyKeCkOE;+IxqKpW^9r4~}C0xy5
z8Z^08n*`=)UZX1p+9-bw7jI*U{}}^Z(3--f{9Zgt`Q$H(Sj@oXzt6H^37Ijy`=lTk
zZ7L0eN_B8{Q8s4b()!k0Qv3FI4M4FHA~{!KCPim1BvfNkV#xsS-L}q^N4O^UIIp5b
zkr1}v;X!y@kCV0(el0SS6g9EkuY)#+e+pPHZKYH_!-k8i1*LyUzGnT+FvyZc9CJ7v
z9Nss*^e4CNJ%d9r*_|tI)_R6Bn%Y<vqguJJ-r_wGHA38PZeNgV3%fbjddEmr+}u0e
z%)1I0S+~|NI&?MtbGC~sEC1>w|0v;2#_?!Lg*Kfb&E}Z=&9Dq$I6t!WJizO@ZOC*&
z^WhWEF#$d{YJ-1rx;gg=q_^!}Ahk1k65}T-j}TVM4Z=&a!be@aOiiNeaCUAHxsQ_7
z=(;cPdq^K@pVG$v)aT}gW=ny|N?Enmnd?)(8&{1%;}!8j)Sop|`W*>{R_mS>AQ2N4
z&YSqd&<Il#zbl9&P^!=zDWM*GTMnB=e?q|`H~<x{^m~87$j@{>j`qA47{T(0*0vwo
zBYU$o!tsM#yo5O<Cc7j7h&&UltF;NtBRfZH(+Q^?$@f&9u$Vf_M6^EpY6GoU2dBD9
zxAP!M+R2RoH6=I$SgOhl!Wc*-^X)y`4#+uyY;#=C#uFk<{N;SHl!+g)wB%XMyjpz#
z=dSYE-jshm!)MEaVx^isnKC)8!^Ozhjrh00-+7aJGqUFrG#*WFFF$o-O1KvsG)Smg
z6EgM&dg*^uU{iU0-4->GnB`PXyq*r6*%RnxvqjZ-iu3B?Lli=(d0LtTfNAgq67mZx
z8m)K+a9y{zK@BM1P#_n$4Y6*N_SPfk;7oQAUJ-v@4cWOUNFm#tRXGDP>0jP6{ce7$
zEe#n@E7qEHTAyvlnre@5w(H=s>F`*hkDl}U@t13_kEut#`LRptpxQM=;(1U$t}ecA
zwbi7dVbNX4gTB6BXj`cmKk8CN{?H6b;hWjRPVol(dYMy=@xR2LU)08OINS&&|2D4n
zh5COyD&a<LaPbh5D0Y6%n$HA73Hb3cn8U4C$yn6WRp8j+7@a(}Yvs8}(|NY0-M_SO
zG93CqAP}=dTVH6x!c`j3t9E0iNtv5zu5$;k_5093@>vVsx2#*qEgb>)yD3q1k}LQQ
zfZmcP{jxDh8;UNWD@H3(YIZ1TZYPfa3DAG_q|f0<i<z>tf2~&RJCOgFKE#<DD26Cf
zbtL6)(o09%xf5DrwjpjQv9YdAb`8_lEMw0dP`l6y;Ty*`1M?E?<h+k5sWVUksv*OT
zkiHRv=-fx^D#?H_GhidUDGQ1%1=?yPw@Etuqi#3LF1Bb`I=b0_YCR|m?0!r7W<q~q
z3tXT~6!P3eNDyI`=>)^zpM%d=7-VZIbaH@c{)BD5@t0E?EyBD(h}T~SM270pm4+$j
zlX6qI65B--K9v!cff(O|2UqdLeIif_E+dU`HfeoIT&pekFpkf^Tbi|>qZ&ptK8W5L
z`~;i|3|anLZ99Ua!sZ=S6Z}5dK8k<(Bt*mrHj((VjnOT~Z)%yDMy{ob^uRI=!<?da
zCkNb*yVVSQV0fW#8^8@rIeDzcA-fYtw(BtGG{O(=xm-iHZc}=N?&0C$g-d<uqtGY>
z>J;_h99cu_^iy#fFuFp>{V^uW4?FhaO;$7#%{;MWDru8s+iaOLw!xOGrDT7ZMzzEs
zCq+Q40zv~1nDT5vXMk=76kBrgR6#R|mkmJGl-|nkbp3cde;<8Wx_sQ?6a@7!D*ibj
z5%uZgsw<WjBkDT5!>xo&WF7qOB)P+<lGBKhnuP+tnk*1(QEX98hu{iMQiteac*>#9
z2oNuHEqm`*E;^=cAF?ZakQ0C8dfrMRQKS2cM|bFc3SXibj83W(h{Wpj10TruiciGw
zQ<maJlsZe98D5qyCmtl_kF;JLn;>Ma6WtUcgTvV~9qD9DW22KU!Kq78(YTHRMb?%Q
znlHfy#)gryAi!w{38c!6M$k3Rr2X|v^rw(hV)UeeA6ZC?Z@7Om^KyURu8>w;DgBpG
zkv@;7;xHcN6==TpBKwVn?No<qw7c8L{hq@9%U;lQDb~85>-o4ZLCZu9Q09zrEPmE&
zGRA&mf_<9az^s@Kq~FrY8_8V)-0>E;u`?NQ3&^j;4v{*#LjO2Z<@3F+{)F_+F{C`L
zoZXa<qi7^|@%$!qrV@W)6d`G>4fPl_7p(8SHVekze@muTUw(PV^CgY@__>xBO>^*(
zZ~w>=K6u8Xluek_5hFqTHMbILXB%_o&s<*)>+dKfMM7)}gaIxJU`d&ft;Z;X0{bFF
z%&t;KkB~v$?vf?84&AxF%ge*tRhFhT93voSMuRF!r7BF4U`l@@5gKx&Q6WAtA)1Gu
zcvcPFIu>LBF&|ADZzN&L({BNgU4LVS!3$jclpz(z#~%jYtMU)KwDt#Di@2C5tssC4
zP_m*VDCTLlBVsB>0{F5%L)`Bw93N4kmof{c(QIEcu~A9q<)3Ch`iKBkS_xhx%+M53
z0F$9PZCgIeZ|Q%NEcp+cKS`$7y=Wh}bh4p882jsGb&13TS0btmC1ccJmU{#n*BV<R
zUk2$9D_6rw6ysp??1KVat?fB--_r72AGpU>6_Q6u>X8!;ZUPWO^0I#Uo&?MlBSD@I
zZ$}bdjsaTD#d6Y(Zcuz0@_(GyFmEls9`qU<egaWW-Q|B}e|%+rW+=sxSXCF`(1unx
z=)$AW`J#<zSot?V?+w!V0&SxZ^^7DZLUYLZ{bru;K}0Fu#0^=}Y5vaY=uj@e*3?!X
zTohRnPLQEIP{qLKASEzFD!=FKk`O|6?Fm08p?lnFhx?ybThH^*Y6Z+i4qN|)X>I^2
zn+Qp9mkxi9CwM?UI5?W#oXl!`>FtK)ENZqh|4m`ROjMsg_mqn1B6onCtl<AhD-*$j
zb6!>%F7W{;iiM`fi~SO+3cr~dIl7vPPv)FVp_CK<K@d+t%Y|dwSs}bWfs2}%%^)hn
z`cOo9KpqRChm_V9<0I~d(D81|3zzeUj0}l6BZq&V_D3Zt0s0m|(0x!bCumnHH;U*!
zK_XH@*IJ6yt4gn<vF|-tc5z)FI^{*0n1m5L<Q#-q@A%#b0v~4_K#a~^G;>Z5FuN)E
zRS)K>0IPofG@?>ofDT!Xw8GSXTu-#;cCg(Ifm~r;D$S|BeQTCpzJ?lGSyMNan&rK^
zyHbA#6$We(MpdrzfQ}qM?VhL|>o}~7oS1Ng6VcCr3g0SPg{leb`gH&pBe+vNX}4;d
zh>7u&R+V+U4Bq4j(Nbivd%GHbAjlQauR?1+I%d9V^0{k075IETgbjEK;2QBW;1X7c
zn-7yxadB5yEalX#cFWm2ayXwuEX?0dxIce}%z$2?N$&S3Cj3YeMGRYN0lRoFKfR}r
zF>3KWi;8!oHKNj|ge>liYEFHI@*=<aNXO^^R6wi0e6POjvH`9KmPsnxv`prQ`Ll;S
zUo^;MS|=iDQYG^Ibt$JRPI48KJdb!bEy=@6=@+E+D%%o0fOI8Su;w~X3EN)LE5XX!
zX$}#8NuTCbxqdojiAzZd!E!;zZ3H_5Q%JhsL0eC-Q4B<=9L?k{JXA<SCg0HV2M+!N
zG4wTBOu1y~iqIFb%yD}fQkM`h$!$u0=RPQRqC&@#u<tT5*CqUvbqmW|A~Fj??LJ^?
zacWtp;GdnbUw^hJiBB5Xpz(cPY6+dilx5t13Q#bt6Y=g`<?ec^y3OCGrcMM-!D3B=
zW$@l~*OaUh#9NjR2*mapA}(Lw`|s7pRsz57TsMIk3-M@{7ke~aXduygqqd&eYUTsE
zDpev}jU*?E)D#;8r?(~PR>MaF$x*EQ$dE~z6KG{)832Rh_ppSqbZeU;7`Cid7lbB%
zR;NIl>?Eyx5U|a@=Ou{UVdz?JTzFSk;Y%9Apzr%lV5!RA9bLjQe8{L5S+OeBn$8`~
zN~AZ%V92CFj~)>znJPACF;m{JIbHJM=7nOzV*hR<Reh=Hg1I2&f)C%d!YIts?!CZi
z9<t9-{*csEnqTWgt#iuu<x{T3&lf*`Hi^n)7dRgNbx!)#ppAN{(Z-CAn<B6{OTSaz
zo#>ak%!98xVYt_3lLetSh`8(Ac(Qvgy+mTFKXBe=oZ8ySbFZPD-RlXfi|ekFJ788N
zN}h2rz9s@hMU^EiF8GyRLpbP`igCh!CDPH|BQb$y-Mty0U2v;_>ti5_!cYr;_$tRj
zAZUDfNe5o&<ux45b6B3gX7~ef=VdKRpJsZIxd~_u4P4Ekp2^#qSygN1k%Ny>+#Ehv
z^>%p#oPuvW?Dxe({h0~G8*B|qcXRlaRr@K(BiZ5c%0-yUhhGxYi}SaXvP%SaAF@N0
z0u&3jQ!wYxURb0a;<5{RK>i|sTCND|t2dl<5KL?4{sPP8O~WRHu2upm=gZR|P9Mvd
z)uu)h`$5$lo{ptz<~odDiJc;yF&P)_uLMep4jIE9@7D2p%fUcF>~8JKS^jYcC+6)(
zD#9)m>M^WS?$Wr(+c73o&mPCbV1Gm4EBo1-WUJ6<iI<DO%&bH#3;r;F6y4#pvsgP6
z4Vc+eeaXiDQ3))LD>m$7hh3g~G7w0))eb8fm!~Z;QkZ;Jc~c8P+;@1vvTDr@nfH|^
zm6yb<@?TsdEZax{<?=JT4%p8mA`}EzA#je0+++zC|BH)viq3@#x;10lwr$(CZQHhO
zbH}!A+j(PW$4+*1zSE<B`}E&E>!L=@wQkoMPhC_g#+e8dHv@NVi%Y$9)DWj#j?>7Z
z95Xe8hO)@iI}uK$?(6fNj%v7fk}eYjL;?Y^OKlhQ95M(3HTv?4(zRUzY(<vUzx-GU
zzQ1edpB$EZ3#r7{+f+Eay*|K1)x1?j<NN(oggJ_#HV)%wh|}+XbDIbTL>6i?hZ251
z^I<ZJrf^>go-}2=J@1xn-kZ2%ggn$0cqVMT?@%8}ODL)sJdsckhcx6?IOVd1PgSm@
zIoanTVuila2+)*j8%+JBoXxW%%|}SPp@al=T}5mP1?0O*eYEc%UzlmN$TI)oeMBKm
z`yxxBN_tkO@yJbo+(5f?e0NlsI?K*Asblj)b!c|<&eC?5mkePmiD+Jb1~e#xG{Klt
z&t>{wdI(%REP|#rfnvc+UGmWnZv+_to50>v9`7m3Q!w=Wo}xkmF9u_5{}jeFKt%?o
z2aUpA$>6aw(Z2vq>czGb03eL>c6fP^*pY)xMHI0e>P^dk*KNe%hS}$`PN-~*Dzi%)
z*nM`louJK!Em{8zIB|DXh?dtRs?SU$=Gd7jOk^o1C_oazMn33V=rW|%TBrU&2zt7l
zQKkg{N9G)0IKI%s@I@cA1S&VsZ8h{3>P*V9*%}PrEazj{E*m=pOVE2T)Y1s{E1q&^
z5GoPaHA!uM7O9e1d*9!sVX8z$lNxN9!L10ld_QB(NT@EpuOXKD3o`eRZ^G%7Cp*7D
zD?LXG3<Ns^^3@vw?3D*>2bKD!Bnt|L1_bqAkag&Y0sRj_0wDoOn%i4g+nK9t!2vnk
zBB+w8BdERrqXDIB>O1aqBm2KK5TPB=Kvcifd){z=LHNK<bV=WeFXMQ`{85CJ`O`Y9
z@<&8v@7n*u@eVuX-zJ&A72e>w6Bpim9ye=iH)puf)O^?co`GJLfZ+O{CRqEwC)c%T
z=(JtjlO8p??ta#z8wNN{r>|deO%s9rA8)UR$GhjAzg}OhN%qD6RK9M^aXd=TuQLrd
zTaV>`TFo0P%>aR)fpGue1v?-uwM{zJI6h!yf1YRobQ!VcD40AY`BZsS9Xs5lPkYe0
z?`5lZQ*I97-WcQM!qc#n?tllALib(_KnN!9*1BMQgkZ%>3+8oQrB?%<?^!iwhWd76
zkL3D=3Ki}Sb?ErItz9vazky_-VPi%ARcr8n$n{NbXBGE$!(Wt~wk3bh8HYZiy@rA*
zUVDA}FmhhcJ^;L>fGllM#xixT8^c^De>->`Jtk+Yd2gA6&Qy~i{Cpio%@_~2{<KY<
zEjf!kVdnnL%#JV*<?P#$PPcyHr_%T99CnP6F6$Nv0(^y4^Ylf8GlcjF_C~ToF6k<N
zVO&5)nNcrQSlg`is{zm06F~P1BF|-{_?%JWGn_tPsqNrr?I^|pf;MsRb&NDcvS&+m
zBz{J~@8@f<fDddFaxX(B>%hG+b<N4O5JEOHtxVh!te+IlvG{_0@@JyM6H)b`B>cw9
zU;QzGuFem~f+7VK9TC*!-q3~d+*4$KHvMld+=hJlEL|$9r9}Dxvf~v2lSNq(i%^sz
z+@IRN20ET0!(FB&A(5*icg`J$`BA<Zu~%a>*;A}bJeLW_k2yj3un6X=R%zQirYQzq
zRTJW&)ukXQ<b^xvWV;fVT5L6A<2}B(@l69i)5~{ijC9rgPJ6Gsh^&j)TtUQtYqL`j
zV~-)HJAd|o2K4vuIcnm-S}I9V_q|rAk!Kcp{62)f)?ee9kegI0S*%KjXHE-;3Eu{7
zx8@m|&f)vx@km>`|4a%$jqFw=3R_O#rh~IQ5y*<y^FXFpBW+wr55eoPLH^M7;vmBd
zYFRiNlkrlicM}}9jtis83%MJALBTs(GU$BJCnX8-AyE>Hz0X$fG5Q}_82%NNWq&_>
zI_N+q&-)8T9Q{MyuXCRo=Oub-S=s5o9}n!+wwV{QVXF^D9|X3agpf~N%K)>h2ljs3
zFR=#R&tFbcY1rUJz0TPXo34dkQ?p_FWw@`V*^qJi*TS{$FjrptY>dKxTmY&)sd;s5
z%3W)?Bx~?&ty^J=#EkDbmX%Q61JHqTV6R=Z)c>VlLe{(h@zSJltb?=B9LnKq+}x)Y
zW3|l561EGRQ~Q7~9#`aQf-An4zF&6(?%JDR1Qe^Y=mjA?=lbVl{i;t`KRI#m1Cy;Q
zNU6pQReKAC3%+o}gm$feQ6r}B>#y-Obc$LS{=FnNuqVMc_nVHj=;=yNy(ij)NQXv^
zS1<`3V(Dx0IaCqp5usj;q0C0}J!L>jI~J3j(!!y@TU0RC62(oe+^upJI&zR*e&rn4
z1zLtI7+0xDn^#{aoM+1JH-g%IUqExJesFk|7-tjxfa*UWzU#n$9h9op7?{<Q=KS@~
zrSXXBAP2^r4BahG!6s%p^CcW(ISkYZ2-s7e*V`8lf%~FvWvSA|6?=Ll3K6$?5+cPi
z)_-t-0xw#8tJoJV%st{nWPj%&TmCTRn1`I1b)F1dIMqp|;A?nQtU2*fjZY<ZEV^VG
zv?V5faYPoj;g&;xDBf}wkn-v<F65WClml&4!Ci&Lw3rnL<-b0uwrn-uOymB*26N20
zGmxFsdPgjqg6e0tjVsQ^e+`@;rT;7=2`{*0u9&gazPYGKTNz<dC5;QMObg2I;Jq3q
zUR~t#GOS9+Oa|yFy4K<6VBfL8Xoo%fWNj(Y)XlsLYwgN^StHOm^O2@qGP<mzWq|Tt
zdE}*M1Zc#ZL=Y=e?J8eoBjY*h%D1xdc*pB^IpNhypp#qu0RAeprj<+v6k#7Sb(Wmm
zXr)p$Cll}oEs~o@boHf7XWmLFqHAj831gm|-LM+NK|r?0y6U#3M2?IcS@A-m16Aqj
z8(Sc@^5VyTI@!A-?X*9sQb)|P;XP|$V#v_9`SRx_X{q_3py5SjEAvAcG>1+y+h0TW
zv|>zsPXw7b4B@fF+Ui(6qjy>a_Oj)(`L+(iS=bG|t-RPl3NU(r)|Zrwd;Gz&p`M1>
z!*_Lfr@EMR{^BKpat4x)SHS_dlG?_Qx!0CJ9s6v5-;4*W#Hh~jnf&9h!PS457sT4Z
zqgoaR{v#S{1gXZHgP(FULPQT*t+l60BP*7+L<_qjH+A8D)^QP|eAtqSCAAs*GI#p&
z;dF0AiQj!on`p}O&wsMO=<eMhtct?NP(3LtYx0es^VF!mJaJZ%kw1NmOA-*cfiQ~(
z5sEB-Zw0Zw?@=)ELF!Y_vIhjkUCHJ$vd{sZ{<J0BOb(gw6l&pk)iv%t3u-fVd6r12
zfg|lmkb{3mN&+#GuR&F$Gs#@AN~@tdq{XNFrV5@*EoIm#PID2Q7O{~+T|e$0O|G@1
zm9so!Ui(Qk0VopgV6=zwLn)N`)ulf(QPL}aSl_mGH*PP}CScF_D(wy#S9tRZGL~`2
zVxP?m%Oo8h`7}X@15@9(vhfz(LQTGp;fm55v-3H^tgO5K*0np+_oL-7QtqCAamxM$
zQIds(0`-OrT!aJyDnteXQj!G*K?nVRvEYB6K?1@3KZGDu8vMW0|A7Rk{{sn1vfvPZ
z|JU64e<A!|Mu+;}6{{}nM~kRep1O*xBY+96|4I{yyDeymzj4K(x{hraZz?I>f}5uu
z#n!2IRt4jEKhbTa4HN{dbn;Fdus5WmL*;)pesO>h>pTAF6p;PZr0BG<H?px{*gv#X
zQM~mwfL*iW_Q?E?DI{^RXX?iAfQ#LK)I}d6a*7)Y((2QKryEWs$igqUC6|OD%-`ng
zVuF-ykpx<yd6iTSF4E1OB60+ujX7eSGYV4Qijk7DoC0}&2$5k}Ut}%zUwcz$p)}71
z+<~NoqobnJOBG@!SZb{GBfYU1*D6A00$VRX;?IdAT&n<YBGfbkGJeuLoI>$`6XZY7
zmbxc%!S_weMemS3Llh2D_H!cE#|LUTq+KSY5Xs$^4NvzcIB7xEqDu_qhv7Cv%J&6m
z3Aau;Qk8D+3Ro=d@uY`LTU2+eFQf=!y(t~s@%hw}_k8A9#W{g5=a$Tkkp$x6RZVEO
zpr=oDN5v*brO&-x5K#j`koL`g-27;`iLl7GtZ!~PRzm#Wl4r;hL+HwvWrr?gBHPOU
zm?50XXvnq}6Wgv#|1ustk4bL}|AO49EtF6N9XqNp9oZdzo2jc_hPshb-M|5+s^9?m
zEp6OU>u4gjNCOmkuc)*V+KyhXe>g>OT4h@wLe?7Hq`8b^u=r?v+E3Mg1$1p(ts6ti
zkf=7r#yUy7HoO^eGKL>%K})Fb$edgC0lu2ErC(2zm$Xm=L!+AxjDrxI<~Xq|C>uup
zlsxka4yON1v2;F20*y6schz1qL<jDJ>&Jz*jRdz#W8EmbSl0Bn&cU3X<v?N`ZF1Qv
ztb)rKRI*e=TP9i)Ee7X*rw1xQsjq6RCqv{OrOC$S%lSR{8=SpJ{jfvbE6-kS#sR>D
z)O$sXaZ%i!Jbgo)i3jt%Od<piX0?PD15#euHmXGeE<ks6$1AI?kaS`;wUduMKmQl>
zo9UsV+NvTh-Y$Jj4zZrqhbCqw8M^eMNThTxKh9oiE9d>B`-{bY>ur=L<9V_)_07(d
zQ~Bn4LaL63c`O5VVTineTgeS=nTZRL&7AUarwN8@aozyHx~Vd5Sq8qW{e`K%2Ce5n
zf-cR3<7y(i^sHwfPN=uh#SF#!{XKRNBe6pGI>ZIL`>SCeH?r%2+bP(@nD5wb>*meE
ztf#(<{<)^W&pp6@pvi0cbQc=at$!!nRy;>hxNJSXYj=CmWIp4CX|Z5p;KV<n(F;@F
zbY{dhiqtLUTlVa%){g+5+_e1?8Ux`;wo3yT#3U!;@X@54YkQ~E#8C;yUh0Vabjct-
zGMFKnIi6z$?Gq9tt09$9w>%;vDWoPkftS{h^{3!&jIr*2-*R1vSBo32qGEFJELyCo
zH%07}l}y4&>^i_<7}NIhNVWS*`>MHnCsVH?3h5d!k%GZJI5j40CM8*ob;vqzDmMVI
zWnl!?Ig>59<mt7N2r0r<H<=*H34`PwXvUy*klEhPD13Xqv<;vdxZWUNdW$9Apt4*w
z_%QWYdgNw*Fpk*PL?&>RDXkCW9^TTwDJ#eD-6Z9}UoXlz3nkcqK4N`6($1CYnd^=L
z;T^pv^<F!G9!#x8vbai}i_4F}+w%$mX(u*UXYz1~M$#J22KL9nX|ORMS>^>sI_3hj
zg@M`tPFIG{f&UVR%XW7_$<PB14;Gc%@zxr+F#NH9^~#?toH*HB5{53mlsRg|b-<n`
z`A+uORy*_k^6%G%5OM-8ETV&$Lck+E7$~82_i48vRM11RKQTF3gz{HtVQ6a;WQCYp
zMz=3-RCI8q3!YT9zaR$%kEaJ3z`XKqv~68+JJ7%94TH5<*)I=Aob@@^ICbms=Cq<|
zX6!A0PU7HiL&$3Njamj{ECT%(`hZ@Am2Fna<#|;XCQ&?tKl{%GyvX*fel-}zqS^b`
zXU1(?FfiNZ(p*cHmXbTI{!a6dU(1|Kn{$a$en~{v2A0Wa8dMECjA@ju{M~CA28{Zf
z+a7jktjh4%?&$TlbW-?xF03o;VEvIxbXN6$uTnc3t8Oo&D&rs!gozwaTE8?e7MfKj
zD&Ik?Xl3zkLM6cStBNaxIc@ncV=&0k>zu4pJU6faO&*Vjg~3>+Js+d5x3Lx4?JU;_
zrt907XMY|@@<tfSn<h+&p${~fS;iq|MU1(QBXu4(>rFYeqS({%J>K+?+l{V!#RYhO
zHMPCW#q20emeCBN@`zY;jbjVM%$%=VX%wry<ix$8Qh7ykQi6ZQhn{I*@F=!MkKa?Z
zprSxfOM5WS=T<LbW1G;_Ybr(4B`N%J3B)8?TrPzrTr$G&#WC5`{52EAvuI}x;9s#A
zZ?Jq|*N20sQ#ZF=;Aj*NIQQ}5AGyMRHh8PYy&~oYK@L_y`9z9Z`e7QD1zXw{x`BzL
z*_@|!#-D}$i`$cLzh(zof@Xt3hvpz2S(`SU$<hN&yppND78{H`r01J7QTFgMc=iDD
zutF|f{MSUXJ8jvw83x}3_mT9SSd&Y3;HXO4urM@5*=oI7sRIRJc8xv-({9{<fg}Tk
znFcA)?WB7#2@Of9+C_S62)RwU8v`M`n7DNVCdHw*AptYC?X_`jXZ;F%?61i(B9}HA
zb0vl^Fnf|?92zQ3ie!pbC{<h&2AS)&nT(VfiJSPrQdQma76pkKH}`G0YbMG4k1#9P
zTdPs4`rq<Q#Ch4EX@hnrm3{<&k7Kff)zLCph{8PK+aI{UNx`?!e}uZ`he?_KX^>VW
zfkk79ydKZgt-@8-9g}F$rFanP`bSDSwCzVXnLjlZq>9$QhT)Tr!vL{?7~lP2gQF)V
zg7O2a7pgi{KpUh&LH4}k3_URmRqZohOI;Edi*$1Ks{!4#B4Q^>^A=fuRmq9)Q~Z-h
z^q${IaY?%ECY&9|lDGt|0$aJA=J4=xXUG@$VA8G;6rfF~(Je&O8Q!sWQnQ|&WVw5x
zhfalrhS`u7WJBjxW*2^N|B@iO;2|zDcbT^{tp<0rB!|;PMv4RLB>KKbvkYE1vLDE1
zd$JR;EW20fBjd3Zt<u1MSuj+?%uV)8j2KM~Ztn?J3hcpqs1(VD8m1&f-_!!%LuDXZ
z<&W5>W-WN7by3L^Zpx%3Y&q&xVitb{7LYWW-mPT+?Pn>wIi`cOimumH3eqYOCS1-j
zCFH$unj@OkbPB@`%fe_M#FW>yc25Z4;aUKf+F~2B?(b_n5Oq?2t6TjeW(A^;{R<MW
zCB<lCZwo;c(r7#%FJZVa<~_-KD<9%Jj;#+=DkrgUcEsrW;${bB@NnDV>b#9L&#m3+
zC>#Hly7c5-OCrWO*4Wz?c1N4ModzB2B^k1G0ppU90V5_oC<cVj2fvb@k{Y8MC!Aid
zUOox%Jfz<Tt?B}Q3e_P776DZSC%wt?qm+-uW#BN20>sl)D`8VfW<TlaA+b?`=?Ukj
zh^hfKf_}AvM{HC5^$sGS+i<u0Wx)Uu*)<e8JBriSbVW74UOwv~qa+IXXbhSiv5gX0
zObwwM9)~cWC85_K)(z9EBci`0Jw`Wuk#QBATpkI$t}Ebw?p#x4xw{t0uHr7(ieW<2
zjd)-fHZOKWbN9}83~|}JN+BqPV84{0&!l%i|D=-mn}mIZut8J*AG`?jT(zxv*C?Bm
ztP1xMX-Y-*SrNp)j^s90!`KU!lC>JsO`-c7gVc)dn&Wr03j@Ggw<dFv?<J~a*wh)O
zs#Ai=2uQkrF2PjLi2+KYW9AC=l@~M?$TPew2-E7sbwnWSTQ$7i5W6nwC?F1;(?yTC
zw`&`1_B1iv7{q7b{F+uDag9c!?=0m+N`_|AAA2ukH}ZO2Ur!%twkRxBN>wzflpv$I
zVHA@pl7#lhCcv`X!tp|yWq?0^bPB3I8dyN@y8)PgES=Fd&C9Ec$%4pORIx-ZYarQx
z#&GjsZV+o=Pp%l!+nPc?=+5_(Xsn5U5V0w`ROz62&QYzRp|2<C7q4By_ZE6lyo<G6
z{Y6F2ec$iPBV4$RJOgkf;b7_SaXEzhV-CtE+vf2l7<|#a!hv;iQrF2ytS){qX(Da>
zpfC%632Dx}sbzYaMh~0Ll}Qv`@it+_C&=tbW<4c@_jkzY6Ot7d6nZ{M<rz>&=b6#V
z$2~l*ejf@Q#-Iag*`-kAt3|U3+pKaQN;uq7YyH9tj9s<41QjU?N$pR<Ks);f_!G%V
zapQ*_a{8!j^Dx?1>En=YCe0<oOimR}&QrvHJ8+k4^LKn;Z!?kur<RRTo#19xELi*a
zv9M_M?sa?Lmp5v?zs59s{Q=&{i<Sdq2@*hp?+y4#hgfDOnC;rDF-bV2iP8^@p-{B>
z;zqgS7%X0Cc!L&*NF-KCNuTLl&(dhgR_=>35=`+gfn1jp<qUs#Dm-FTK@tu_LjS#g
z*+)r-;9(xKK=cw9ky5(nxvpjpZ&7O(--MYGU*hz~mr@`cj*ZEJ$hRAdWTU`h^qY&s
z<Ekmq1Z~8VEW(}z%Fc(kg5&t>5HtDf%7ViB4TtsnWbl8x`B{c8GGIr9ThP4?+0&eW
zR*EM*R1-X^&0sDR?oK>8*yGM1IQz_h-p(aS*PJ$6MeTDgC<J)=zyx^Gz$o+wFUg@C
ztSqfm1iqh9Pi0I?*#z^Dh0`7~M?rHIc%Yl-)GJLPV)r+Lw~`u@elLoLNZ>(WVFPUg
zmFHm~jNN82Ke%<uAe&ZD9aQJP$5%q_02Ge2^a_7KeKVm`{WsKl`Mi_#cvk0sY@}pg
zuW^QE>Q6DzUX4zNA2Qb9CM-jbTVCJE9J&yg6ae)eH^kd*l6pqKHKozC@dnOEf44YF
zcJ`KS!X~n@2Ke%po`jpmuOp0v-`z+FtW2kmLJ|Rl^^i`B_@DR3kv~Z(6f^hdaXg#v
zmgN{T!wf5VE|1@ynwgdjdxrUc{%;lrXU?YRV_ELcBS@~Tgy0TFAw@;NyNxO_pLF9@
zJ1X+F5a{&QxEu_k!*w4Q)1WP`t^Dqp){=2yW$NYKs-*0vV4kHrV#f+6Jed13x6xg?
zi8IUnb;w2oopqFSKGt1s;YdqOyb84IBe7$UBMDb<N&7gXwsK)%AeVlBdZ;DCD-k>@
z<W6!wJidA?2(2D4cu{N-5;lM&zDn*sWtz?+UGu#eOPcBfEm{7|>vd`44GQHjDsv>!
z;98Tlk5R0mU~cR(GPwNJ*JWop5b@N=5s4)Y!^hG_yLDTJ_Iz*E?Y6P!K#HdT1c)CK
z?aKg&72)v+FqsYSA$BHzw0KE}gdZ#YN(abxu75ON2o!)Xw(O8;-W<Qtw<toI*=V>*
zE+6A#*YahUmV1}@atf3jr&wAXQ%$RlMfRAIdyzc-f+)ClmJ7+ES!#6yo!?Qp=(u)(
zJ))};1yem&&OtLr4G+-2J5bbZ$?<J=oR_@k%N5E!j?%flp+Sg$HC#jX0GNjj2ORu_
zPpp!*!0TbO0tkDY7ZE_m+G1~_e&m4a2QT?kHE3?UTc0+Hl5<wk)$#h3l1_B}N}#&2
zKeE#$D}9(0_H=|uW==I13kcy<w2)j5F%~;^oe+iiojz5GJ_<H6$*|7JgdsGg6)AYX
zJ3{E_4G1mV38`;?)9|!kJMXsQome1gzR~-LyK0?nc8!5a@UPDvARv5fLwHB=(~WSz
zymae8An49#!GQ7^==PR!z9FOs5-=ca_`tON*kRFOZNG7gqnvM4(x5YYVy@oFZ>M7x
z@j!aFRukAGUR_S5a_2WmRQ~-)VTWwT$zz&85qIEcKTzp^jUgchC+}n%6`3WJ+Bqir
zy5u5=h{_mu&)7lqd&lxKWhHpB({ot=*572=YSA$9z+6{i=6j^%e`MRLc~CR>kM5Qa
zFam_03Q`GP%v5x0&$^6T5~1=k`&G!|B<Q^pMw0t>(AI1`*6GN-`R&-$9S76_>Xk<;
zP0uhTf7%Rx4$4q}FuD_xrxyNaHN-Ex^@}7NrO3^~pj*VpLR!HjMMTHOMA}-}h8Yv+
zcMRp7$19I@g>{2<k(Ns46a)ya9>iPFEAL5`Ui1H6pQg^VdJF<YV+1xR@c;9VK!C^t
zz*4#YvnLs)XKIz0PC$T=R6yqchx6YZBsGu)R<AsN-p5E-ATVX%nGRsDyq?DA;r>{7
zp&e)AiST?FOkD=M&c>%<2cR)mB#nXpx*6N;&FAV%J69W5F;}0JrShA21wDD9V#v9+
z_@%i+dEMLuw_BU-omMESq0_9CKS?^tc{wFX|9b`I#A!-Ss!mD_9T*U)H7Q#wx>ug>
zfbZ{r@%U6?t`7lcec;Z{Yp3=3@fkur=I2|#7IWy8^HcA{oG$_Ajn@v>xASrHMErLm
zE+(Ht-oMVKa9A;RcL4`Icy-)@oRj3j!tYM!-6!!LgWA8RK0B+=olQ4W2@w@7=kPF7
zAzw!*yMI<sW>=fl!r`LOhG!;qpsVyAI<3EdkIQ!xb2SAWAR>l`C*);?uFItd8@;yM
zfPnI4W|SI>ApdLNbam~})Ukf%Z?8St)0*iw*EwpBHCi+6TUv#2aW5c-+@Qx~LQ@*c
zUz;}}E{;5+*_oZ2PU9uy37I0qaxOx1v>2ff|1FM>jzX?tc`v>%;_OFDph_D}{C=^2
z&dfW|w~qf=e3G9x?|&);1pZ{Rp3ly9?5E8s2CDT1fm{dSVy{3pgrc6@&K0}|+vLt<
zl~Y&IxRej$gBL1*UcMak#mI^qmQnu&!8jO>+{XJDTsSe|O7IP;KMXcAVRPm(=gW{H
zP$c_Q8Hq+VO>ayEEK-``T1-v0!!mAv<;2!L(tVVfMY5FTeaBxfG+v)yJ65E5BMZ%a
zQ%(cy0Q?s*jXPwY!mqMZn4Oc}0cIAoa9?CM@24B3);%kW;;CCUI8eFgBbXzGVV1)k
zx)#?MEWmXz1j`^HO-A;W?HoG_$y%tApxYMR5jA+$k*^bwh0ZWwN@R>zwL-3co7si)
zAUgDJV2|W5GR94ki$NGg8$|s>M*SOB9g6ltc*u4zkJ6<ZP*!u56`{>MPp6hNlS{!!
z1al2>I8BxPd<ifY7nA4@OgPrPuIU!$nC1p)Eja0qF;V<R><#_=8kQ5LFTghf%bsnl
z>FMc|CQ0MA4pEanc;<^$X<H(H;aK1-ALLC3d>l>Tzrq~JBelfd$)|ya1a~^biR_cG
z+*Fh2;C>ljh2dYJviI(?_j}}C{>Xx-%Lj_Sa@ZMVeRvrayLb+An9iNlb<R<2vzFn#
z8~J~H4LBA58a%^|9aq(*cpG>!(WhWPac8ZsEZ&phyNXasj<rGSg$9U!!cu5QC78r$
zKSc{V3TU4{jzklg>n(Whp*%u11s|K+nFr9EZ}$uYY^?3?F3tFXKE+=T49jgRKVfX~
zrp2VDu~bYU|G7drrsE$9z@(`h%fBdh2E+=)ktC44CKZxUOiLn0IKx_rd=emc8Wv&s
zaY!>Gql642EfCirTGEGqY_F`j8f^%8T6Q#bII<P&pekZ7S+!O)a;}sHzmRweOq#OG
zw-hBFg&<<S5R=jEE_j-W27~goJEdc!b&aSez}iV8L-u4#tmNAlgwvWy5ZIM$y1<w2
zdGf-GRSU~#*(`Exz^nzSD~oo%*K3A`{HNi-usG;$e8vEe<c_O<4^_u+kC$lT-_M3C
zE5D0W!r{?;N=>s0$;m12BRp>I6*rOk$zWm^@+8E3mmd%X(fXxeP9y$-UU^jzdP?PB
zat?H(vd^bw&qVmqM3;Q)r~HHPt3qh(iFU?M%s(x|1?F6Zd0G^L1vwn^c)57roatpn
z|5HwpM;2Fs8>g~=nF|~n?-lcl{{-D$K)Zf2?`r%#d^o${PY_aS!u)E0YDqoO*0=e=
zZFhfBYTJImTt0za0uNMA?jW^vN9%e`zse0}&4(DC?uLQXNz6|J!d@6?G3}?sb(N1g
zXka$su>xOCiXZ%s6_bDQGLv@smHt2K<Q=;5)1o!{VAuzLy%!{OdCAir{{=N~-20eu
z4|uI%=~{1SeKMrXbmj{G#e5|dj{HLL;FO?}I~qgZQUvEC(amTj743LUv8X$l$Zt-L
z&ZdRu9l&7M9wFtEZ=-Yu``$?)rDZq#fQ)i_wcNtKQKaX;7=p$tDM|b!Ag9MW<B+BZ
zH;zf%Db3=4sV)qh4SKm-ajE$Q-}>jN3=g^!!Id~Ty}j~^`0vj1dEfiH=F0U(*#Eb!
z{Q(m&JT%q!-zOfQ#IFd-3>WeC1Ct7#q_k3P(lfJ6f<eYZL_EHe4-h*^ne<S<-9O%o
z&Yvp0yOMOcMY#J+Y&eDm!%o#@4+Q=`&;W`TlfDptcql;XM3Prr)E8SpCLi^SptT65
zMVBWm;LM#EXGqEA8vB9|zF)<eIU;|(D%cu~5`C^aEI?`j9xe@+gXSiR0nVmrm$;X_
z?sZI`WRQ~AjIicF)U9}H9nhcN{iWKd-lT_VcCEe5T4WV5Y<@Q|qQMN=pmOX76^vt+
z#CTMHfjWX-O)OTqL`hIe){3kxe;Mx_ybS+{9r%fM&gq%q@XAZ;_U204ClvUJ!~X1N
zG|kS6x@I=3uuG>CZ)b@b-W0`|AtL$IL9Qa}$(|-MrX#i`rU)el<i<6v#_cYgzb)MT
z)88NHZ#eHRxaykso`2PQ^S=EQIL|{;JOe6!ptZ=b0D=Mop;KTyIKf)%GqbeuTiI5C
z^th`}aZK9a*Xqn^rn%PBs#F~0&pJhS-9*P#b=B%@ukIzG<C?`keIi0BN9?Jb_YeVL
z05*#F9s9jKSP5bh{=uY^lZ%t9rL(iErK_#A!`s2LyVkp|t<CGr)7#<W>*ce<^ZW9D
z<NK;u;QOrsdmczJf|{`wCh*YkDEPhNA_nTEgTpTpW|G1&Isgy$7N*<{0s{h?{l)$J
ziwS20MT*KA75>5o$b&QjVrF6y8k9qlfP#WE0nfL10*UjpT+NEh{wT8t-wD+=TRm<z
zgv~kNxZw$yb%Y*b^lF5O94%byBqOPR`8X7w&s~WD`w5i)8h98`+xum8hdBTWRp5m>
z1BV0lIdF{F-50l+g*J?A^S%<zeP;C#PPoEws@G?7MB-u!mCO_n-=P{55q;=EVmGGa
zTTlO(e<Yzw1H&F^z^0LTjQ~|SEdn?n41gJv5%{5v#x?F{OVIc!5E?x1->yx63X2D6
z{t>D=(6T|(<m!ht;;~7_0j)MAls$kh29_bMEZThb7=v{5PUbWvn9mlym8%6Z0ek&}
zN@R0bW&OBdOd^>h1MayyMlMPxe9dY9ZQRgxuZ);GXKVr(iT5gZc+T?(a-fJ2EQXHR
z@#MQk*On>wfk+hwO^1>h&j*!%Osj_M5O4=;m`fGV`vU`WkLZW-Lew^S16G8rMNrFH
z0JK3e&b(4}L^RLC!0}X`pvV=md#cAq1%uShD*C-~^-dcqac6FefnxKA6sq&}l?rH(
z#DZ55s4s~D<-M9QZu0h(r8g5SM~nz*R@O{{Pt_rcBt$L)DFUPT59gSFbm<C7yxT(D
zQE~BX<w5P*_}ecOa$@n)j&H#N&R^duXOp>`;cXiWp#++eSrt)R)dnfOV;eOw2-I9>
z+5#e#Fgm$xeyB01^4Z{@7aVJ9g@Yl(5NUB&9p<=PI-Rslo}1;)=FV?!go3p~FzUXd
zl6!=)=x~p^gm*eu<a+vlS_ca?tCC`Ar2B>J(IK$p#qA1{thmto((AY|>+Pj-=UuU5
z!nh`S6?Kp$2DzY7<%jJIIn^fRA^4P_G};!_GW31tkL*KK2|e9ivs{nMZg(SSrotla
z&<m&;m{BM)F;11E&uO*igy<yC^mF4^ozjER%7Jq{OUi_HGDhovG!~+>Ge+TN)k5g0
zQb(2uMLJ|rf>)4EL@lof!4JiV(UMt~Ksnv-_^5C#mW#|d1p-tsOD}4JyZ)I4*PB$7
zQ92D`?%pDr^%cVLM1--l!7hcMCk7OcR5LdU8YrRgV3RDMtxysZDvM((=?Ug}(i9va
zu&^wIqobYggxny1pcNf7%$VF1Wt8HM`;!cUd@v?;$D3rKjM)J0L$C3}{m3z`Z)6$^
zc++hv$^=jZ7RBI4;1QcF`Jl*u!pWH5wzxW1tku~1V7ZCyJn4(YR<|G+R6GWkQtHEr
zJ97N@bYs{j3188$-Te-G36^;Oe1u4W5VK>GdgXPtVUZ($Xe%L$7M0k*T@fVXvdHJJ
ziY#qd-%D^26LZf-6o4>P(4x-v7+8KgCesPg1>PnrNttqd(8z$Lp_l>bh)p)Tf3+WG
z4{}~7l7&kvk15WSj`X5jPLMO!^6`pjepu1+aoVO<0sUy2$^cgBJ9Q4?ev;4tzp)N$
z-FzU;WMbieT3<F0UkRfHYl00JpZR^lSR%2_R(DA{rp0oE6|hnR@o1}cW=UW<epr;Z
zw4IVL45{~6E24kz#F{+*j9!sIu}Oo~{_a~sc!fY@Lfk%yQgfB}1y8?#V1(vttKwH6
zJqy;l9<Y(|N~dmE_!z&l;laAZbR!-EtzoBc0V%Y9<PAr(_5E2~)JsLMm`VWH*cF?T
zj*{{&SvH-+crY=B=`IgEECn>;`*&g`By4onz!lAVY@5cwj!EaJgxcNr>B!0@7{1h^
z+|_y*B&<SI^NRJsZ%$JL{+Nun-T60j(&|rQ5o)8<mDAr@=G*(p9-cvkBR$&jFL`;;
zzB-<NTsX|M8<T3kk1t1V|8Uct>;xnOIC&(THLcKaQL8W_2tHJ5wnJGG2+nh?wrmvt
zaP2ch1>dzkSe$OxURkfCr8)G`0;C(Sdf*b;t{$REZ4I$8<PawWR2kJ&D5lTip4)~^
zTbwtvk=&rcy|Osi^LucpOH`wb$=ebu9l>6IB48)a<k3UzpxCQRp(BYdA7Dnr6*FUz
zSXa()uxyXh34HAKqkdfCRya+j_Nr?6m8A%JSM#*0l|qHC1mlkOJ?B)JK~WJgAD+Iq
zKexlC1nt)rG7xvvd!q^eN~OhQAQHz6apb{3;9q(7&>>tz0hf}J!f-RP7jAx$;M_ca
z3SPjK^41()>Vw+jfLM?mICWFJ^nDHFNqZuuoQq*EJUrCV+RCuKAyA%|pG(P2S>h!p
z&K>9^3}af%#qfgTtTgLw$;Z#K#3Jchty1MD0u&q_aYG;wE>=Zc*xO*1pd3u0pC%3^
z7>ao7G5nUKY6#tTJd7g4<3i`iWjcj_EBB&A+`r6M3dvGl<$qF$sFq=VeLaPra^nnf
zSZl6JG1@l>W{tMENEpsmsZh5G&@&>-k(7wQZlE|*`S86;*O5)>hmG5jc%ey*Tiy5#
zS(q9G3<CD_GuZmD*>qL9(B{s2rARMeg-<cRnm7~Ec`z=*&Mb^Ubw0DWMwHfn@X*1h
zi-Z#kMfJXwSu7i#XJg-|o1k3@Jd#D3YXX$0R9q4$nf6q+<+^Z3%H}fgjrJg#Swb*6
z*qJF*LICBMFxyuNI-G{<{TOw3E6XH#Z5@pJKK>A}HF;Je?9yF1{7;`F$Q0%S*fkDS
z&^FC46R4z68PtL;|FW`~ArQZR&FCONDel~&EBFKvt#LHfw$?@0-B?Ezqg<Pi-1g5+
z@nZix7zOUA$p}0JV(d2U{!(Wt{~5R?P-T@snv2<qO$I}ELPnU^p4i4_qK;s3*q!xL
z&KcR(LFI)$`0`7pN}Co85pxzCC%;$xG6<gS{N)VYEfatmC@j6jLyXmbmtAFQ{B>I?
zvMw&x=Irzbf8q{^5sD!=OT|sgW;i4O3V|3yh<UN~ZH3;FUrjn7tceS;dSF`E;ng-5
zQyFJaBJSVc<Y3NVuGGuvvz6GE;adGv&d#5Zib-klwZAff6&yu9$jZPHXm<X3na4HI
zfFiDvK6}46qsv?Hqm%J}5q*5Umx1C$MLys>V!Cl@u~}{>nOAr}MkL(MRfj46s!Q73
zeVS#i$)#ZnT>)T4Y4mU|@?y8SYsixgLKe&Ztdzj-%<}?o51K#5=H=Fo!jDF5193@|
z$#<C4v{{1#U7AZ4y6|DXoyd$|n~COPJ6Ayj5nejVfYH4>gE<d>4^2cCG>k|KWuT8a
zCeRPoo;V%yXK=F}gHVzKxi@8g<E5p>u#+VM8#G?GXUpFS)u173fwmDtpBNKk*v1B{
ziIP)cj{??3bj&1(J8NVm?mbs<cByLLcOR}GkA_XH_E8VWR~ZoEVVY?HiWH|NY5fQ<
zKT}jPgcMR_CfZJaK1~}U85P@*Ky~ZqWDn&Q{Fxl%#e03>nZ;P{De*%wVQDPR*81E?
z_$#e?J;q$nUaUDc(;=URf!nf;=!gk0hJ0V`NMp5D=~9#zb?vYBnOuD71w`{5Fr{^G
zT{ZzXz;$^h?mBsI06WJW>J2nrtwLIXEZkMI4Lh=|{jV&4To%ob9!?6Xx!ajjMD&VI
zitWB7Ia0$yGMC8*Xh~k(&RSePv%PE2QvjztMj@j2!5L>;uJ%0c-R2kZTXg=qT(U(o
z6B;$og^#;7v8-7;FI$t#60SE7#AdJ6M<44#R|6^+;6;Stkr=BB!-W84z9;|FuHTkT
z-ekbpe6^Q<PH(hSBQ=tNVDB*bo1&D{xD#{JT#?X*cDy{ECc|uQG0~SOWJvUacK{It
z?gc4a<L^$)qMl-0^l@iE##Ie6%0*q3vLrx5=JOD5mi3=a>r@p7Yz^?W5K<aV?rxJY
zrxj^st@0*^&q(J=g#MAr&fd&Z_c<t>b|qC~*$h#CtxRwPyPc_(l=!MU%IO@%pm?x(
z@WqI{SvK5=D9Jl6Hsb2PD?Z@kE}0JGP0}@8DP3K(tHV=o(w5o;c+3U(TS2V3pUZ0t
z67qlyA}=fVpjy)fpQ}3(^$8ZVx*qWpAbMY=WH09l5}gTbJtMei#3kn!o2p(j4C0Mn
zz@?ji!advyYNVSj*xM+tHG6XOZ3MNVwxN{vm?AGk@$LYd-}~_V6DFklZ(}*SSMjKb
zA3{LGax<DvTDTRd1sV^UN2#X+4S7Gii{?fBv*PukCb+saBL^&QwcRy~eWQ?6aY4LS
z!P8-h+kLC=yK2KoP0jGi3Ud#}ECbwzp*bpl+CRz#tmd-4>;UvJ(m^J$Q_-(F(s@g}
zPaY`dCFenry0*FQ78{(a=fm<vVIa#CjwoyMNU*e&9c8$sjuR6Kr~{?EcoZK58o_C?
zRhqlUg}8E0Ei3v|GIesI&WJecz_<)0QXW}*BH9e!gEFXJhxkJ*tGq4QFZ5_+0oen8
z5}Qy5?$&z&1;c@VHgL})EBvE@Szg4lRtXEu@m>;G%=3Ii%mJvNNpHzKF(Q<M8W4z{
zM$pM4M;<-4T+%E^q`!)%SyURDIGYKtbXE?uE@CvD|32|=QNVJ_An$`JB8!)UEp>vL
zUCb=8_MbNX(wMW+(~Orx(Xf;3wQKc%aObLi)Kg>O_aeW8Jz^o}i+~+_3KNhm+eqkg
zCPzkJ!&tv_Tw?CcR4Z|?Q_GPORnySN*+1jwyzba8bff#Z(?{%U6Sw^I8hJy^-J0y4
zBc$D!LTz&sn92>0nsRE&wtO?=KExR@w9v)RYLX;t<ghA+egm;O1yJ<PW0g05F4cQR
z)cmnP$my=qc`6yObh5}g@Mz=;bU4#K+LJ)aZIH~uq3h`4mqDyQ2g<JB=m&)VIQmeM
zhQBnRIm3b$!be!a?km$%u*k!V)1z^^o7?lBW$Ep}$>qE_7!_Y(IQyDhm!|fN=O?mY
zAZT=2pv?QZo$!0~LaOWxu<);cNyBNVN@2Ac@0?Jnsk_l(ZM-O&&30JHCz6^yROk1w
zN0hm54)}AaLa2t865@fGu71+HF?AC+opy5=Q%JZB-2(JIMFk>R2_59c_Fa;>y20La
zc*_xyD!M~aVaDICNT^p!Z|zQGNl>3?ZPG$q;^9)n{XATNE880n>dNGQ<&`{q#=>rd
zy8UDVj`-Bk@u(DH_7(J8L7Ca9aAw@MI7dh)TBWf%26HBd^nvv%mY{mOWCm~;8Xd(7
z<|kRZFWJagVMdj@#Bk1YO2&WBlik=2U>jIf_ycHrZcHi-A2xgt$@!F0DT+KQXqmWv
zz;kh3Oz@jR1cCL?oj=-tVbHDaZ;`>vmV}<!{g&y(Kro*AazbZ$-o61^LId@LVoZ|I
z*STzoJOC#X1b7Nw=c#X8jr}k(n_uMj{A(f0RA+l0=lSmuV2s;<A!to7<qi}+s04>3
z)(pWbpNeFjr}0AePbo(6&*m#Bh9&wZb&Oy6?poceHwkP5=(dl492VKgV#5%Pfj|tX
zrMt!4o~$7^-S2sj(MaXrm}p%e3V+A`7n|4coK~y%at4$==!JR58i7)>t^KCddXvSX
zQMp@szhyq&#L|F0Cc=YWQ_k05>2VCovZKh8X)U#e9A|QiHi}Cqx#W>=Tw0LC5CelN
z&Y6`z2&q%sE{3Ro0nA5%s9UKq@>cQ@_S?b27{lEcUDAFV1@3V;#c-CDleJTWB)pVA
zK&V>EVQw*@$cfQW(v?qls**zjyqNy@dGT;?*OWhdipmmJq9*FwgKa&CDBGJ9rI;cF
zF@qhpevr}vSHdL=YHReF|CmI&7qiZ1-$X)n75PJx7;ch()@_Rciw2TrY$Oo~pg(tn
zP^wi}hlEn2B4~pm&@n`PCv5w1H3Dq{qlVEv_Sf{{*A+?Ln@&RfGO*1TxxGsQQD*eV
zUmWM-haI<WQ_Kdqf5#^*hcb4>MBO3_^4qmmEpo~s`YRAaah@$41^Tip*H-&ewCbEX
z7HM;qE7q=m^yqxvUl`E`YBK0jaU3Cd%t8UrEkRR^V|azR#T!cnh4Vang*pczHRK5h
zAIZW*;UG;wqXf;PAi#~T%hEgQOym8}OljouGK#2`Zcdw|$q17|4(C-4Vwp%UUf#y-
z#PeXcROH|i91mg$q6Roam(9!+0Fil=c7mrQ+xwP(rf6^MHCAio!(@99%WtH1w$)cU
z0vE6APJ=$R+OH23nSp#bGr(0p!04;HLKcQhR0l|2CfwEt;zv}{fZ5I_cy=3c@Ps<+
zsgz!G1F6jG!>M%|ItdP5!A$NieGcvQLJlYhzzVGFbRv&AS<KU*?P&XH>n`-zuFRhp
zE?z``P}#*G2!n4>5N1{BMWZW6VY{#=e|$;963z*#oDI4uyWQpxybr7$mHTh)#Ugu1
ztB`zf$71W%!!1O4T@Eh)U9_JSB7iNE=AdPP)~I(H+O-tlB%AnPx|DONPsx+>6lG<B
zdpf2Tr2--)e4%+N0u_<r^54co{=?n2meK)#bP@ce(uG$snw4l7Sr+%BHiw-Sd?8Ar
zCzPxC_?@W>Ke_0(UzrTnVuoKV!Z_C}Ird!L9&=u)Cz`;u-paoT{jElBbJZpT_@0^T
zho2w?rcWibNZV3Xroe!<T1hEIk8N^cKQqlQl2>jtFd7o?EG?`^s(`1pg~fhcjn?CT
z5_yK6<`c!_3$9WbT95T1NE>7ZOhezzfPA($gyUf=9$A{0VdOx)aE##X7>`D@xV`UT
zrj2(8$tvLJ>#<(f*F8G(gNxCO!HIpBv*;f#JXbx4ecshNY89gKpLvpkb(vzafa;mo
zWpl<YM?%xf08{(756$;$XdwsQD^|;Y!dW}iqOcIX%4dt#FH8n8oo$S%z&AUhTW^1H
z3Ld7*!g0PY-{H+{iw@*^MQ00$)g-G95Qu!wy1h59)=Btm-&O^I8uZZ#B-6aZLbUK0
zkO}A1l&))d)@;QoRaZDLlW<~e#PRP;sJ`w(7Fo7em1?pQ)M=KiH@&Qp0V;!ks-&lx
zgITd5XIm!2FuA3?&Xgsp(6WK4$Hfi#Wpgiu^G9WSzEAQ>)txQ0N><_{s!F6-UMuQt
zsN)!S^I^9qpsOZjW18aI(sv?U2A>gNa)v0@#8}%DBmru9K%uS)dWv+#cnc=#q!&Z7
zE8$iPv+N7}vjONr)~;Z6!IDdV%|jP@SAfH9R$4J&elle_ci#2Z9J7=Sf8j%VF<O~Y
zDF43bqE<a9qw^SAiRd*0+I}dYN15c~oh@=`<6{ljg*IJ!*2R%Q5edzbb7|6vtI!7B
zVxgxyYja)vB#Cb8FKXYYP)<(5_6el8RU1bv6gzA3S#y1$l55aOL++J-Dr!6#%GCxb
zv>)oPQC%yz#DFHKeF(5j0k1usngF?pO(Wq8qcE+nny9y6AM#&gCG(q|6Gf}Q{nZaz
z;MsM}?(H24*Qb!tMQA3Fkg}%a3TeG9g|1idldWA^MKv0O5Pq<)P*t}A9u2i0TPf>F
zJfIk3`b0d943>atstl2ToCnt@??2YsnAi=9PgIADQqg|VO%{l2wrW0)0@RCh;elVc
zZp|9kD7pzUa{v%T(<JD4IG;@O-{rSiSHkOurz<Jg_6}NLFwA7^x~i%9*>}$_oi%c3
zob(R9!jH!*ewam=*J1yPIprL&y&qUIe(GPmlw+PTWy>xfIMwBURC>3Maf;#sTHe!M
zKFQ{3Stv~-?~{P&4z~2noSlBf08Tt-kgROXK+gM{@$}s9w0RZknV&_Z3e%~}y^_pF
zKn_~9;G@*;5$p3Tb4A}(wN+`YXt+3*P&f;gH!FNta0x8pTw+YD&8wu3LHmemdcow|
zm6rjc+cma*<BX|)%3H`!^rTCAQL-+ol*Ra>TKKcR&iN~sZuYTYQ1)Zzz|p;pWNLw}
zotS0vfxK#>=SZTYI*4NWdEj}uP`CIa{0lFB$T$IE<4xlh`h)LZKhD@e;ab?*6}hB8
z|9}r}Xds2*hi0oktZf!g!5Zc7Sy}ie2?T)eyYI6AU~Ef&<b8=)xVj)mstj{1Ld!e@
zi7#+rzMS%^rcs*z8q{h+mTgS{vRo%va$d@{2Nwd!^^PIbF1c5FVx3hbIfmw<cTgWj
zjtrQPHTQ~3O?f;%e`3?hAvp2c9GY`u3*7B$o#wtJgjfSN|KgX93I=2T%A?tAK3{Yw
zNy$+#-Z(RVx&Dw5&u|FM0Bk^$zer*YhxKs}^8UdUzE40qtkR|H7EMD|h@NPtpkQS*
zcId*9*0xwGD_BC@5j3bHbWg){h=3T+5c+HITL*#E2!>bpR3DOr(BzS?&Unix0y0yd
z{KHd~1yuwGPzDJZW^#b^;T?e%iayr{VJ~&i;lL|YILo6ve@2cguUPc|Rm^>Ga_RBp
zy%N6F)B#=pdAw+R#Xqy?8mCH*xgq^<iHRgn@4d^mur~9M<6y!A#&aM>FvhHqTI{mF
zq_o?8h!8>b`qOHn0QZcza>o#u^+`WJ-6O0Cz?iC1i$gZVMahAAWO5>1QGUe37Q}Mc
z<ZzQhsl(vve|Fvf+|2~#CvSezEU}hiT_j>W{x%dOHRGjh>;8;uuf~Bo3NHBV?t88V
z^LB*UrjjJ6s*2Kr#!eJ62pRh~dJ20Pv<kG6escR{lD$sKE!@A5c|YR{97WMPsTtYZ
zfPt`xH`C>|jjUm^$Xl_fHcLX!B=%PJm%iSV(2QG}f5u|oe|SM2B>dV~>dT2s#PKVY
zujGBd_rd;b5h282y*4j>2w<%8J#b9WBrw$?!W`T~uSYqU<u&>ZPP+nO$I8uR5e~j&
zx26Q=S^)bkiF6}lhk&d5=D4yBA$^Xdw+w{*&86Bo8lmial`6eR!iR2fb(@Kbs|C~_
zrgriQe>GtX`~Drf=}PMSFYR*>J>rk`te*GhcNK~j#im+d=cM;*B^rzj*+dF!_czW+
z>821<KxXXQMOjjvrmnS|SA#R@fk&&iN1oCsx-e6Bwo>jsr3JMt0Z)JfNs4=02(ccl
zi=}y>!14&MAU`-~&r!ga9<n)LKdIBf9i(?ef5r~+k4<?DUE^XEsu6n=@6@RFuWEJJ
z)ph+AzvICi5A}W4WX+C?_6nvmO~DkKIs;)ir$UwHP6z>77a_Ga<{A?7sXp+Kk~O@c
zhk6r_dA&aGPz6c3?X>614$Bo*oCC$FW%hrM$<6tFH#)6zs*0A(tdeTsWa#(42ZdXT
ze{&^}^_Sw7rq<>$ziprfcJWd-`-~FgE%}CDg%qRuO;H<5an5cUo7f@0jn8dM`r?#9
zJ-kkakjQ*(!*a~+>%*n|2q&k9L{NtYD&Wr9e08|pTs4)l6#5*?hu<`$x1jsIX;wHJ
z=kP15ZRlI^fgw1<*Gh37^n91qw8aIHfA1&!6O-{yzo5&g?7?{F%DV0rOKq+lBowiw
zMeN`ULxi5e@BnUnt14~fpny-8ewW9#zU6L)cAbk&&G&J=tiEEfrK_-~I{K8k0(3>L
zz{}~}>tcD#U^Wtc&hYAW?l+uLL)b!DaJSD*<t-*lA`-VOCAPhwTcVIK(7CC$f1HEq
z$J)EN>>2kCKi=~amBK&t&`J&qG?y{N3f&@rKW=NAw?T9w&{zI^>HFgmPYv}e2QEPo
z2b96#JiV47`s;rI7(nO0>Zwt-#jRMMP;oCKYC*2D!r6CT#^)BtCEslUXKBUK4>7QU
zsYRsn3|NWLz{D@KX|BKgscaJdfAvVg89+6|<jE-KdMVgn5h1(#&yAQEf-0)oPdNW5
z`Ca|c_Gj0g5(|wlHwY31X15%4(huAFq8hi}#aO%j#9o$vCvfwhQ`!ydCAk;(zjpDZ
zu@(8Y&28xK{-YA<udBB2)b>C9?)Ciy;Cfq?@H?NSu>56x#Tq0&Z10xre+iIUL;|f>
z{slgZG><juah~Fe-f3ndj<fmJHrcRgDQv0LERz;&^)vg0k?!FMpa>E_FAsppxfvnh
z!s3|)!j2v<;932+JR$pO^6F#H8{%dz_%}wje(}0Ts*s1yh?X9Uu~k@+8y=M76cO&N
zw<fzxUQOh@r2Vn<u0#rPe=aT2tZ2hw9qsaGZ;toH^HRTnhhRL&9>!z;X(pZ4Ob(<T
zVa1E?tSPw%1kRH27e&DSBS8@~f{=+`i(8PT3|?0NZ!_Q*&p3i&N?MbgtL7`6?Is;!
zZwceHKGfE5$G)QOb1?iln|#kbw8fv&#&R8FiT{XYMDtQf;+MZTe`7v|lZa^diy)KN
zJ)PaDWT48=L$o$*F~KuRsB$su&9k-%G1K;TdPjm|0Z;)?h$aOD7`CbAW+|^to~=_N
z0k6Hm1M;DwZNG^|e5LepU21@~i2bjdHD&kR*Ab2FhwjW`bw;ev>~F$gzy}@)jH<Yo
zWpusz+v`Nk{9-aMf1tnv`kP7b490=K*;YvO>AHrJB*ty>Y(R2917p|c1nl6)?p|83
z;*?&uK_ttJ*(5l~pcN~_@n*g8_7#@K0n#c;;_-00xL+AKq%_qPeMnybBup&D1P@_q
zz?OT{{=vK|{ofTwAFZ2>3}n2{)9vf<5=$13#;id1qZ8m7fAGV+rV?3-b>ZwIzxqpH
z4!mgX>EUn!Y&{kZ0S8pi9<{FkgSPPnhqqvOnxWz7?kz`u^%*V<LM{!wxe2q6W4Q&(
z>ds*zucRpO;Ac&G<C4A4mwm)Br%u1x>#BNG@y6y_0dsI8bWX}!h0i!58cP1tcAo!A
z&5|*Pa)jKUf5hswJLPXI*|QM#5>ML1)GNeZB=t-f5WzjTZvu%gzL3%xW{V=^Vu-M3
zMrze=_KCZl*<a6>SG-keNLe>@`}5f5zQ@@n#2Irf10q)oIP4J7IKL5oid4QgTM(|=
ziUbdqg(Xxls;axaHn#JWFs=y}bPg|1Kr;-<N1JSBe|1wvCjR{uvm_)wzH2a~eA-7B
zDbwGmCUhgD>PBvFtp5oR6@u_`QBJ*ch*aA8<uYVMa7ztMId{*_<W;xXfqPb%7lT?w
zhG}g;P@3k2!hB^G2OZ;9ya;RWcisoCxaoOyv?B+i<P+@vZ~QwUI5agagsuF3Dq>M8
zI12oxfBSaI{zB3Od8f-#!FycLSXj@6{kcG=T)6uKxJuh3;X=)OBZ4_Pm(cn|J5W$D
zQ~pLd!KuEH`XsL*MZ_UH+GwL}Uc5hO^kWUi)zPJC@Ax4rq+i-d`Ey<-Iw9G&oVTCC
zYhw%t^#RAksL*@P9w}>Q6o)9sGwq6I28puBf5n;j4+j~M1n6Lh{Q$db(H)xHQoQ`_
z%ew5~kVx2(p<b}@ZS~i`5;Iy;wEvVkz&>Dg!KZ`<zTAb@%tWVS6&xQk&`$YEExx88
z)e<pH-8ljDcoSL&<k_Q70F_;W0xl`j<yFbTD=O;TlL1f!!_4tRm`?sAhKWo~zzjde
ze=kq201S}3#Sd`P+5QE)d^ZQYGM%Cs20XF*?xLcf^6$C(@ixjZm#dXc6<<zv7*z4}
zr0ftbg*|ih(&x{yD)t^dhcvkaRlT9{B$0W)q$}!SLjC6x0I@c8ZNxyyoFiu<<%Q(h
zAl+}n(hK6^?v7MeR!McsY>k<5`)v^Mf6VlT&0g#^Q*n(qx{(YIA|_l*oStHtQ_t-s
zWPG=LZfdC{aW04d#i$;?NrKV2!I~dg0Zx`KjsgCC%>-Ww{oBmWc3-9JZw}aByD{`s
zmHl~u%2Msd73#kx;k~Y~B>SADBkSD16IZPSy(Ggd&p1CyNprtv#wKjr$e7=#e;Ywb
z_-o4G@SWwqFMYGcnr$DH-gIs(#h^?u{-1N4X$WVXP|F*Zh4yzJ<jdwb`1({ykFP2K
z;tBg8{Nx@xlWp#<)9+dp$4lS$GOdiC{ruT?H{(&f*sFQ7Z1{j4yuK{|Z;bATbx;%O
zP~@5JD!1sl&km^3i$3?uo+&+Xe=^M}zbKHWTvnTh6Y>kV`6yD`O@B?t=fA4YbtMHK
zP@OEaoGL8KDk#k6$2m5shB~FzA<9$c#K;4crxHr9?xDuU5g{dE*G6yGtn)1%1jHI+
z=)OgXk1*C1d{v59=(vt|+q0M6JDMI!($-a5O6&-dTr=2Q^-}5X+-Y{le^aT?7;uw4
zbt_TE@!8L_l%Ad$tT()yt-A`d)})*3PG(7xJ^|(%I<WAUpeu=I3rwtNB>cr1ESt!_
z72yzDcD)pbWOpp${kr)<WF@4MbI_*~dSt21r0jUkg`H2EeoT!!lP-E&dd9dlkxoDA
z5z2qPquIliO3eJ*$$dW0f8X*mw{^g@|LiWr+5Y=giLd>s(4Ar8I`ZO~R39vScJ9^2
z3E(OxnjP)VIP`rc&~xNZJ0-EK{Qgggb-|=+{|dJQ3T92i&l@%nWQEz=dOY(h_2M{J
z&x*v9CYPcXekdg7xs0%f7U*Gn8~F9@q{m_@FiCAZP1~^CN@oKge;JX<EF{a(wWUV(
zfQx%a*7T9PY}_xF3xt>(2yNhcwSq|(_E&@cSSktVezR0;lJ#IZ5DoQ@XIY$Vc_gq!
z%A~Fw%zd2gvkh;QiBV;CT;>)1%dJe4Yd8UL34NX%eVz{35Bc=HgE?G^vkF|><4bR>
zeG<KK;BYgA13oG~f1TCn=f5{6&Z_vJG-oZKo3jgj+!QQueeI^JLD6&x=(rU-D*Sy4
zp6w_v=jm76LO%hlHaAMjE)@JgG8Ivf)^fm=<aAvD1+&S$?7n^XBEN7M7wLqMFJSk+
z?h2lnLUZ?~Q6&PsO?o-`s$vRs4H=+pwg6ndeEZzG*#}K2e=mK?2%bBLZvY)1h+jDh
zvW2qiChBfzhVMxg<A@C(h0}PhuQAQVQTOH~8m`nmf9sNk4kDq0lQ!Tx&Pqtf{IR{Q
zSSp@Z`6=NUjx9ouKN!h~trf?P!P9L;eK<em5+LM%4Omv4$PjfO$=V_`JBO$YOaC49
zS+DrDxm0SxN|iOIyO$4*X6dcF)1J9Q|JlojXh}~JLc;vT>~EXZZk)Pp6?&&7_@l3}
z*Gk*MV({pwrRcd9Gt=>0>>OWk#~B-;Gw&?MJR)?nH^eD61pr1sxxceu`eO#Q$N`=M
zw`&9eMK%QhGeFG0x3D?^^a;0V$_76+1ppgB<iEA32lNkrki%82Vm7$<7>qSA#iz_W
zxFK0c2a$XGbNT=<;zshG@U%~>N%I3WgKTatJ?uX(mKGAmj~cuS41QFeTIuwLZ{}im
z-~+chN6!H2hN4P6a8b<g&Hx5Tm_4s!eTB#*pYLoZO;q`<QpHo4tWgFGT<pbw*lu>*
zbKCR27`qgIV;ouvmbhTY)IY<=2Xp!R;VPO{g-k0Jxg=d~hT!ft_BOCL_PF=34cDhK
z?6)qXiL&N`M|1(DPys#pA5)1F7?gn<3zMV)?~gx0U7FB|prNa%gbNQgIar~wxo*TQ
zz1Oza-qLx0iCsfVq5)DWfGQcMTM$LVSdQ23eY<aefYtNrQ%x$=mM7uC5)}0vJMC-l
z`r|&%(<!Zm>ZdXtEoi0*<+0+z+uUq8_QRQcO`?_3k2avHrIlDIWc(q>76;hwc0ShM
z5f7MTxt!{mHDumwq>!wv>mmMIaz_1yzib04gF<7fhOENgR4U|%?4Xt^2^YEGlYeea
z@m{BY%u*;qPR|4)LJ<fAFulnau>+6AwXe9gBy$|#<&`H)vCksFRY_1#I2K)pVP&@b
zjt(Jnid?i$OC@CW6)!L*sIzXQo3KE3Cf2!cNEiW_XN_3%npJnCSzpUh5I$$~18s%C
zw$~!}=GVmDRhv_1Ftt8(kQvrRe>86+X}KDIhQI)HivkaPS14G=BhsswDJM@ZBw|g;
z2EyXk7Ut{78*gyJl@gjdsOlk*m5e$hV&p!?$9r2z=X0<Y-vOak<(ZCYmqkHCPc>w8
z(gmiWEExeeumBLpcRcZbZSh;rY4VJ%;>-NW8!(1Z9PuDf5kNQ8K_gYvf<ZfsHOsSq
zI=Z+F?O8oFL{kXjRFcYJBh;g7mKV9P8z0XRZv)XqEb~gu98(yFs1^hpg5%g0HYesi
z@DjDMN|H-gQt?GmRP6qh9;LtG(!Ymf=VC$j!WVcd60k`a(&3aymH|ip*17CS?SA~>
zlygl}P_;Dh#L-P4bP9l|)%bwE+<~!wu(FQ$k*ul9eh8#(8?=s&vaYKog^0KoJf8Nj
zHo!2bu9{eyN|@&I+G{g}^s<mkddWNRcG}nW!t|46@y3-huf#GK+C<f4iCV_)*1fdt
zb>EIHzatfXYf*{B4<ziZB#Lb60|BG~ZOAH4_rCYSIi)E?mI?gw@oG8^l}xsO{94B6
z(`#}sVe%mAvZ)@iDd6&2IT~XeQK^<FLl6R<7zO<|y|*U|Q_1BhqJo?`SB$xpiIPb4
zZ+mv&n}geO1{+UP@G`YjYYgHhR!1n3Yh@ehYhK=?eZ|Qf^0Acla7tvPoz_|6LKT(3
zMw9u8HsiP->M#=(lr(hl#*3MM%Tq|wk`&mOw<NK%SPy%94{Re?l+#wouvK}{{pGC*
zO+@;;pH0EP$Qy6IA|<M*Ftn@!g>3+l=()K&-_mTh1dhk$g(xL6y)7)$EHTbThHyzl
zCf0M_<9id>+*;QMk(pX&g1SKgrA5%_>UA$5*xQTR*X?VK@ydyw8nq^WopmD0j8OqA
zb@beuUwe{H);1XKU6oeTSFTocRI^9|&aRIg`BSy^4Iu1x+<V;ii!RJ6>ctHtRH5e}
zNh2}RY%ic#Z@94|VYcGp;JBk|nmu%dsw9&_&K5xwF-8ag7CRQX79@8!0@lLO%}`@n
zEUt;?)C4+7xd!|t*b%UQ<oh3enpDQ=Nlsk46pW(8q!xB05=j>))3%Z=d;IO%e7>{J
zRi~Y5OR@!MkEKCVvo*$&ZU_gt7vEqBOD4576%{c#tpX@1<d71Ps10UjHVa@!xFGuz
zj?ht7y&Sa;!Sdl9%b3Yh2(Umwxb@s$YpR2BdmdJ^!1L2hltev$Nlh!vn<?Xyx|>{F
zd-7NuDM6IfRs3|+m1$Bb(LU>0e_IO@x`xCBxFKveI*qUmvrLCB>PQ3`<zad{dUEsS
z{$PR`Bn(wikTRE0t--jlH($hX)~1dW$r~cdsA;lzJdp}h!5X3%i~6*x0icy3eLd`W
z2ND&LB`}Vfp;Z-snOr1iNDiQlRv?l=EH!-G4T%9+>M4FQIVoOQ9jVYTnl)nVZViFI
zatH?3-sb_ya*UrmqmGKU8Kr=PGE-C3G$r0W)j(Z0P(39@!QS9v7O$Z?0=|k&wtkMu
zr==3dGDg%2Rag)JAOcvMDBqs<#Oac&FIPvH&ojcIuEqR+h9DBFbG^?Xc0W^!S5<hQ
zmSr{3(Pf_#o#wR$S!1KiL2X28ZA7o(7w#@SJCT5{)cu(JEzC1o{KA}O^z^3ENkvOh
z8IJz|%A)CY0jvi01L<s0yi(}D2)reMlcF1{lc;1xj*3Vs(W6R&RLUpyE2@+wdy6r(
z{aBo*>EE<}7m4*IY}YYfippLzXzA9dM=z9a!E~pn6$4T8O77Mi+ZMcw!2X}<pyQ~s
zsd|=r*;c-EsBbM~ux%<(0vr}2Rodq2KMt%1nR5Q4>b#1CWZADu<h?;f435<B*3$Wx
z8o@G0Vv>N`9B*S{I3QZZ)BZel9!&8`(!DX4WiV5JdDB$W)6_Y+5tTt9l(*flpfI+k
z14wlix2-b%m*~8kIh!)+o|?(2emB$RE?*38q4{B?9V@v4-G^awhgr@?kv&CK47n~*
z6ow5Fw2&;TD@Lvk+D+TkK)%4S_rQ7ABl~D))N<3ce@fETPYm-jOG{E$%ql|$3F-(5
z8@<_oPUH=(i-aF*Or++twE4G5R7>Y8OF4$QXLT&f0$3{DcQ$K_5_Z^P)0y~H;l^zm
zQf56HLjy|DpqhX+8<QQBEBPe>8VO5~c5`!baZJs=$z3ZwNvr8Tqnj&sVJvagLX|Mc
zQLMHktElxI%MvU(u)ta}kF>9bRT*@dteKX7W0%h>zmz4V(;J0RsDtJmL=pztO^?dh
zReJvbqq?q%a&DgLSgPwImF8u)<Bdn&j(x4aw+Fs5Ec5LD0M%JEanBAxlXX2Yst4ie
zD?E=aH8CnyMNYtPsQ#FMk1OhafX{O5<2dOW%%U19nFMmYO7P6nl8lnfrL-o8lHZnp
z+>i)(qSH|Il=XE6?u{cfQb~86*paH@<eQyak?B%zaxs$TX(c9iSuH4ep+-|N3<`&B
zz$V;_+wwTwJYb`G>fDaFBy8D+UZI1vmE$&yU4_b(Bn$F%pL}8+O!c*WUm%W-6Hzpu
z42ujP#~}(36jCjFao+Yj90i9ep`YM?D=VdSrKgH$BY0GwHlj%ty+{t)L5+c7ur}Ma
zE?HFBRQ)fOt!qa#@zmOQ;e@2hG{#Yj6Kgl#02cNr*E@?+796^gnzED(!lN~(nP83P
zai%m={{W9j;@h2#uYPeO(>bLywX(%itb~<W1bg}s4U`LS>K{$NK%SdBrO4`kC$H&>
znJFr$<(g_q1DEnnt1As#i*a+XZu<;Z=E{0vn={K-HKdu=8n`8`g=W&yL8(IneoGfm
zrH|JVwe;DoT$PRF%F08+B#r^2MT15gYjDGx?6x+r7$q)C6iO*-mNu57T_%-?F;eOj
zdXF}8huXtz2e5KVI(jO4e48(SiW*vKHA5XdEDUG{qOqu30ecO`-mBp_C~7Mz;DYf)
z=2IKDn#r>k1nLLUHn9hD$ry#{YRu6(S1FQPQ_N{1dWe~Wk2uByqYgp7+k>{pxWwH*
zPgPe^OPiFCu9ZU}P-9sM*hpLp_d5g6&g2oNq?)RErCE@~OKU1M*@!oP0FB78UCG~V
zk7gOAEml<~xg0c;230K`yv1%!#~XG(Ew?wArK+pSv)Khui58)QxOqYrPyy6;wXJRM
zW4Hps;wrwk_=QjLLSmQH$|H$FD#vf1Be~lD0KNgdl<_<*RZy}~yfLqmwDy$9tjf1$
z4e8raJbJ7)!>rDd58@JkQBYHmji=$|k^IRRf;BPZb7QbAYhL_e6E0qp1pffUimtkM
z3$DOwV|(7>;^giW_pl1{Bs95{P)hJf;Q~uhtI?~{zLE$6{?@(yt$=Y=k=K@{hc;QD
zo>8KnIe=iXpYalXK-ky=w$>Ywg05v6Pvz4|1TTHw20p#X``ufAErSoBP%Lh2tt^sN
z=aE61d683<v${jN+Q6o+*3!J&jz-`#&+4g$V)a>GOzBNd1W|q1D<ME^uV6Pfa&At_
z1*`@FN;=wXwx~e$%Tgn<Od@jH9ZB`6(k>2>Vt2ic+gZTQsf3L?CE~Z37g<(}kWKgB
zRl&J8v9~?p;gULkx>tlyv687>mPr(y*eC%=7TUxU-rrmjD&tRjd1Q_mqfaeTGR^8N
zOJBJ=fyw^32@2KKmD6SsJk0YAI)Z|Zz_4Q6klX3F-qzcEH%Mvah9?O;N>u4cP&iSj
z*c*$3@;mW_W}c{+rmLlN0F?l-RboN2ZZ~6l9U}h#hrbhlS$A1hWfZGLNnH#v0$o$#
zp>`UL{Y1xm@A_K-ef~bv!jy8s95JMiF#$-EUaMG<&(8Z}v{~gaf}(2JA+D%+K|vX_
zQ?MgzaeLc;V{UN*wL&&&VP{!U#Cp~}TJ-Ovf_DUt2R!=>CoD2l)S7Bmh6v<85c3UR
zcDA{U>HboGds@q3axH8G>CdI9T6A=Z5lb5g)l{j~ax}2u@4d*!DKqM7SmdgZ8k)DW
z$2~JQ+>>C)>BDy4j^lHPhN?<=8c1b2!vM@91!S_26}28OwS~DS*kQcogszmw%cqU}
z>1BoN=%j<AP$KqYvF*k0fR?DJO7Du6S&x{=#2D0n0><~fmz#GS{{WIQ)KCXWoCz1q
zS!9p`G~K|vN5QbwV7~nEi1?l9B>1l^y0I$z1L!=E5AH!9J*|UN$e#-tauhrHW;YsH
zka1@BKTq3i1IeH{Z1KE`wN#T5S5X&h4?|k^2c3^LCwq)XT9#=w(?jLR(wH7qCd?Yj
zbgjUDA0c~bz4qdAjdinCL3%}ao-IhLDuy7RLYrv^V0S+E?TBz%9dz?j`t+`yYQjk4
z_5*D}fwtR=U=$(i{J5zglg<nO0FWqBfl1hQBVct{ZY}e}2B-52txFl!odIJYKY6~b
zL98q`0^6I|@rgPGSfCXND}+rVP#BfA`*$3F0(kj-Njy=}0MNXVG_F|#g&UD$q!Gaf
z*x$DJ1oY)EPe%1MHF1=GFU<2vC@-wC#=fo<u6}<!7Lr`DaUbyi0E<{)ZC+!rcS4{N
zZN-7rw`+n;#_^RjlhQ&|RcUZEn6#E<QbMuRyDh;Bx#MlSqZ+L(B}7zcj)*jXW;(8a
z)&+o0b|Z6P?oF+LnW4;1Unw;)(-dX%RqB|?RF_lcEwAe%(nY;WEyoVB>PYkInu@tI
zI%+6s=}3P+N%NOeAG=X~i6_|Gj@W`SFrhX28lVPf#nilUfqf`7eiq;X-tBHq1!VO!
zzGBAhOecULg<al4su+&;<%rvX?al#z%c9g55?4yG!i){R?U-C0Bq_6BVZE$vzy}Re
zClvl%kmfQ>D;Dnp#!Ch!>MU%xzdP&?2My@nG)QXRM>9NS#9EKyw+hOmdk`#r`{7ln
zj$L%|(dw0oiOCIR8i%h^pTu_r0bz5s?SPu2&1P1nsLQ6cRMKcyB^a)z7t$Vo!}Iqz
zX)N+hBGpG#ENv{TqG@&6EKRR+03=wFHn8o6@zK?05hhVZB0MQ{jyRU~Ah8c}1;?jr
z*qe?oyIP8ccz=gf#!jsUK`XAMi~>lFYybp3{XA&|e1T=m5j4=6Ap{|=!busI$_C-s
zpf=ULg4m7@+^v`85ZBh4C}yR9q%sv`nx|5<eCz_={-OrUdnqQv4`Zy#V(Iwel9qX7
zfeH^Q8wY(Ql!hbmSP|SCA8bHDG=iJXfW;yOd&LxGq=A18^`198-uE{4z)5A)(N$CY
zM3t;0Sqz3fwMNGJgKPsZz4jVS$2?)SZ4EpV!Inc)GD#)YM5uLrI-8q+4&+>rY<Jsy
zA}EK&NYqtP%qpTuDdixhl~*CursOfxzLRcv8v;VI$qfuubr8fRb7l;yau}|s5A(If
zz_29PZ-7`zl+q+LGs`6C`chFk{tZBP7q*Z+KQMjqAsW3Du~JDO1~+pO(~-$iZdllM
z7vF2&dtts@tddjG<?Rc9R*H9%R7{E^M!l{>5q`$j({esoLYhjXGPq@sT)^fu^8{oY
zYPj4DwQaGu8v}rfuIu%WGt41FSy@v_F#D=1W~nj8#k4fSH6k{$>ab&RVlg9?b!|WK
z4~$-&>KCVuK_h?INQLY*DzmUqr%2RWf)pP1#P(m1*BWP$C{VzE(2YjS3xEM7G@B^z
zy|xzl2f1!xMGSScl{rJChQci;SW|sY3isyM?Xm5A1#2*?>v`akdM=7T37BcAhHS?&
ziYG5(51A5beStRaE`7+yDf~d{4u?uXQPNT7l{uYIqo-W9rF5AJGXTNBDePF;03SPI
z&Ss;ir)laQStm$;#E}4{6h7bqYj!^5*qa|{dccuXMOiH$-v%!BRslh<RVoF*-0lfC
zB!jR3A-cD!JV2<<ss#QNRS6@70nwUM2!RISLDViy?f6Og;w5MCS2wa#^!7`UYb1)w
z%Aq`w#kgQ2fU@f5z_7WnzBCN2^GdwFN35x*yq}BC088e7l^3vKaz>MXo&<S?9c3{T
zw2@Bgs?xHE?9Hf)s2A7^-qyC)z5<0=(EU@HE}Z`Wud_elu0-DlNfdAZU38MpMyucD
zu_xt;Ors^~-k+yAeoaFiG<6Z^mY~KOX+aj=M%<f$$*>AY91KmtNn2GUQf2h@I?8}N
zrUbJq9m3duSOu}L=KI*302@H$zdfX*R-mqtrKp)tn9wYU#RTd!u>ovI0Z#2<Yl021
z5!IQK)Y=-E6Ie!BY-7_Zt-!N^>TR0rdx33+BPNqBa<r>YG_bsYMhVbL`gGrW?g=D~
zt^l`eK8l)}vSN|DdGM5p08}dWHw6I!g@y089Q)ya&rs0GK^(8}wA9KOlgZE{1P&K+
zE;je#{9gdqGMb{Em`Oz`nkSGmuz_@=+<*Z90(K+;&#P^)CYf_I_}HitmIZK=%$kUK
z7#7k9)JY+D-_&>5W0^)@O`g=Z#>+g0Q6m!=6A%{qG=aEXtaq>i_uLEtB@;A0YRfDd
z*vk-qkT2mj^A<I?ZGk<`0al$_ImJZO%Ti=<<smwTzHVF7ZUFCl0quij8I-wYdCUxP
zREWz(6fO8uQjiO8*!JU(o+Wdhlx&t7Y{g)fmWg6%DcYdxa#)LM0P|~n_OZk+UFS3n
zB}%f0(0&vF+!1trE_b^VbE@XtcL3risG4JcN@RvvWS2uzrME2H>fC=h9F8s94Ku1N
z>Y79RQYB$J>z!GAz;Dff2W|!b05EsK7N}8#G3iRPpeuq#f~0S^b93%CKT#}d*1ku~
zvsqH2Xs-^57^;J@Bmrx$ZSS@IxDP-0myu=75QkJKX(W;$(neKdRtx15r*UI($G^{i
z9(wT>7~tw^+G;w3G-!2Fyb2mNDt&S*TJ|I{1b%$m9*^<iCYLgZWYxlwGe~OZjX{y6
zP{kM$b`}RvC%72%r&uf_so|OlTB0PUSmb3=mlka?+hct@6K;Pj1U5%mG)wUBTP@~&
zOoBoKkPV%#Y)B0DHvVJ2`^#m*3OvexRd!Jqn54UG2)H*k?YY=*hUt{#Pvus<bg@Si
zZKSv352os+EoKD#$2@_)C8W%9Y{wydx{ocrX^>26*{;SAwy+!lf9dnUPDPt%RJoVH
zMDG;^XOz(_Y{4fR0U%So$lAx){c$OpW^p|O<a1?mt$tN`jH=pS&Wvk8at*eB<a-MX
zVJwbXYUhk+RkB86ggQJ)szuP+k}-YNZpfzniTtMPsbH+mjM<G}!_rikCX%EeMqpP<
zHl<z37Z)deiTMWQ;q82R{{T!;SHK#7K}lQ?9aL$t8;iAwKi2!=*C$g~QRH<Q96`k)
zMqw7kv^S);_<^uCJ-;!Gr^2j%#-AsTFzHiW1#~p(!6b}a5f`Me1Myu|THx=exVXjs
zqFP#$Pb*70x0YFik%Gra2E%}&_S<cTC$J?u%~u>%HAy7t5=Mw*{{YA$)4!zq6&vuN
ze84SRGm5I9%^JfIjD%rtyw&(gu-j1slWoc0j5(_`(9JYa%Bt=PKbBvALtK6<ek&1s
z_U-b+c%gcIG_g%7HDLsYQh_e^ASfq`5)UT8`&c<K%j2e&N_x=*s=%yF`tC><QWODw
zK$1<pAPa4RIjKqGp1DNqkcqspX!NNzT|}w$o9%POiMH5znC0F{F*FgZF6|U@Ya-eJ
zAQT4V3tTa{H$KM5*UJ@u6s4jOGbE%=tS=I+il?hl2Un=`*lBC<5%M)8(-yT=H9Ow&
z%}=PWsOTx|Heg9R?k%>@Yh%4zQ;LbHhn({iXzZY|K9CD(4Q)pM0E>Rs7%pQ`PV-U6
zPctL=7YiJUy4gW0O8y&e+W~W7wa)8QK~)WZ#8XYJ49_b>;JJu@jjn8U3xL)epTtKv
z4vLz&W~P9~VvIb37Lr$p#u%})5Nx&pdT-nkP4J~PZCnw<rZ`!tr;2yeN*nNYMmMmz
zw%57ewZfE8)>bZMo#u^GP(lL|kgaNoTZhrGI&JB&<9l#fH1z_aZ-mtrf+ImJr(92@
zg*t}Ab=Yx!Er57`f*jX0sDFr_WvOq7cU5L`_BSJ8YY;cM0k*LiQ!tV&s-V&2=`BS%
zU+;XesoLZ*Z7pk(FZ3g|u+hyd)ih8;RXnmrpL)wG$O_r7q>CMa;O<CkTZ~4`lrv27
zEqrj^*H=izj*-Uy0NK6+NsbmFTgn6j<;I`|U_kzD?S4CdU;LDYf~D9hDoE<8ULnX*
z$iAWm;FlKzxHs*2rh>O4N>QANAc-PL{E|ZziH+3nU^YB4#2rp`Q&bvQyvBkl50t8b
zm5M!tgHRvJPUI8L!vNZ~st~;-)S4rAFuAt-?_hfoziVxU*-U7aFEj*5bR<{xY6Jio
zSCBT-#`pPu+QF6Pw0YGyk2#K+>SB{obh|W!3j*hZa&755e&z0AlxF$uH6<N1bv5Rl
zj5O4*GO&qs+UN@o1*}N0APxosr4<calg*~47~9EehBC?$(S{>%Y`*(^@KHTH>+nih
zo);z%$O~%SOk&r#+!23n{NZfzvYHv9gDrQOvYps}Pogp9piVgYat*gT;gu81npDqP
z?HxR`vrVH;rU2hlTnk*D-~cbSp1~ndkWy2^i7b#^1d>7=HI(0VzZV>Gc<yl8xJ?Y+
zbhPyVL+VK6H<3lJz5ZM6+k0Brj*fS!otHx?gbzW8Hye^Uu>Aga+Sq+i^=&-T2X&VI
ztKUk0o@_zh+aJ*26@s!9k{F&Tnv|-UqDyJfb8@?SNwBdb@5$|g)U2{p7$Tp_r)ZX-
z29aiH!#5?Z>1~&5cLRJyB{Yfu0G35WHqbXGliR)h_uyXE!<w4y6>-5)8#CL{c>!Z9
zX$o~*0c$Vi^uRevQ9(#%WQeA<GK7}J(3_Zl+;DXpcMJ!=A*EAP(a<xK6foDJ0u)~`
zgLXm}RqgBG@&PvOgtgR5N?K})caKP0$8%{54cKg1`3BeIVQQgAROO<K%MX;1nC7_C
zWzr79!1`E#NWZ6T>;c|KLtC8F&5^BZB^?}}5bd;x52aLq2FCZjiQJ8_SD>pRqb`Pj
zhAOvMFC&B@<0!$22fBb3+<<TCHy0{JU6w1>B=H(~Vn<|EzJ&w%HGek1?ZvD%0}WP8
z(w;hu(PxZG+FD6<Z7N&&K^yFS>~$TFI0Q2KiE`)^&p{njFG^vCSBqdyo3J3><9i-?
z+t`^TXgu_$q-LdQG|y)RSJk@<4V!>}FK%$}Ca%pR%c}%cl_DKWJAjt+KpGWw1L*-v
z-AN0-z8S-rEE#=F5muC$)Cp<fP!ZEr$zywfJ8HLKYuI2C%`+Jmb4iywn8UoRG?32B
zNw?zT{6G`1JMu8<madU9DRTN+G==J_FC>!bx@Zk<LAK`GY<RK5dWLG8xoh5kfk{B{
z!eT@c4L}rKEHxj)u>_6D7T)b*GDl1CMvW>`q6;!PCPb86YXj1(Eqjx7^>4-j+%GDr
zs-l?^7}bS=x35Ryxw@7D#PMy1af2m)#IM6tOEIF<jFD6R24Qg;Cadhap2UN-@gdXo
zIZNfz!%;gxF4I92k_jSlaIOe{x1`$E8~*@68TATKLs*9<jZ`w9G&Cg@Zoq(s);g?v
z@;@*WRlO+|Pn9lnn<d$xX`+shU1X7fvbwR@TFOrCx#tlTRTS~%Mv9#tzDQ}j(;`Ae
zkelA%leePhZ&mGk-vc_$+Gr!Co}{;z3~aRb(vdT1B~+eBy9<`F0F!Qi11!36T{M*O
z<<BELs*|EE*jd3Oo84_|aCj%%0UhSM#M4012;@TnX$7?cpfg<GeZe4F*Bo<+ntDbF
zrm95qP?&^KyLq9&xoiF(Rfhin`wfoxEmlOdG-;YgP?b{D$l+j$Jx>`^_=xXw@6D~Y
zB66AX>0|gP>Wq@KwjhFkt>tn9i)vEG<$b~1V}Q{zuBnb%DJh{q#yHhn`I>+ub8Bz%
z?smjQJoOcM)aHhIif2BijTX`!!SxZO?#NHQh0Xaflr!b;9HKR-r=C<%1jLp~aG{#o
zKpVM5RlWZJ5x6@X7A~lpqh4O1prd@k3UHH86*C576R9Pwqzy`cN|L76vkTvF6IIl;
zq6Vo-eJevCG65+vF<?lsChjhzmh7jIY!-(!Y`TtmOwu<q%8ana=;Rq%#hgDI1E}t8
zzhiL@`PONhq?v<M#~d|`v&_naODO;;)O&jY#>8Chg(xz?Rb38MlSd;wtg$lAQNzd%
zq7<Ff5wIj3%Y5*E2+b8mHF!vtrP@PjJeysP=TfoRTXSGL4Y@XRK$UaJ4z+k<g-bXp
zS<ykVr~r318-PX0`maToL~8#4hEv3~Z5lMwHAq<10Aej0TwMZ-jqh?T-tg3Q6TWLD
zDoDy?Wb&tuH6ucmvE&i8&BrGDo^TdOYcn;~Uy}*omJm#TO&rQ}7bnud4L7g`=T6)M
z#v!Y-Y|4_E<jkJ7C1fgvmf_9LvIihsv9b60;>DIA$t&Hak~f4oLeYSAGT1QN)nHp|
z+hf~sBKnT5fJ)0uM8fm4O2O?L0d)&=VfeZ2e)qsLnrJDW8my+dB%M`@#ObkR1hw|D
zBpZ9*j4xAvm>kwN6E?Og-+?r)qIB3b_agq^h=F1bA)t7os8*qzdA?GIh{<(Zf=-*8
zFw@(PK3JNj&7+5>YO=^E9qTGf&F813lv?hszs%TP<lk-g2H9<7nM|;XQBKZQNLDwA
zwQKw#Y(EjUt@z(@+Y#EQHj!7zsE%NZtqYRc4+hqMQF0Fe`;ORtYvrJdohf{znOsck
z5mh0Y;9fv=2C)Lb?eaU8)7C`np$TV!-Q*s1+1<S^)(fx<Tb=E-@D6Hgv+T`Q8mFwN
zPb<tRB-)3T6V<D6plaU417mT(#DJjk;}vq~ln8xjvuadetPtFdM#O?|$lx2CL}nE1
zk*#!pUR_6+KxutMYFp`z^?(Vr&Ao`<-AHSsR(P6X?N9*IxeUu|GcCvj+~0sLwT9RS
zlGZ$J6jXIFnuc$OQm?uWG^n?yQD!_IeeQ7gKg{w|0#ar1`H^UjN2(6GnrB1zVB7GE
zur}?+_QYJ(HPp1}Eh4#yun9HEP|AdaBoHruwd`+h{`f*k(c_9KjFJb4x}Z`CMYwe!
z0s~oE+gjK8STT<sDkY)KA%Z!nB$l?87!;NPqR}HZvX?qlucRCQ04p7>jDxM}#(VI;
z!|3EnY`lmob0}iGWq=nVz!RlN8he1a*M88HRS%d_2Z}hG;L#%dOi)H%`djlm_9wP~
z!x)BTnr2yEK~<E{RJo%QvsKc>?fw-yu0c??u=gc+;{jgHqKZ0rs|9OJ#x;^<lH%?N
zTPg3czWe}i>!Pz-x(BVu=kp?`t=)!=H)Z@bBwoW;17o%L7*8>%sLd+gu9BUNC0i>X
zj-6sfu(8shohlClbFn?QaVB$A)xW`i^4zwb2vwq*XAbPrI)J`lEOlLrf<m3{dtfaY
zB}F|d)>AbkvPn@9jV)z()qQ%l1d*hYR9kgj`x||CPG$4tr!bPR(X})AF%rZ6<IM+z
z<PTb{wKa;fT;9T+muEG34Ro2ArxHC=h#8r|&>cu(00P5;K7f46nU-f2UZ`q+rEs!2
z5uHVhLgvk<=XK<h>H1(aZjq+{0HrgUEZUl?O7D!z(=?utvWY}s7@1Y?YcY+G6S|Z)
zQ*HMAyEw|4hMzFYmO8YMhl)iv(S7(Gz~ycY?{l_1<42nR00YeOs%h$|+LJV^R=9mq
zd_=LL$hTX73!)M)$Tl|z7O(z)u62b5W-<)7FUeLadBl3FnG?;MU#W(e-3ViIuu*Nz
zuYlL;=%`u<qIeBj)iumTEU`j_e=uRTtAn`hzWX$*q^qK$K{jES#?ebDT8etv5u#}{
z0G4)O$e~D7vjM5U#fleIW}ZDkl)+rNCP4$p%4UIMQzOc<2!IF+3l?>MP|i@=<muo(
z>E6Ea-n%EN$l~eRdODhEY9^=5qK1~7v|uqY1}?s2Wrme)t_ULF078E=%KTNDN@}dD
zFsIEZ=5aI7=CG_1H7FJ$2aeAnu)Tz2n*c3)V~6bz;WtL)S%of3NOMfaXiQU9W)!Ig
znF8sNY2jj6t<I}odvbSw3q$I<ZocY%pI)HxTdQHqqNZOlprM0RN~w^Qhs|~%T-f?S
z`rCY9w2b+-aa9w=nMp?qpE94!Y331+LL;zSUhJb*#C{M>#lzFyAN92tQ={kAQf2j@
z4^qelI=n1|X#z%%)uU@2m~42(hO&34sGd0^c^;+0$febqSvlN)n^<~{y9?X3@tycf
z(AB*G7Hw5qNlgtddn7YQOALlpr-*AJH`Tb17XxFqg*Q0Ws;ix9Xw3@5wKRccSkYEM
zWkLaVcfE-qxd&m$9q<fkV$HJEPbNhxJ2XlonPszB*{r*8KnCNzuWMoCl+Bk(@$e~V
zh+i?}40Sr%_G=4&f=2rra0mkUe@oMAK3`p3HgiiyAdNhjr7EgUpv3`R0<w*}X%^sO
z7Lo>#HC;*>o*nh;WGL3TvX8_K%G>4GZGch(TTmyi%T}T@8BK1Ui1ij?3u-nS+hetd
zZg89BVx`HiovLJvDh66cAx4mN`gYip_2<4MYAUNLsVbv?%mrmcb&Rv6{{RS)V>&}v
z=@!y=H?Ups;)a5Ts+><H0#ueK*7^tN2_z2oHaqcvxmQ<+qe@KBJxmX!9yFYS6aWG%
zwZPun*nzmV7*ct`Dk!IsQYqmp3`1Kh>9*GFqzn3#@IJUK^wmmMn!cI5%0)<Yvje5Z
z<8inf_Vycp>@ZkhgjG|S3o(Som-vQIqQ>d~xzlmT;@AdoD59mAYN_5vjhD+2*o|ZH
z4O*>lm<IOz@ij`hwM{zBB(-T$vNSRXpYECxoGpd+3~%c0EIpN~YIrImaz5=HbcSbW
z-H6y)!1_r7-R*IFI?AbpQA+3v%^)O+8qr9M02>j1bpvt_HavTUU@dJ86Gon7jIK1O
zvHT$ExVY4K7CUi%0K<tZ3soj@J4sbjB(eq)BfAZ*&2TS%NVT`I0Bwb8>m;X)#7fn+
zr)F?=02^6J_uSl%VmsY9b6Mk|YDcM5dWm)U0FbbPK_u);*>7=s-qye(5lvgw<?09p
z*Y6X5Z&k+H#fZ{Hu5WRDw!vYiXr4)Al_e3eP^t;u{1&yi0E=9HxGVJ)UTG}SW)n0p
zMLPLHD6Fcw?sTp817CA<arD7*?7mo(y-h^Qnv0_nq^Ksq9Y(-hjn2a3i(n=OUsfWg
ztB#GJNyDzU^#a4xRa(RnZO6GB_l(S8k(f_^2Bq>tK&};w1F4uN%eWx#Zg1K2xxlNT
zih`nPYAK*MPrvmeH~`+(QNMoT+mX0t)h{)qXvA>ErI93EORCp4BVZ2}z3*du1Z4DP
zx^)!m1q72ZAV|deSl+_I>~|Nq@5tN%OGQkT62x?erm|>~GD~S<Ln|-`djo47j{Dkw
z#*&~z@Kv?)q9RlQtf^#Gu&_3}SOaSSPQy>W5=_nI@}8nfh@*{_#6wsysohF8u2`Gc
z+hNJTOTAc<k~#*<fQ4fqG}cyR^;%7e?oG|k_EiI5FBu&TMJ)u3iyZNJ=p+tUCAEWM
zNCMYeo12}s+X+WW4r4s%UTBdDkGxfX*6zd+wY?-MxEHa**k+Q3hD^FjnH{E{)e76D
zkfOkrPyr^^x3M2Q0qMGif|fekM};Pk$^QUv<qHHCI@GBa-ou95j6=#P^BTA*qN%4!
zii(jOfzt8Q?QwS8k;t&PAOpRwC1-@wz~!luifAHMRzy>4ZZ(og1l)75AYRyihN8=;
zswpccIn6px6WTb1fKt!u3dGv`jrKf?k5&Mc{{X|GqBQTBREp|KlCa4bA~hO~?n2tY
z=oYx$$pGvzC7Du7m*tsEuvS4VjT=Z}BV=vc)EJ9gTw4C1000xTZA7I+d2L(F%}FDQ
zZ3=w7Ojxv*wVKQaJ6IcIlT+1yR?^8Vv8qcdl78*2=~TVgi!TRz=^9(`i(np0JziwO
zrg>XcOof^>vLpnt4eQxd5_#=mdvSwOMf^y%xz;+QeIh9in9u}oSNK5%ZO`H>vEIw{
zwqHp@nZ^GAP|?;$POb2-<w`vh1yg=(K=TKvn+ub*jhoWPTGPoqwUq#WRO*49&`GXV
zVRNXHaq1qf_dEk)3AG(ELku%XB>r16%7$N9I=w+n+^vYdpaDCL_rYnQsG+BNd8+`?
z8pN{}DzdBCY|H>HzT;wdAXuA8RMk=nS_)T!nh9x)o@~(Ew2N{SmH=H%!8-14Hy+5=
zw0Vs)Lp4>Bw_H-006@Zj+v+#43@^X6fIEx_V626F(w<JEWb&hof@L5r>^1;<_ZI+M
zVm7vtmPcvcw8cDn+M1#fsY8x0dlI|uq+E;K4X}l8#AR7?Hd~j}*TVv%Q_~qA;&oiz
z5t)aXfBl%S9ouCN`d>Gssy=NNc8wh<o;s+YeRBnn>6Rx#JtWwFk5!v*U@QpqzGWtL
z1x#-e(pAw2;iwZx+F4mnnWDR!qgjgGZVkf?u4L+lYe1o?;f{Dh2MZ|j<W%EGwe_hM
zJPTY4+}jcPo?Vt!W)J)>t}MQ$o(EcNz``eZFJMb3)E#}$`&z-QaSYv2o%H08(`VT%
zglO{WvuaRUo<!Myi#%ZY(XV^xV5D<k2nk9aqNd5B9Wh&$3VK;*SIeuarSpDGOL*Lb
zJDomYQdK|)QrOrkX4jHQB{b2ZQL&ILvRRtz>av}IV_--Jk?e5+Mf@YFa)h2prOaq+
zRJf8iQv)bsFQ^Vd2IG)*aKmL{>Q8~KgD%ab7cu-i754If;b24ganu&FM&E=7CrgoH
z6}KThOO{1ZLY4Hjam2*=EfnFyCfC%*NwvqTo(Ft2>H10QYFJaGf$85+tVz9bfGjQ;
z-0VRcb|Y*WpEJyoS*iofiB?IMNeQ!Zg5c>E;P<((=I?TBwzj&q6jhPDi3=7QOsyFt
za6mS`+uOH)q#cM&Pn}g%!kJuaOE`oNze8nxK!#CssM~G(9(Lap{KqQHJoJ)y;f-pe
z*BCJ900V2GBS|`iy(Yu*Hot!Z%aVC0YbvIKoLQ7YHN5OW2Xbt?`bn_l?ZLzjYV?aJ
zrl!kV(I=A>l1SIJ_qC0=ZHT$H_y`zsy3v=@!$g{Y>Sxy&Rb@jW-ABFGbYrZLET;P$
z7Mhl&B$QN39VBj}%+`&jAVt6mqz%;H(h1|eu=1XcuC{N6%W104m|N2x_X}%)+!8Op
z*5>;R7AlGBDg`}TQ$ZX|{$;^gEpFPjBoaU%lex8k76i(}QzUa$SH!B0R{l$siBp?e
z!?@mm*S?ZX#tD}BH2zn?OXfo~<(RZ8?FEPdRNK_HXB^m_$hf)XbXAqrl4bcVJ#59(
zOphAtX53wU?Q_7mECI2;#K@}U%V>;n>sYh|mOvHC-az02K_hUxDFoaCY)HUOX8kB=
z)n%)hWFQG72`MF6m(yYnl6@p@Ir`!@qbr_&RhDUIRRR{4F&J3{x>RWZ1L?U_dy9*E
zn`4!EtwwVTLvaigh>0nYh2**XK$0$dceu5I<R3Dpq{#BBTD<bJ_+&Fl8psvTnCReK
zEMOKoR2!93U`aN>HS%A`DyHZnX&NSXF7j#wl?!zS+nvVC?|T|EBy|%`D&TVRnoVqf
z`kY?sTUi9@EN*z`oHjMtT}?a{P}!|#NfL75(Ubr!u-qN)4}SP(Ln~A~LRwW=#X<uN
zlW7vLBIwM0J6*Z8u1*56<DOBM!Tch*rw=7Gl2Pk{imZNOD+wE$gXNu$<eTg^K5pvD
ze3LJ$hMnb(K$QwAie6W<HNkPKb8WVNu_`ymsk}wX02x5$zlTuthGkC#vMqM6ii&S6
zg)$?nH$3`0z&|TrV~-^DNANi-DQae(K|AY5i59|Ax)2h3e>V5nt*{VTy&XhzPgjvo
zQYf`Jj%ie(l|Kjy2HRZvM;|O)d_JqA%d4Tx^L(kKkx=>sRx(Ib3yo)++V=c^zDe}0
zRZPj5PfHX(4**+4TIj0xTaC}G5*ol8{@Aov^yIV>wD2rZ)QYA+B~k*$8Nn=0*Wlac
zE#ARi>f>7$e$-X@Jd~1Axt3^J!%5o8z*@u;+?~j{+ZEb+mzH_qcM-9XmAuCMBVF|i
z_ct~e(zC3T(`FLqRg}##fRf37F_0*c*y^)f8{L2`N8te9+ZmQ!TT_<i(;$K)RZlJ3
zSyxszxjPaFzZNIo0eq7sd3-Kx&0ttwlBP+Pp{~+>-JUg62FgDc-~)E&Yhlk&Rb~`<
z?qAd`1x!?xFhKD;Mtr57-l)hRkmXeD7XIUNhybEMUB3*Is3eYopsKH^s1Gonf1;eI
zRF33}0uQR=+ScQ8F=FeEpQXxjItjiURl`p#brg}uD=G476;=D+X@T#x#~ffczX<w)
zyg{O`r)tS_JlclU?JtzkG_W$W6*@@Pz$vpX?6+=lXr`@Ndbz6>DtY5E$@g+dr7dIa
z^8;c#f!`id@XM~@=<Mn&;)guUe_#eN8yZ*j(=y4i3|h_<2;6QiKwbdHq_PbD4x-gr
zD$*HQHB<ypwC<&DuNX@OS0pggYaM{z*a{bo*@*uDr77~9#(`Q2%GP-*EC-fV8p<qA
zl5X}Q*3v#$rv01FRYTNy)>&Uu8&sI%N_pdnOU#RNU`>UOp^eDdM|)#vf9k%JqwCI_
zlP381X`jtjq`t_q{{Sl$C)5cnx=-?(-u5HP*{mH)(7D3aP-L~#Q)RNXEnLx~5#}FM
zY8;PHIvK+3bntjM4_9WD)U@(TK4Vm4K0-??%Q%PtLAA!#*l}z6t;Xck{Xvs5OHCGU
zL}h@8?<Dbwl-v;DF}?4-e}%M;LB$KA`r@0!3JED{vc?i++F)5{aOPDB84T{;uQ8un
z8*p~o;A2$D@(h}u8o6?;u9+jMThu{1y+lZaSP>B@i@y9@oq^m1c7Kbi4^>Sn$pBVp
zWk*>RwF?kRu?FFe{{S)#u=;AU;MCHY?7?K3N0U+17WXzMP+xLJf5&mQGwjPR=xnn%
z>MXyi@_H&9&n=QAp{L3m)5R?_ppgmkwED>;zOpP&q-;hi9X;Z<anYShl=YP{N0&g$
zQ5<y={6mB-vKL}m*fFra;dUm%_zldkr*)~#CJcfHNTX(SXSiSqD|>8LkT@f2?&5<l
z5=SJ@mPkahGpLZZf2C8v++seDud*KzD5T9YYADvFqh^(1OT~7uW+2|i#8?f<1GvKD
zEvl7PY6$>>L3R7tRQqlN0o&*GI1LLkjj1z9XQC?tBO5ewU1B79l$K+t*s#-k-{uY(
zp`B{tX|<!E^xiNVMS&*dcCffU_rjIb&{NK)8kZ`_vawiNf6C_GkV!43#P9oI307)*
z%j6da(pOjE2iy^3&&Z#q_zAVBf<*Yh8_M#5LZ^SjYuejxPaALZ!+CQ9O{!_w3UwZa
z1nJZGt$%)S-kUKoPb`Y0YeEBS*LNxDu;Y6WZMBaB1fz~RgB*2LEPy4E6*Puz$lrr+
zb8}<zz!Nmjf10l`r_8e0)h3O;?di7AE<XSy>D!ZI#rN#Nsj2GKK}e)}g(0PtU4k`@
zkZ*h2eaW@?!r@VcM6Bqx43X<U`0NX4xhHLnfbK9UD~$|`JkimJW{kJ}!s{FYSX%mA
zZEJS8?|^CWveG=0%cX=4v3X*}h_>d$9lm3Z*1{xDe<SInV<>B*L*{z58v;nVBIJ$O
z3z71<m#3ts>YB+7R%-<eG1JKHB&sbcmNHosY@+0CVaWpHn3SeTWN0KS3&$IVk}}Bh
zY)B^Cwa=#37TWj+lTVe!mrAiDPRzFt7-m=qx!Hg_pGdv*_qn;UWhxSOhDjPY<(WX0
z;Uy$ae@IX+4<mb%^Y?1GJv40<Tp5Jq)e<CuBPH|`q*~-DJA3VJ_P!)&^Esk|DykU-
zb!y=jO@D!ol{UE3vC>H=`Hvf180M6vB$*mB6pLk$gSlIivi$9TdtZ};=;orRnd#8$
z2_~~4LaQpDNRHo#5FXmj{Cks!_4NXIs%mQ9e|ar5kcXYOTN7g1wmSzG0041YshH3Q
zh9ghvv$<#8)s2OTBXDd*h5Pot0cfcnf>+G{0EnbAzL<QHzj{{XNbUi!AP_g-j5rn5
ziwab<-xD=LvN;NM6+#ArEnsw!?rm?D9-3-e)*_Okwwjh=0G4C|K;Z#52pGEqaJ*jQ
ze|wxq%ARb==e<Qpgt(B~NRc_z03@4&KNZdJ4Wp=}mUosYeAOBoMzDz`?RC+%_uEJZ
zlf8xTSZXRz)K*i;9Iqs83d65;O&|FaU^WK8Zh0g5jvd$3QMO*O(o(5rm(GfLj;9H0
z1Oon|8}R}DEAN8l9VwWuLqnVV6fH?De@B!|A9%(>y8;gD#l^qHz5$hOX%kY-Pe@W4
zML~Ggf*=U0&RLGD>taDB#^Y{OmO6^5D&H?jCU~WlBazEp;!<u@lfCrWc;D1*Y;vMn
zs%lDlahd!)Y$FV+%;U?rJCS>`>EQ8k^CCwfTG?8%u9lveg(^t$q7x`s)jc7Re^+H@
zKZ@jP-rHP+v~%VetTj#~mMLk$x=FX2u)Vc!du(sTu6eMGtgND>t&L#IMu}h))gmX=
z8<IxBM%LAC*0^hNhEdkk%~SDAqLt7zlYy<!GTlL70XvazrLdJuFE^C0ADdG&$|jKr
zM{uN&6k5Xj3y@D1=EMs%RYbyKf9{Xv74r(FkOhr~*R{=n0FZYa;&&^~vYhUEoTfay
zPh5zxNR-d!H8E5GvjJc&v9l08?|ea5u3br6QPO$Ar-Yc6XQ!Mf45r~p2X+@HeTcdG
zVa8ikO_xwo(PT9+Mu?;pjWkCE-G!8n0bWlVoA3?-J*zV*WRgmZwn|5Ze>~Xy>4GR|
z!P0ir2)d36ld$i64F3S5`hPWw9M2}qr^_>^Z#sI4I%SSk{41hIlUQWr+4lX0FY)9w
zb#w?PtV(ot`}3em2qSmq<biSk=WX!<t(Vg0l~iw^OYr3-nifdON{g3LuemGfW*e3u
zH#=JaS%)#tGJhtAJ)oAje?tCY&gh^71k!Kk(^!=k?uEys+zWw)=_md<B|dZ0x$(_1
zy0jj2YN}YKuoy;IUN_UehO&`+0m;xS#RXJtS(i{pBB6>DR3UaCgaxg0sP_P#_S*Q)
z^L~ZN^9<6nG|A`9a(bgs($`jH(i8wl1+@YhY@q4f777mcCg9Syf1$c-FwJZ8PKD|-
zlIE1hdWtK8M4HPjL^9kkH($n~uW%aR;pGIKL6Ky&G+lp`<(bYxK}x#Zl&tdwZd!OO
z$4OEH=+r#~@HU32@S`%VsDaOjS^Y&qw3KmqQ^N~+dP=hs<N~3Mt#U}SZF}OEnD|lD
z6f;w&k^FgnA|!gae{)Kdt@1=c5e~gkS!|a&Z>B!lfB~>jd}cvUNmJ2T>SuWuk|&~}
z%xVKm3S8Jy*IAej!DJFBHv+`mPjt^)WO=0Z{zh<DH=$s)`CP50qJ{1ZnxkR5vk~`;
zc=!BU*x}h%O!%|mjzpB*Em@l)N>)cosyOQD1ccmqFLro`e>Waqs=%HA4J7e1G0k%r
zEBb>Y$a8GQDJF%p9Ir82Nmo=}sDPLYmTg*;ix4b77N8!Urm2Q%gpzn}q;8KfP~;2l
zZ(=M6u(A2!t#unxLgXw@PnfVs^s6+H0BqV+8>wOsxF>sd#(mIVXdaV#cpIo=OuWjP
z7Fvjlqa7!Ae+mgubdi0*CtxkL(<bUXpEim*yr(vy%@S9<X$3-STX%FLSh-{PhTtD-
zU^w%|NUfnNV`~&wN3$Y^H(LvGp>J+5*vQng(q=yl$R(Fr%pwZXBJ9kX_9R~CzoZk+
z997XpBuv7fw2+|?=(|Ozw{m#0fo-qGB1|w(O|Bste?;&1k$DGI!U5{jadJVh7QLH*
zMgfaE(aygGTHsD~IA*zGN8l#gdlnW4e~>O|GMeE^w3Rgwih7^FiG$e3Vm%}d;DKv!
zZhow)>nb$IRZkM1Gg(K_7tAEs6;MW=NYiTzpKK~=CV|8`hPIAK)lgs3z-cOf#CF^O
z1-&P@e*_dsO2}$so(W7+EYBQhF0M8TZEY>`9nP-f8SmOlBI=&G$Y?VRrlO^4ayj+N
zzL5lU@@&t&!3tGP&jVHaV?nK}%csrdoYKsNWhOXcK+)+^KpM%t__n}a!rKejW0NYA
zFM4@ltA<b?L={j29N~cuhfx+9Kvp)~4lo{ifACkYGW@BV!ussWI_jvQEkju*hez~Y
zSns}_C(J5uy0x||W4Bb~vSs<H%5upe%~GgEQ}Gny>1P+R-`Kbtllt2pB=G~Hv(E_p
zL5><)rj6l|YUrhbU6V^ODiF1=X1d#cH#_6l9Szi(pIUV77g5tkNkv^f8#P=s`m{jN
zf2a-_Z?FtQ1HHk$jm83V_&r0cwA8SbdU+k{7NNAsWz-vT#^FG)1N<kRapJEXdQzvR
zyhf;@%d*O)%PQkD(n1ul^2-)*wj2g6zCDpinP+u!K1vLeQp{E4L46KX7DKr9az8#d
z##i>TSZjJ8EvzyMTE(B?Wo2-NYG~C0f5w&oJ&89T`VGn70p^^RqAasCN@_~^=Ai*v
zN|?c4EzX}%Q(|<JrpEUo{l+$G*{7Q)qMI|OnmVV3N!B^yv5;#lBs}e;YQF^g_Q(2b
zD>6qd1qzypz9HFLSU^!xUs1WTkOOc}HW%-uRpzx-^f5=9*NUo|S|5j!W+Ztbe@)DK
zn%c=XJ8l86HUyBE&brby$rB)oj6}q>OtT~!ogd03IM|)63ZK=E{0vf}I(mUuRZCFE
z&!}BDk4W_qds~gSJB^R!jUu|98X6Vq>eC4Tm?l1HlUq!t)&peKx{dw!8)00(r3X-D
zH1!#M4K+O^bdOn8QXNLP;Q%=Wf4`ej1st2WBUSBys^pZJTyRuWM(;72IDrz2C_K5_
zy}=#!>`B{S$U349wCQ#5_M;-BHLH-+WNB9F9JZ>79VQAb>r%qx-`}vtYe@&hP-fY_
zWdy$#(@638+Vsr9fnw|qkbelc9myc<F?8sxwz_QhBBG|ImO85BS)z%<e_au$k$knY
z7SnP!Z&5OLz*0Id!``&(IH&1co2Kg9Fw3JcJOL6igIfZu7y{C2u+lwd*0JrVcx|0e
zLt1jzSEq?*_&S<&I-?sOE!Is;4VjAC{94ukTb?KQnU>4avUG$7mJXv5nDacvB50bT
zHekyes$EsJA6ozkAm8yEe?@eCZF&i_o~Wd!N(rHnBdCg{B5^82L}?o}forNL1Zp<0
zz;6|qdq9;Au>Ha+$t>g#EDilP0C9Vhu-N0wUNmHMT|d=Q=h<@Wqs@e#Q!KuG5!Mr>
z++6xB8tZaY6L4*?-1F}fvJBgzrRn6%%+$G!5lVTf)Jahr1k@UJe^d8G88w|c>@0DU
zboWSg?^|@Pd67*`MViQ}b2>kVo*5j}#<nPs+({{PI!FtvUW0H=@D$$;DyHbJxu(cE
zjzLufwNFJd(MZ)2YLToYjbndZ$}OxnTX!mPy7kA}HzuX&YOJrP+L4he5|XB(YsWp_
zIX8+jZ(vcBy@*nHe^Nm{VCw3Ka-70!t~aYm{KDxqd}bP`G?q}3M5Kgbq!KkT762aD
zxp;TsuU7brTN<IKhc3HRR@KSt@>Izk-&1;;MkH8lPU8C<1jkJDzg&2gmQ7vL$>hUO
z(npw9)f+5K&Ar%<y>oq(T!Ugr=Nbn|_#4r=8>IQIGj$zuf5-Qe)~>jsazF)Q2TJMR
z@fIS)@r^Sr$tXH!CK)zKQ3P~!P&1e%eNry=vl0L-+PhxD);Ab$l~cu0sv0_VtZ53O
zsPh4ggR!;kz_)NgC!K&xEFw6msInR*G|@yAonx}bbd5}OfKUbFZgv;j3(#d5#Z)BF
zOpOd=AS|RGf5H#8{PTtsmDRaJR(u=IB}j}jYjy#E`^yDVYz@e;_xbNV(-~C;WtFW|
znW|H>IUZPqy{rd&?tT7uz)#oaFjZ93l#y7`!lLsBWh49}n=OvVZvOx)##f+1^z^js
zzY@zZV#dNIUDd^py@BM8+usrytra{2%BPf6LmU83f6{^)H`sasJAr+yJ8jPG0*EB?
zXPy*I0vRL|1Lgy8!;{Ugs9x6td*C9YFH1L?p_xc6tIGuGAYYDl2XA|EhfwF`1Zz(R
znj;FvRgn5vZUG|3=Jvh)g~kwix|(*D7-Lj+lUNdxK#UK=ZUE<jdy8CSwJ@1!n=ljW
ziNc8tf30`8KuG`{S_cH*Yy7Yds`G&D<j9)aO~;oa-I;+Z2)MAa-0#8T4BaM<l4`a|
zB=Tf2HPD#h7GtD>Y^LqCyXyDHbkWD1!SM4LDcRG`Xld1S@1&)zru>_Nc{e2PB`7GL
zG^nh~X)CH%Ei)REEX1~@U4s$a-LJL+B5CSrf5cNHhLF!rV=>0;UDThEuo}NY0qii#
zE>voG>sgpd8W&(>Z6j+BP!|Trx#auYVtw-J$mW+Pqo#_oR*Bb3t(<oR6Jy4Ui`W~R
z9x)kAVvj5|vdpf`#D#52#3>}$=>Xq=Ik_BfU~KiV+DQHg8ka!K&Mq`+)EKxo)-7uQ
ze^%BW_jwf*^+05FOsxjLEn$gSk(k^Ey8vx_+f|LNZLmmb)g}J`x_654w9Dp1F8l%-
z$6=>d`|-y3j;_u=A}JuJjUrgs`craEyWihPzWeYwwgJSmc~G3PTQP8|!b!tfLuz9h
z+ep6m(tCWZhhA1$Q<6mlG=*n?g!8m8e@`um^sxZk006%GTG%ZmK%~iPX%>Mg-!W*s
z=^IX!Mbve1?oRs~0nPCrBoj?eYN+P;bcum8fuS|&EpISwuF7n@>}_%dfk`S?&sk!r
z5Wby~DM*S3U=TK;a#ZqcFJs?p2cEV{N*I=k7^x-mBx%4Y<{RCYQiG^&I0DD8f4=5v
zB#J6pDrYb>%F-F-HZ07b1oYf4!r?`U1YYBENGRy<Olo3aZqi*_TN?{(2?Ez3Ud67B
zxBvnOSJqS2)IDZfQBxf(vNgq1`lABcbuiekI}dA+Hp6IPoYS;l#A1dTi6a-Pm(i#x
z3dC}KH?tl1`eA9z(o`7bWsoeve`TIF3Zxr_xlma5^r$x^i+j15A)~0NYRU+HCE-{M
zY5`=i7hAAns_F&z;Ep%}mE|{)qoBzuWT_#HDHsHRz=SP(f;~a4@81heLs3PPhFWX1
zo|yv5Lo`>|XkSRVHzXf_d^^heh8&A8sgWXo;<W$@X-g8GP;M04OAWr4e<We9XPH%I
zD<rQoL}p}2(lA&kH}oFYAYS|ruK^sC^qy)23KCKz(-Ue1b|*}1b{{4n65Ad?d7D*b
zGR7;Y<6NbJs9JcUa$fs~R=@a^*-5>?-)uFOIP#%SDTz<smJ-SowxeP$pl+b5n=Quw
z0Fi8^vt?wZiA40V%I)iDfAp%^iv=f5*7nk>J6lmF0Iqq^-7THFRCPsOS6@?G95crp
zbfITZ_dZ{paiNCAfbK!$oL2{h*=au&)3q7)StT$cnd$Q>2?=w33dObB=Y9xXwi{2=
zJJlJgJ{&|hyiKOsHEv1v*jO!rBn$D0Ji9ihrp#q`dX`X7MIuH_e{NY6fJOJPQoqc8
zcnS`Bz9{r$QP<Mut<_W#t4k$4Oady1)lc&<DjQ}bk)-an9NG&h@%Jy5t#bVHr}KQi
z2&QKRVlN3t;K-)N`vIr|NEi8IP#s;<ne|k%<%On^#Hmn~Ni0<lQRRlvB5`&KN}dXx
zUAsr-*#<{lM^6e*e*}|H0|&h{PVtMI0p_a!RI?GR8(eG-0xq}3EP@z`sL$Z74=PV0
zQ_N=27~Ciz?rnd?V132LEIBWTJtUGfURTvHQ&pOXr=KsKNL8mNTDe&*tZBBzP5B_=
zfz^HucypWd_@v6C&8V|IZ&4)gHR3fE(mcrY*bCm^TVQzaf0I`DA>xYA(r0;LNpos>
z%w{1}l2lvF)RQX<qNxB9y7d+;H0}WpQA^bn{$*WT<+-g)vl%3gN=kK9xgeGy%CQ<X
z9Uy7B><I!pAn_}vYCnqV&11`@qDOgE*UFPXUG>KxQa4gBW3qw?zMN!~o;CHyL05b~
zh`lSBp{b$rf2mR+zFLIRS%YZx7FN<&FaX$-d*Zp)A80=e`hjU`rsxWqoZ^>MYduw9
zWQ9wHT_FyWadl!WcsKSBQow^VRAD0+<w)g-BB&CfBT2aA0lu#G!(_<1e$eI1lhgSt
zQu7ErwJ~ceq_K{{Ljcy*Yx=E^GUYxf^=z4?PFvC6e;bpEsbN|Q3Tl^@o*}iEyO(Ar
z%)1*7!9}kp7usv7G8Utv&0vAF$gMf}RM#_>)K*<%P%LgsZZD^jNdf3uj|lubhFYBB
zi>GO6C#+3T%L6QGmUg?C$T6#uK=lg{e@w@!JRIsnUsIU$wK0ZTI-fBs=^i?3INYN`
zc01p7e><C!NU+4`MSY-pW34Btf_A3LC~2V{OU)nv%;1eOH~?zgkOs!gZZ^g7GYBbX
zidL0a5+Y#Il~gttzN-ye4X$_=z*Z=HP|vzsr#=g;@(Sv@*dNLh#Y-qzT-<5d3}<_9
zaBN1`Ck-;+6}&m>3Yuv0KBJaN3qsS<d_I+fe@)9fG=*buEn(^xxc4I#3cj}K-k!{<
zs5+OZsi2BLDwxSoJ#hQ3_hVqdi`<R3;}z=c^TEv8GV{klQyocZpsGqlj!9!@T_8CC
zjnr&N8(82iG*nU6WpU?F)k)<e$n?=h%B3dL>}{*?FeP@mu)f5^LsvCywKCTaz5qm4
ze@+l=NjKPSdwZK5ueKxCT~Xon47Bu}CkA;>QVg!LiuR5z2*8tRSgQ>;{45&aTe!px
zSB%+(W>(p5bC4!!MKn`1G1X3L2x3T#ojQzQd2HG?EVe&}&4Bb;LgI)e6>*XplqC60
ztyNV205H9d_HC|r7@5kY64u8iV<fcee<W<Wq-Jys6xa=}EK2-^_5>-vnYv@eOy4Y8
zT%#kC;U<|a5X-5wd5v%h(u03en}!{SByVhw#jd%BF@@ov%$`WgYLw+H(dtl)qy}B>
zq>?uTkOw#k@3kjQ%hX;RQ}rDrE|r}>J54m`EM<a1u}TLUsWGu0#GWyp{g-;Ne<<^;
z*QDzt4w-~Tx|ymZ-Q`6BMDM#H2mb)T-`f*COXD9>^)5?S1Ep4QD^_Y+s#qvP$r@?U
z?8Q>*8HfhpTGkt59`NR9hHnL1X6cj9M`@y*Mq5Mc3t)NNl?Rc#b|et?MJl~ZC}A{J
z5y5jK-o_HzZlv-Q=mY>P8EhCFe^k6`$msg3!<xLc<Z1IPqK0bfaU$+k6xPwB4^dMc
zfl>er!`x#pZxZ}YtPc)FndaH8G)1LZDcRkTcDtRuDo6t7<?V*Bc&FCrzr^by&2t(l
zVt8bdD4s<~-C0-{w;DBiNwW(BYa0eFmDIAcB|Ao7Ys(aDq=45|y~*0cf4%W{Ws5U0
zWO-!NHFB*wGRPFf>i4sw7B*(}wb-7+fViyWk&4zzGfs#Ml0hPvtW>xqOKBDt;0`(V
z#m}T)DW;BjDd=RVt)`xLpExef1aYKJsRe@hY|=O~7AJrJ;NT>)jH;ryp)yKUV=S^}
z5z5i1+`?dT+L^f;8I8X^fA_`b!w(EAbIJzkI4G**mo}DICaG5`6jX*V?Do^8Ln7N#
ze1HpWSd!_yx2Njsx^*6=j$u&)G?bMZ>{NwR7drw}kQndsw)ojP2D_)cHLU6?3T)0;
zGKGX@5?0kQnmn>7VK!Gm_l#M?AOI<{kP93Kob&Gwv+juM1*Xo@f1Zw-Kw6dvBw3)>
z8jGp1Wl~8=8BXU_`-|vN$D8$sP1JSo@d_$4)AiKUwd+QjW{k*PM4-mQNht+A!h>+B
zb8tXq<A!|wL)VdIl$CXr@<BXnm%x;|T3G5KMvgW_1-!r=6<Dg04Ugd$m41=TD10ld
z>NqMD>NBT`rCb>je^`DU{E4ESoq~o9IUz_aM3w+pTr*D*hEtaHRRvvD4LsBtnlv=^
z6$&Ewb`GRGtOd|$H?TJtb<*A$N#g}T%KA&J^6a`0tx**ob5krHY{vJF7gS{pYY-Ux
zuipgCa;(2T>4}z4T$x&}Xhe(7)T?!HNAn9E?Z2dau}Gq?f6e+vI&Q9{ny#ve<@>oK
zl<0Z2Y%EQ{DlTp>aH7}?UsZTD;)hFD!;t0`ba7_p;!0_BM;z)-<IN1n<<zb2K<sQm
z#T)*p>W+z)uOh9<<)o`12aZ`pVwuf|l|}U2s@NOu0`>zNR!QS>b&gcIXHvSni5Mbh
zI^F_U5K)fdf4>%{`P+km)|n?t_|KPA^u!sww7LFVJo;(ohs*eg*6NKauAsoPIN6w{
z+yiU|b=0*o_@mRgT$z!Yw>in9MT$y|D^$@0)#MjY8KWwLMy58owvl~Te+oK>Dw2k~
zEz7c6e3(<U6(rvgIC-k7R7g~Mg0NNu6-!$8xbARze;Vw^rL$VBw>^-ow56j_O{A={
z#?}m25w|uZacpZo8qLzZThuh!hHCLenp0FIblz+L!7KT4plMX80o1qw(OBvOYm&e>
zS!iLRs}%W=N_yx_de5p$$V#y%NwCtTi359&V}++Q@Suv0Mym4JRLwOwhCsSTr2sLx
z1h7(1e*sN}?S)c;omMKLAtI$rvPjHBDCbJCHfA7$a1SF4Jm9ij#A_3{Yli`mHI?4P
z3xY3g_yg()y-hR}g=ed*rZTxD)k_0lMwI~WKq^5$;^%B99B)>FXX-jy7U7|3rM{ub
z*jsac2_HN=dD@Z&sgj+NoEadGN!ep&U`Qv}e;ZoF9{Z5ENK@uAlQb0aRV;*(<_lQx
zM*`qmW4_kk0d<CFbYz66D`iyk7|AFA>NmZH;9ld}zzjH`G*VE>C!ghnjznKpfgw|U
z3P?9&w!rdj#jx6zS(-`VW-ONz$k7i-eK%`?+V=T#V{AjbGfg7Wm)7C*ATOs&vwtu=
ze+!#f*l&CQs#)W^)hUV@RLFsfmjhOjsCGB)^ftFL)bAAahP7$psU;neJfNv1!3Mxs
zTI3U_W5(AwWmQ#8PEW)<@>D9dqGEKQHa6Jrwa2xsxfp6{WXhmtYI$H}2*x?$WivwR
z2yh6n2E>EGup{OQRz*<&nxd$wNflize?fJz*+A?`H@7DD?|fLR^U7ADq^6*nD8hx0
z@CFTK9%6=Jvs?>y8(R>aHRs7wB(fxKtplq{5ss0<mf5{WWfrjAcN^oGnG0JSX%#zD
zN(u*#P(l!QC9G_AH@3~T{@|q>RZk^Fai~|8SWAfm5L^``006lodz*o9Io}6Le<`O*
zIOB+;ykx@6Lotl)sYR`EcI-ctj6)?X)N;J)&XGo}P>@20Hn3A+y@jsbhtms3n8`>M
zvP}|#z(T6JMbzzau?pLc$NB(wS(zG{Cdtx*D5Zolp=l1eboU2V%trS8jfNp8DQRY?
zc4m~xuvK`A$suFSl-!UmZhuX%f4*dC<eFLaUG*7IM;N<Yk3c)wzIHYMZQBRSx|1;J
z`kD;aEv^YtsWP-7!DJTFx7G-`7q{Xg)vyNuYMj=BT~u{ZB+i3Gazim>dt47Nt)9dJ
zb_0$8;k;DPk0O~X>Z&1vpU#zE?$#*<*_A*eZOFdlhrc5Z=;V&L1eO?@f2E&Xh>fL#
z2K7CNwaxFg+w1`0v=tG;m6=t3Q%~lzXf-9+l;`m+_q$jEEo1Mnz&>osIjNzg6vzY$
zWpNlR+Q=3zPM}ukFR<kCu*7Xr&HO@4%i;v4N`jIzq0-IfY}!qaVnDZT#`eNZOHma(
zbq)-a^=#-~G37?iwzjJcf55sCa4t{2_y@`{PcufKPxu}NR(6u)Yb0_kwy-0AsklCA
z;PYsyB&4aGi<3@?3EY_p3$YA9B-jD51AB0`X#}xms4AXXBv@9#p>0f-EEw!=ZpQ3w
z#@yjNxfl4HRJG34QbNV#@?;CGEn#T@1y82rfp5&4u&I+VT3J@NfA0{MD9tt4T#ra5
z+JGBx!S8@mG>iCbaluPd=SmPoB##T)GCvv+;{K8CU=6zfA3)&0Vpl7ps;r<DQ^6!8
z>(MEVi?)qEs}rKffB**I^R^)>ekxH{P|=w~S<H$`gp{q#`PhI!;{EtIoXWa~FwLu|
zX{7%EyfjJ?B2+$He~3z{V#9VNy*q8`1G&I5&Z(uVFv(3j#R@8<#rBI@mMm9K;U3n%
zef#1eDj=!REVOZzjL3Y@bdzzq-|M}<_Tv*djb(hbMg)dPon(>bF}|qPL9k0&>*^NO
zsPZf^8JfOtm*vnk1ayc5Y>iDU5=L54VyDd4Ya0$zV{>D1e}I1PLbTw;D=LNEnY3$d
zTgSzSEC~Q?K1UJR=*=nOmUdAj@Wy3{k;)ArY(o{;Tz@r=!u#S<iVVVPSt+wx%(99p
zBLbmfrIX?$a;Dam+eiw<E;RbC1(x`R&M0#H#@TH_uFg~8;-`*E%7$nMmI+WIk4$7M
zTwRr~vA+8Se=35%CGiI-gEjvEqca(yqmVTfVp3?TT_;g!S|$=3HnL5&G4r+XPNC`?
z$DyD2oj!88+%mME_W6Q6Jxo#9g6}4jQ(<jeZS%JGN(l2qJW|r-@>CjV$z*}Xn2{8l
zt^WWGtWWt{a;-%v&9eGxdYB`o6~oHVB`i?`X)3one`>#~_qiiYw-^l9GW$Sfgeu99
zn%aqNQ{~ZDii?76>(C33PaJM;PWW*_;*VWSQY@pwoW@}yBU71Sni%C#ZAuHOtLh<z
zjfIVdqq)-~f}Vz<r;-|aAu_s4CzeYY6kS%!W=n>>j-a}@AlnfAGgZ}T)LE>ZNs*;)
zL^BtTe~zw?F<N<{vX}E-H;w$%H`w|PqJ3Y70hNxR`$y)L(^1m&l@x9s@d~J^<5-Co
za+;fPNCSI~?Xb0>B+UFn=~*S0C+XNTXegzHUb-%y6_BM{gbf<VV^sh=fenqB$D+tZ
z_ZjN*rH-}=JiQi{qGf1g6=mKc#1QTgN{6r{f1;9aamEm(uIs8QdWOpDGQAZaaV<M6
zj3!cVt66J-Wi7IsTWj%v%_;m=@$w9^2_o>81(ru<WK6!HOuO8Gh~NvZ-GR2k{F5{L
zMbhV49JyChbP#%nGDlREI-`u4Li$d%(txV9j*vyh=VCTWdY-kWslUMF)zW&D$SWNI
zf4W}aIuu*2fF-Ogf2AT!{7__-t(<f<eq?Iv*k+3-kjX7Wl=9U^rUY)0pa7Gk*b*B7
z=iaH!ItMnUrKs@sk2ayKnt24-Y_Ai-%)9CucG0>hTNOHQdxLYeSBbzf=Tp_CEShJU
zYMWD`PzVUK8+nUnBEsO2r%B@*xBmc5fAcP{%pj<frMhdSDI;A*NTNw=RH~7nA}^K+
z2e;wA_punfuSw((=6B2TgsG^BmqkpZ%{Wj6j=3&SbFd(MhS&@*Ch()LbIRHp{GyTx
zDQb};*PKok5=(|3bPBrzZ6eW&hrb(*Lm==vET(nMdc!@g&8wZ(380z$igBaTe<M@`
z6$x!4)&K(5w%C{PgTnRd`YfI-w>F<9N?$dks(LdymSU*(Us|ex-uKwsb8>SG)1y2=
z=^Ah1zaLwjv?_@eO*0rC6dQWry_j8v$R_tZUjejKcw^z-ZCIIg4pWt836(%<e43*2
zkXv>>qQ}%%<PUReS*}yiJsU}#f5Vy4^mbb{O+AxTS4w>sXzwC|BPPvj_Y6k+0gLpu
zZ<*y;{Jl$AnacH3Qwip1s{=D?Kzyl+5Ec7rHXxlW2KL)}H>le;6q$WfM?Fl@Eb)cA
z9pf?rQ){w|x1?`oHWoGqZA+BpUSP@SXd*GfV=GTYkt%9Bw5q5FVr{7Ie>FF?#C~I*
z)8v&pE5>6S@+_K7EiAPOCBV35MREnL&98oW#~|t~in6;f%CefsD)TxvDO4UeT?PLD
z2myd4PauJ9$2^&eXd|6jAoEQ^uoEJ{tcn0@mLL*;5ba_uv9LG<vM#HFf(m!(zYX)q
zjYRAtGf>4#@dcPOvk8iNe@iTEZZ5{#9X1&sjvWD+_1zrZR~LoV**<Sn%#*zgwDT=N
zKpN7+6SE=pDGpZTUgf)PbXAp9GAmNjnu@o5QUMam>9+xhBn{T=IkCi*RZdk?mqk}C
z7P`RxX_%JvYq0HW4S@duOb3oS6E>5i^LkvjF{r4ctEN@>sHIOPe`uWgz$)VW1qDwx
z9D|DuW>4aK^(685X`0hz8COxxtzlCXP{ld|$9DykSQ4m&>hl0wf-ei=*GG7dQPg!^
zThx>?Rb}u!qs%hIM2RCTzDaSW(VtQ&Q(<ySf(G!}Cx{gv2^@K~Ow-lHPSCV%5Ygpn
zVxSA!#Mqbi9^harf7Sgr`~#`Fm#4F8iixTzB`{4kjReR`dVpba3Y9k4?`}DB)0G+C
zYZ&R=w<~IVyqH$D5g|c5%Y70b<+;B<Be(?K=ix6*=YAo0sZ){BQ_Jx+QY6x~L~P0_
z77|`rE(>W9`wNk$V|;pNG4Ri%I&Y`UIYky(U0FpuwDnbze>ny>m?%UFtOH1dlFhn?
zC4E){0fbRz6!|87Nt#fGnB+NFghfz>Pbz0Es!IX`S)1qhEwI?$`gg4|eznRc$~ul(
z`e=mk=Kf6-f+1Z@Cf4Su<WcutYQC7yKM<e`foAz7A4T-WPnvZmXvv#FS=4DmJo$vG
zl(Ep$F=>sJf6D``MH4e97nyc90M6&MnH_FRly$@8ks#_^v7VzZ%O+}CXvl>gAr`Mn
zDx^@c1U8}x7Agh;=a=}2(N!Hsn)suh<TTZ}y%jO0$_*Vls`-joq6Jpg>;obz*tg?Q
z4}0H9WwiY%;SOuoo;&9hStNZ!RW(*!mF94oWmR-ge~9$A4I`FN{#n#O(~mm%W0_^W
z6_?jn^-DoC*<YJgm(d?n$mpj|hh0nNE6AiJtV)JEDHx~twc|fZc-Nl{>p#!xx_Y9H
zi79DlY}3Nh#y5lTjRBiY$t*V2wd?}N(HW#v{ZCO<H5{J=?qsE>lmcR;bE#u(y+=bJ
z3{PXuf7Z`DQRrOLqk4hrGTNB49+=Li4@dAaiRs{X3{`cjG-Z)Zh1*t@y}d1s4<e<h
z%w)@+Qv{0vXyuxond6AXWb!0qZ9$}0Hr2h*i;{Mz^&WMU(B+*&(w$Y)t1V@0jr=OT
zoDV4KIvI!55H{Y-EN~fI2>C^LMW`^eMp<K=e=wRRm0Wpo?h3KLKZtHM04@fV;5K_p
zUE)_xL)5wa6iHK6Q5#DnT2<tgw6jzy5(cv17O?|FdlgVImM6#N+0{Im%}OM*sAg%E
zk{E8ld_w>mQd;`LFCNEXA(zcZOH)x<SnD*D?()E3O%BCSjcP4u(2HMd>e9rWHXzr;
zf1JjXsye$r&N7O{mYMT9snnY&3Feb=a4e)SAaXVq!8|;!n<(lUe97jDN`Hz`EgZAD
zJdq2xE4_(bB#rNDV(CHP#cxsOd2{rBYb7pSQ%J8-TVD~5Msk{*MGBJ6qTh&qE7%jb
z#iObI%RO03Q|IK<RaVf`QLKYAo`vCvf2bOR5CO6D{{TJiN#u)wz4&e7S4U=9)VXg^
z<_}LxP^?@;<Y;E~x0KQ`S778Tsd2F;<l|)PzM087UUj6V>g>BW%vvQrLP8^!ErTeP
zOE4E!U4a*{KCVxPX)-#jw=jj}%j7c!E3MplWDrKLSkuS^SQ~y=(!4zC4A(R9e-0X)
z)YQ?2gW(WN_XO$Co}vJ?t?DCgb{Gx5mY!Uk(^W`i6S9RyeKQ~*%V@{|FuB`Rw;W;8
zXL3?gQ_)stl~j*SACwVppn%74G<%XiEE<<Bm_Q_IhKXPdfHN@+)&Xy^+T-cZ%JG6K
zO4pKE^IA9vZ7Uq}06+k6H4jbie{HZ9*=#(=tr1Nu>Jk=kz^;-Ab+v}}E8g4t09)oO
z1WJ&^j3lUH>J#q>byLm7_ckQol6K<^+3Pa=+Ddheks)VbqF*#ds<+awax?(M2D!f9
zJR_Pkm8$`#j!0yU^nt7@J6h}B;O&3xU=Yn3TDr88OiDb*jTLugP*|{YfA4ZO7ZyL3
zBhb&8N~>Kq%IO@6Ztg<u!P>w9VeUS7kf*Dv%QGl*I4=!UayUX&9Rb?*UG8stox!-b
zz6nnaHgvL0S5rq$k+5qz`E2FLEo~rx6bB~u76#Yb0OAIwk_K98Ny@+*PNpa|X=_Mq
z2{!iN@qTcm1{}ysPb^DLe~1IC1@$%f2KQmW`3rZ!qM@Q$WT(q&QlX=R%=05VqZY6?
z{r*^C1sJ1d&4yyM)e#>pW3kpfU*h(?#-ZGIzTse}dYWoafoUcW1ff}!6KDbuN%V`{
zbG?ASJ(=ca81=&!kg6e4$YJwD?!@d=n<=v%{D3w%<tAYzA}sZle-cbYtq+#k(c)D$
zRc)+Ag4=W3<%Fx4MOM;O%Da+OGU{YffG;NZ0D`^lFMx`o%V)27pz^1b#WB`kU;LLQ
z;`{y+ZvBt89nFyDQPTW%WHZQ=@f9nj4S>_6kO;Qtdy;p*xWv9i)No{(e9F)x)rf$m
zLZGCk`<t!TmMw1Ef85+*o@t#=T#ryBN<iTjG+2+MlY0?h0d2~Q{{SumJ1U-5%p<51
zRkb8b=#EK1^3_F!loPNQ{Dp@E;nr%fO<E}8kU*_zL1KL@K<qcYt@)nTClOT{Z1hhR
zN0`dGin_abORW5X*nIc=t}qg=8VYBH=%lJM$igYjm7XzWf7AzHP1vay9-sZN4s$ln
z7C0lGbcPaMDCPmQCG>(p+>vJIn|tqu@KjYyiwdDiSy90OfMP-vVheni<<7&LITdo6
zSww6n_=Z&x)DThjxcoq>kV{*G>iXc+^wjalkypnnJ0nLLy~dZh2U)$YEJvre&j4yH
z(B<^&Q%glJf1h4s($dT3$si0as=xvij^_8U9_J4;oc@lZDHQcINGg^@m{g^iqFXhP
z4hg$%EpcEr#APH<R8iAHtsJtz<(`}~m61ixvUpn)roe-_y|0JR<xyrVzEv)TcGEJv
zfk_%o$=bmBNd#Pb-?&G<W0_Cpd_-v#Fg|7Cj4q<Ve?c07u(>|O?d`(vom0}y6mTHb
zPfYQ&(YAx=1l-(Mk}d8{?O<#%Bbm}oN05xVk|+&0m`|xf`LeJ(Yhz+`{{SdEUjG15
zURIVV;aYlNlf@!S<^mNw&8>Yqn_FYpki0GM58PH&(@R}UkfxB)h#l_8H7RBwn_F-J
zJA-kxf5qBCRW@H3H8D)46Mw_6TPR?=+=U7af#aV|@Y<q|X(omN9Ig?SRbrI9tvg=w
z1!4fXwfG0q#qK1jDFsDE993^kOBBeOWRZ=XY*d~38<D>nM;F{XNljf@<_3rd(ya4o
zP@pLOadK~KdV#&K^bM=)a>{uWQ+ahM3%WA~e`Va8g3N9mY~1omJPu?d%c!z>ouka5
zH12{kR%R?BQX0tGELag^e#C$GSt>}Nhsvl-(LA}7i55UYNas-&jB51huq;KxmLEI<
z6cWWxRV0$eRwq`DC8j|d0HWvANDMdKt#fmG_iB9D<1|!~%&^R3lTm+J)T7q7xdet=
zfA_dQARxMzu7BwWgcCuU)e5++@buL3%%v_q=(Vi3a&|k7Hy0x2BA>*{IH7u~ev|6y
zs$xVCf7t+nMJD4>+yGf@0Jo^#)));Q3O7n(RSAk0M1L)BsNbFiw<L|W?lH{1pD7Y5
z<^n|vaP~W-i*@qY5;TjBPb6*JVxcF9f31~eEt;=~-B%aHp;|98nTDhUYSOxO3$@8>
z0CyHQu&dN~rJli-8R)uaKcJ}dl6hF=veZV=ml~s$MXX9*#9!2G4()*4so{!P)K1XH
z8X~N$WDO$_SSkVz{f&<raS2aF24L>g^$!#w?Erf!#H4c1Z~;48ZouO`$$VqUe={ek
z$hyjo4xG#&^2~9GT4)PMw~y8@r0=NgbdW=Ddt5S(qv`z0DoE(Fx_Y@(w62L*j<2x$
z)e<s-#E-%Si3az+0o;{MM$iX;N!KR0q8gzDC{k>rQ&`v%Z?|EE^4b{lYRX#5s%9}#
zNi?vbjK7v#1$jun;tFgS@NaHNf5B@iW9iC9&nB1d0uxgbLh=wbSm}@G6qC3Xz3q(l
zgU3vcKD({yw`r*Ido@foHArAt0RckH6|WHt99e-UP$aFl4VrpvgD;dxnz6K$(mZV=
z8?3t5M26?qG_l*>=e6-#k|$;*kr>kw7F2#!{{R87-0)7vgTS@1S5Fmue>stsicYr6
z{wXHb(>hGSQ&oUa(UJH`+-^>u%X~)W{yB6kH7<!=SDH%lx0f_=r8-F9t$=2+W4H#(
z{u6Ds4ISz#Drb(nF|0GbfRPDQY9W0q>GnJw@3pLPz+??3TJo#{Shb?Eg<U55DL2&N
zfeJYKag18C$GPRHnv*2ye>vpLo>lU`5;<K;E}L9OW>a&&qyu63zsa&c7rNTAI(+vi
zY?_(|l4YAQlgOSyW+EUJUPWS1C?#*H$u}3kRy;}RT;HPWX>*Ld;gxHodU<N8-Q6pT
zFPL>i+rsFe0=k%75~kX58`HgEEOEqOl1QjxRgot|^CYK`5I`D*e}&j;Qh060wjcHM
z9Z}(BWh`BNlT-D6S(up?s+p@QLPan-Y#nsGY&EG4E(!FXoiC=mLCq;5&S)w!^_~pM
z7^8J{8Wm982>@6!o$ff?iyQ{aM^B!0)pz}El2^%;=aiJoPg0ce`7vg&2a?hwfu(0(
zQL})Ip@yJV#LwcwfAqd#n6&wyQ!>=d^tG9Bu2_)7s>=GJL{=9Bu>|sM!&?=Om(OeR
ztfez2ta^N!vQrcYl`9<UFh=td-lC}6f^A|Cu>%?gXG7p4<f%su9J2KWSx&Ih#hEaw
zf;i=j%*>L0BMX<4N)2rsfY>N2aDeE~74r^|rGu+Fc;$4ke@|H=Q^MpXV;w$^RH`3d
zn9Cgivl7G});O|I{h@LSyy~|uQI{G@xY_7pnXL464IvG7Srt$@+}x34a7E3q?z8Fc
zm!-|j8QxktRqEf(ngzSE!mLq%P<CR<f17{^8}C!<K9tFFl!G_vE~FWR1tn*uUocNd
zsUS}>YbjM$e`94KoAC?S8{j>EQ<>+I(>k7+%QNhkHIGzllUA|?j@m-Y44V>MGK)Nb
zuqRg67`ImVQ`C)K@nszspJWLzYN;~nni)`2n1FJ1KGLVCGb>;5TYKD4GQPR1>Aa3w
zX)0;+7^>P)Q$WR_VRc~(B&vMJc;qV9E28Qy2)-WAf8&RX8RaESJ!N%$eFSDX-BIV1
zq%k5wfs2xK+ejLZa4%p04?}ow(Rr_eo*8FdRh@LDM4emJd1sa?Wv4}siUvYNYwFa}
zd1YLZoV+SXAcIX6G?a7Fl$La-()#3-x|U@nmFz8J>LikF`(r=p@3eP`84YyDFzYI`
zvJ`@rf39g{f<<K>mtsRJ0E$(BAT^J2acamsZOQtQvZJRu(vu{Z)yWmA+VI%|uq9`S
z9gE)0b{~WgH4+8^uY%IY)$mqi-FuuvQ$^EnEk<9JRMYv9%{iV$l6j)k3A~d;rsUj|
zxFJQg?yRAr>x#ac=y^JpmXbV<nW0}QL=pKBf4U@cwcHsQkyMoQk?XeL3`$Y-KiQk2
zs_6QYBI<1RI4h%Sp)yGl!L6DpB#t9E*4K1Lunpfy1Tew#x}OR9*E64}I%A;|sU(tP
zETUSOl=%wau~;K<A@#k>XaF_!HrNUc2SS>!5UP3;GwJorDi#`*q%{a5hIt)4*qePz
zf6AK%ExMH*h^BO}*<U-RpFZmTtfpr8Kq?@liA<FmOqygKSkwy*xF=!{d~BJo+7CX<
zrlpIex_2PU%S6<4)wLBcq><A`Ew9Yf>;z!yDqW45!uK~$yX~Kx(^F-cUQHDwvR(>`
zsUKL7v|uyS3pMvU8C5`XPLM2c8D%d`e|X21O#Ly^eIwWHOPN)N@}-Jc;En)XA`K(e
za?H(epo4IrSmNuQ_)D0_NYeFJhEYpVStOo9%{?p?D$5CF0KCa7009<X<z^s{NGv1!
zP<WjaPt+OxepxeARK)8|R}6&FqKzpMO9P`cY)QBz++O!Mnd+~!<sNI7dG+~re@7N@
z^oveK9Zbyem(s;F&lUWp09+)5iwkJnU@`iSvv-S>F}`bCk!1BB6y9PaImI{>?$$(&
zeCGv;YgvWHqitJ~Hctxv$a%j^S5x&rjBh0*m6b6^@KIHUXb`v}R+2J9-_i=BSaM5n
zGRiL$dV;g4sCt@~S_I27)vFpIe}R<K)6_dNMA}I66=avt(gyO&E}I2VFI9d%PYp8X
zxprwZQ=+vz5G<y6X=G&!=0vZkAr>yG0Mc0Lwgb_tlz3&~wRJ{c@b%Hj4J*k>R%%)W
zWKb@VhAhpzh)ZeQ$Sr*>rwp(0n;<#Ql^<GX*{(|{n$(rorkDv3>QgT2fARXP29&m~
zyBuSES>nG`X91y?spoMd&r=FXX$s9_<`usR%6DE5d}dimtLtv7q3LL|xw6QnsE(Qx
zSjo_&fv8@<G27VRYz6PDx|65hs3)q9X{hPgu%-zbO&hI_i0{I&V8ELJ$pBw;dT*(E
z!>VhbY_F$hl9Cqn)Xct2e`$Lb^?nj9dn$r=8)H-HUk17#rxB&9$)L<?V<I6wU`0fV
z!G+mS{o0Z87aF_sjSH!gm6j?fVPu9VK(nCGNgxX_`SX4VBLLSV%>Fv2O=PTwC20hb
z5g7mwMS&Mmb#Z=3#G%f5b1iqJ6m+66Zx|2>z+Y`n+O{B#{jc8@e~NldH7f)&q(WzJ
zyO5AVufK0?{$E@tQC&$Kg<4u<H$InEV!NHLq+j#j_P}Zxo?T5%B~7WIsn<HHObJmo
zqD8?_EJcUtJ@FAO1azh-A+*fwB@oJhs-0u;8=H+FT-%aGx5EDb#8#pRX{1J#=V0vK
zln8Zf6dRqv*zMdAe_?}Er1MCVMFjA$R5Y?Wx)n{dItyI>6VAf-+SmxX?6yh*&r=SQ
zF4IaKY{c7}ZU&MDn*e{IT8@eQ#HfZgg_&Vj)E&mHRwZt4U>N&a!`l<o^GNm86{H_I
zN=RmmA4>&Y^z&<wZh5`#aStUhloxg~wZ%HBsV{5#fVURhe_H+U4Ct1+hN<U<jijK2
zDi)8@gsHG=k5=Py0KXR)btOyHYMOZFr&-UIL~KIJ%dva+wY3Gl#@(=C9H|q=vN9t|
z!Z1l7k;%6qhS*gSnhG|kok4_ofnthJ{C$WZivUOmlg<H!MMXNr1toEspotvEXqdCM
zpaQc2#mCpRfAHfsjzgKkUS_9^##_vtTU3f~CzyHE*SI{|+ngjTXlKmngox2XvIb8n
zl~^r_vN0D{^;{pF$i1+JVOg2FRY_jc%Nn!!Pb})rsBdH}cVp@|7Z$&@unA^R{{Rl9
zfg`Abu3$D+Qp*u$1jgZjDs9cUxZcgYj!R7@V@M=)fAYqo7@tA26$@gkvJfs$H@U=A
z&s9+K>Q|cZ30b2pWFc*GNwD<ow>q}ICE)71wvL9Hw#^+4uM}h;lrUTBlKg>TVSXE)
zz|T8LD<Xj@9ZM={o-?E-yWN5w*0|f;o-8n=nKdq5SE6{7O&W)gB9r$(T>6E`x!Z0y
zJKqXte^OM-DX92E<TRShXvgkAHr4anQk_F!-!X(EGtgI5&6LYMB+~%N02e^$zdb_v
zED^<+1=K~D>KCyF+h8N7ju>4fp!`n3z;f1&Mw=EJ_XO?p+SbC9)ES0nL0_2C)W<2M
zNhOj=omGrv1@jW*G2X*#?sx5lBoF5q+hE~de?XCg7~7IJwylojZhH-aVwMpbub5+b
zEYSI=ZRtW>mInO&MUVct2N_h;O*F8nk|<s%%SS4UHI43|-%mFL9qfIL>~li#$s<ng
zRXq)^Ff0JuQQNZK!~XzLzAE1fSsg<P>Urmhkt`xAWsR4()NTI&skpU?A$T~LIrC3X
ze@|9LL@hnxmu)eIDlS1i><#b7^a+=f;pwBKjzx)N^4zqLu~uI(ZS^AS^R>r&7`dwG
zvh}Wx15Gt$Ou`{hr~r+HnMf_^0F7qC-#g)a&oEi$hG>EXU2sL^$f~It`ig<-H!2Tq
zNjuvW+Qyhj=|L=>R7^ERF8aY2^)VI}e*<rRcEB--Wu=X(Bp$T7NYg6F07yW$zja-e
zblegz^BQhP%MT3|U2@Y?%E|~pu%yT?`csm{d$+2`VRLKW2|V-E#Z^^38qR9a%F1PC
z)QZ;yfYWnfe^(oG#uccJS{#}gC#j{of+So0<YvWIGp5%+g|D@Sn{B{1&NBR(vm}Qt
z&wny#=9OGRWT!2zHT4Y*ppQ{0{dI9-jc11SIjjRX>CDzlmobKny&XiAa!)JTLR|zz
zkS<w$FTTd#B~_XUMDxOBrl>2bQVW<Y>U_iscO)nWV{LkQwWQOh>mMkp&#2vAjybRA
zYq4aJt#@8^17-uUvA|Phpz#WtoYhoj`F}QG!bXLjSSO};kfFO0jLCmdJPs@mu_fvZ
z+L*0&ZPAUCQOFgHwDk*Ah-^iqU>iYZ2h>4L)Le0k3(B&J&xQ{7A{Zl{Z@gBp9%$OY
zn*p#p9xMsPw<_qIyELALf>_d6CW=^Oh)k;@wx(MbP!A^H_Z(ZC1y-9q%X7Iz^nbaP
z9Ys{~nu4^_2}2zroJOGfc+@}u4efg<#A@bLGff>DW-$+isA#I1MGDI$@29ap$~{8Y
z+*omG>aPNN2c;vUr-!B}o<A-@0<*|;3xZ0u!8T^qYu@~uoK~}M2s$b%SSzdgb^sYX
z$)<-;FkV76vzGM}bIq;|g~-4!oPXurAz3%URwWHvC^6GXQ51R@5IK1-=>Q!dTXXqi
zEbGb-2BD&os;TK|>++b&no0_%hFL&hOi{AxMJ=ol+8J&}t6Y(3&b%zh>MNuE7t=XX
zJd|@uL0wSwI$m^;Bqd{66}?Ot5_k$ti8jPVH`%`;>P+I2kE*i_k1MLKtADjmz|z!H
zqyzAPB$8%~0!`cQ^<94#B!+uap7h-<3{do!Tjr}1$c<N%W>t3+LcxfST7W$yt$+X>
zR=Fa}HPbyOO*1ZSoOpwfY0RnK85yb}ouOGhNRO<dp#^mJ^;mO@u5;iYK^lroi!SL(
zNS>Cp4P!++x0@VyE{QU=*ncp#{e}0w_;s6j4dH~DRYf&ESyx3Yg(O*CrObs_VH>H^
ziU?oD>`!t|0;|&>XWoOUVGd!{Q|2}CO!7@nO<5dJ#)%%O4V9VIRZgPC!1Rvh!o`;K
zPl5dr5UrTe3d#fKK-G24B9UP0AVzY!t^wE+usmA%+&&f4%hXAh_<wy{RZ@&J$(pNq
zVv<q>Xq=HM+_a1U)#(=b-8vuay^zC3gRDC8x~5pCYLus?jx~+hmrlY!%zZ1(fb@ZY
zn$P@V@ZX~1Y_^^WWv*!g&e?Pe5F*3U&tR@d_S6mTNjJqVx9v048N3v|G0|NsPg1f}
zDbiWwc48H8bt*#Nf`3lmt~L|k&qL%?KMhHe(`Pg04vid@&MK-GPRgKxV08f6LY?%3
zhY;k_^y7aErOK)3RL0XtwDGE#(Dyn;i6coq#9M3b7>;Q{<A;dec5>N8Z3(3KVG_Em
zG|}l>O5b_wm5ZY305GtyYbhq5Cr#&znoE_?<X;1uDoG52rhhgt25VnIU^F);Oa2?(
zY%%Iu+~U74>AI@=HWFq`h8j9^s?5v?Qrm(M5(_rg1LtE%r|^cqHjb}1>faF;FwABs
z{#_j!!whPRC8^jD36Fj*tSks1oCloq)1G89wp*CRR}N)H%Jp$bEp|3=#Dlp8L1VBt
z+qNbo%-=bpSOF`yZ2<zW1Ggas0w6X806jp$zqj$g2R%Rq00}_$zqc*{2=WiNe~SVa
zFn^4_b8sb5*Z+HB+fF8$*tVTaY}>Y-6HSbXlZkEHwr$MB_RT!+b8pqHdw*4L|95(Q
z_op}3+G}-Jb)QmHCLe=rg;($QyXxuekIIu-P1{tV(V}orDX9r@n{y>_bdsH8pI{Og
z8OscL8MCh8rX1s2<Yy~Od24a^lqb&eaDVJcc}#eYqs~vePF>6lC{k~qs4mB>!>g$B
z_!`+P0W-svk+*2o{5zBAxa`CrZxnq%7q&?tA}AtLIO|@xLI8oy#W-fF@9u^Q!AE+$
z4@COlk`lilP^ZIeR_Pb!VieZWo*9bwgJBQeP+d0r;yN#iF>X0%JgOKXgs**+HGi$>
z`w?!eQQE%teOH&?M9<;(v+Y}pi_vJVooSf4i+G)>oK9rAP)8h@F|n#V;mX%5a@KZt
zJFMh1mwsX{FmcH@Mm!22`vjR>1$7r@V~NFpKrJ}W@1j#tXq)=lm!HxHY59SRUL65)
zAk&NLFb0?!ybLm#zrBJIozM9T4}UJtkfhgtVhf@bH$P}^Y+~n;Ezq?Uz~nIa+;fO?
zD<rb<${3FZVtJ)jRymcg<-OK5TA!EYQFK=?Tj%cO+x1L#UV;6xQBjmPx?9O{+4_y=
zej86>{!{O{Il8<LH2DcrlnPGTXT2BJ9h8R={0|QXyCN9+5w2MX-s<Rc(|=!3ig%a4
zwxfNNI(`B-33-<p40i4J5i6n9h_a;*Eh^_z_lr%9x~g*e0vC7<;N}|KC7-G+_7=WQ
zx_>yJet^}LLMmT=f6l!@mb2THdF{B2Xq{%cIvovNH2|kD8aNX<h-TsErl`1e)Yh8d
zT`0F{)>-tJZ3!-8J+rkwyMM{A8+hAnV=8Zh?UbrOl+;6_vsNVqbel$SU}rTaxYkdP
zrBys=DDY#^kh!JAAp1pdrABd=ZcbU5Xed>u#v3ieQV;Z;O4RIrC+fit6R?(NJz9l}
zM6p^9#-}a6y|-y1F9O3wDXDB7jzK6xtxnx%JWoHuG|cQVi`jUXM1RTSt|W-K>ONoK
zrJdu!%^1*8d!~h*C(g?31%!)vn2LcR&_iscv^XdKEoq%UR&>kONkc3D2G#!z7Gn!~
z`hZIivp`^`r<<9hl**8BH?g0G?(>aIyQ1%OwETEFTc;RWIDJ=vKyiwQ;wp2XmtOf3
z*iTNp64#FD(&ft7cz;a0zmrs(b7PX0th4mj;?7NX?;IHB3d07}gMJG;Nu20GWP>CR
z&%VZ~mlnn7uA$N18On*gjBvm7Lx<2If8eL_uQDJv*rC2S@4m(TG&m_Pui8y-+RM2o
zKWND%Wr357T&YPeuq%;8Cvm!!(ZR@cux-JSY_2@yK7Jj*%YUS9q&fYhIAK$*?kry@
zWj}rN06V+)$1GwamflQDWfG@zeJ`nV^4E%#aj`7|b1uP`QPV)h2ugvoUr0A{R{)hu
zbX)k;3a(XBWTG*a-cJ0EJc#9H$Z19tX6$CPV*#?W%hIJyE3V&7Lp7{U?LC{CQnLg^
z>XH`ae>#c#JAX-^%O^MeEkl-m)QYEa`NoOGs!H5ER$tTsgo_Mz6Z**7u{Q1SGke8i
zwh;O@cUJE*9W*nGF4&!LfKbnSFGtlNNlkp=fgW78=+X6CetNM^C?9-Bw3n0exvm*w
z&ERcRCE9Zt68#Q6)HXQ~sEj78AFl}p8joEsOmMEUh=2d1RKJgE68igEeavM|?k;po
zNC~l~Xv9FXb7CW|7(`lRJ=L{nLcSe}R5P+GkZd?+Emz&=@Vs71yX6|PtpIeMO6cs3
zal67If4K%%q2%C)v{edgW9Clo)N+mz(!PbINPq|g>lEi+5fuys@2o0zrv0$;xC0mt
zY=+_~n18WooKbhVRU-R+@23V-;;nYrXM?}7`{-LyCm-xmGyF$$K{+gWugM37vxD(F
zpUB;hVSzfEZx|Q(EHCpp;<^IOhukV=+|rYJvaH7Slu)9g_=p{2Zbor-;IXuMB43)P
z!nTVh@-xwP3Y2=DgBc7ug#~M4juFBfu_!9KtA8pK*&js-$s%LwP$YK#HkAGP6NJl~
zG=fo3t;fgSue9)4m0^d2nyacQX6&Kb4sJt=Myi%ngMY0``U;SPqfE=TnW}6Ixa=Z0
zyj$7sXHh6kl8HH0sm7g!Q`42WGnV4d5V^R!R0U((dO*2}6J2my1Z9I59=L^SDypHz
zcz=5*F{;i{)=D9?;W`_!fo-lP+)fwPbwqu@01MuH#amwzVlSK~W$(}+5PZNPqHLzl
z!;?5%wO_6omkD9jS&^6a4mXutK4nFEmO)<bB-wilbRxB;XEOjdC*iQ>Kd(qM<iLbO
z!|XZBEpVx!!CUnO*Gx{#CTM^sOYbAAD}Tk!R`ja2=*$`hUKv~IQ}mvQs*5u^f=$>Q
zo+~nEG-!aPw)u@{jp`Tc*)7o8+XC2FMV;#9+w|%K<5nx?gjs%nj3kbS7)}{}*d&Mf
z&{1;>S)<M{>-l+lFRP{5Q%_U`W{1@4?xNf8trcEnFpm2I)O`mw9=CsURB-8^tABjk
z`rR3B&`}ZFYsT9RZQM0mU&f{W%k>Q+DYa$2Dp)HDPViz210sV!iNscIsLiwigT*AB
zwd2*s>JMYiMnwz(bYr(dM>*ROgkKi2BiYnU{S^jPC=3iUL#rgH)hni`1GeLfxs{jE
z`R*X$cDtuSzetigf2%PI&W)VoZhyC&TrMf*mIIO%pKacAZ}J>>p(&iXH6nH1<g+eY
zI=1(-VT<FuUvdR&uL`ai8u#64u5p*P%=G{#FR|VOn^JkJ>vZ*+hws4k=+46x?x`*7
z;O$6G7-U9}ZbW_a!y5|r6j<Qj3r4Lp;I_<kgKI81mcMB9forJfUg~g64u2uLSg;t{
z)<V)FCW4NwBgFpT-H<Ts6fdI>%&sDRy2+cx^#_@E*Q`mYq4$kG5+t%m_*lZl;8edj
zqsJ+8$SZ|B+3N^IJG`a2NJ!j3qcEfc&AvopiRz_lKr2JguZ`9y)w?O_!;g+bWPQu~
zzq=#kjBVtOc3EMhc!8hk1Aipf$ki%&hlhu|lEIA>UPDn#6FLTno%Zq03tDr*_?S@?
zLMk4X#eC)wCBBX<OXO6E4PKnFI_LEUWvl9~AO&HrjQp_X^*`nypV;|+xn<*y=0o5a
z5OYIZXa$=&8x{>*P)t)I)P=<wfTCYS#kL|`#N4ZZ*X`ief|1?ykbhtAW-lb~3nG}<
zVX_zw2RtIsG?BxDX>WUDVGFNy)cxX_CVpWUUMpEzA>T_5uPxYDYF>8J3S5OVVm)5^
zmV><{##+zRFNo*)z|ILcNuB&~3dnHwBI|JR`Uy6_yRa?8y-}b+i%+O0OL}|EszvWk
z8FkmIGoTInj7?wgtbfVxQj!lfQm?Mid26g+r0sW}ySBl(vG}RV$4Z_SkwB7TCen_I
z$--H4Vt-!RYI$qNvrs|sf?&lS_!~b-2uT+&v~)+uuA}YskcBc`cF7VUBUC&!AYX0k
zsRRXk1(g=teAH;tt<vDJz>B|UNaOoj0>X}>k`CuEBYGfY)_?hhZjs0a;NnhAw1?Bd
zG!Kc$rZ^!q$vwKjGp!@crod9n6GDQKpiW*%g0_VuYUTMrt+~%L$;+I-R<z3#d%-so
zpGd&>`RsI@g9A@I9lKTNnMTbH<1I^0NEg>>#pJKn4c`sqYi+Bv4{y|DPm<fJe~9G`
zSx`vk0KXN`?tc%HHvBKk0W0l%oo4%U*+TC7{5qX&aTJ^aY_)>9=;1R_z;I!7#fA2P
z%r4@bfvUr=S~;1ahvU7Tb!9sCwfQH@0%sMzH!l+yemg>bqsujZcnJrg>Wy52FefNR
zbMhBM?z$9l9#_IFbwxd40v<V{U*7mS^uHpN+5TkN{C}WcC~gfD<y&zUxlPCYxx-$m
zA4)j;j5(;xgZrEav#qvPm;dUiC8>!sUQ|W$=X!XSsy=$<(ogm1Ib!z4bg+rUo-~gf
zSL+)G(PDeD$PzB7%x_vFzK%&x@vGs^wKCCP)0~!I<uLRGvYYVi(B$875QmydJe2*w
z&uTrx&wrH9N|dvSQt9N*z0wx`==|i4F@7oG=XWE-D7)s?3Bjj*mv&}V^2UD*{Qizz
z6w?=``D&V5Jh0sa-z>>+ubP(bjoOB5n8lMR6Nl#KikCZcsVzM>5I3i8Y9ahH>$>G#
z)4xvA0uk5LTn#b!P_D`BS(S_Wid=bTy=MCx>whYb0HPk6o?58ol3IOx%)QtoFVn`5
zhnjn@$C#qn;R~W1r_62aR8N$|s&FK;fK_Gt`BhbNefF77{iZB4LV#MRt`!OOEf3)s
z-z|ZT_H|WcwbJojrT&wyvCEzUgT#_)A+K?a?K$m<hZXx2(Oz1er!Q0<3@2-tmO0Rr
ze1C$dk_XXllr_@_g+ub5B@T%&G9Ql;e?u(EUrM^GamLAE%hF>gYWkfmPi%|V=k_k7
zo>P9_Acb`7n***t&SjAp?C`g}h|0TD({O7k7iTK{D7A^1C_kDz)+J;Kf~0`5rC}wT
z?r-dhDcJoI@704J#OiDUc65NoC4ZcSSbxtCK9d$$U>y-0yjvh`8>C6!X-_r%l}>{J
zBjEF@a1x`sYX6EE%IznujyHu0?cPZ_yasDPTl(#gM30qf7_fQ;F{N{6iUFtvA9U){
zQ!3OTrx-1*_l;Gz>-FVN|FHn<?+PFDPlC-Z--Yt74(HdY^R;n?6a7O|x=M>#q<>}_
z1SvUX{h0H@?WNQp1`=8}(vzIZ<GgeznpQazR;vI#VBHMT-Q7}zx`iyScl^`lw{|n5
z#OOuQ0~Y`9HgTmm$jP6{8(<-TXuQZ~c@Qi~&beeeL3-MgpuHvNr@JOdNj^SaFP@t{
z%uvr&Z=KmnLOcw;6Knf*$;}iKLVv~O<sKws!OBYGF7^G0tteLwGPvw2u3lxkMDNDv
zwB}6PA<F}#kn{Lk*8!<xuD6>!4ZOT?O-4my9X#3)cS*cfPiEkGx86OUSX~^pe^-h)
zRKgLio>Dti5FZaGy)B*%Y@TWcG@QV8FN;L!U^N4-&$g~_nO}69%02dd%YPcvvgLp*
z-yg$^h;dRp(|q2mzTCN4sSmBuX@gP+tp(G_JU|-`z`&vCHheK%UjKakgRLIR(gR1A
zO?Ffs0(+1l;+`AHIaQT37$sHUx$7_8#<NZjA8o4Vy2-ZeJZ%_qaz>{1>ygr*xpR5w
zP8N{qN<Pd?ivTbohME@Tq<<ST`q<coYkq_}d?a<f`P{G@;N`+Ifqh-R!+28EAM@CF
zfP^?YXK2d-{I4OnmcyZ5UdgFo*DWN9_Z;=$QC@C*iEH8lgmr~sFj2n4Pl9igZ#zi(
zZ`a#h1Eb>Qx8FwmF;rW*pG0JmTI0^+k%RjpZ?t6ptb4k}_IkUI6MvbA6CfBYh1k3g
zwH<oJ803U<^_ec58q^w7U*uRlZZeVwIratSZ_REvB_ziu(1o6EZU@Ol<6$-ZN<jLF
z1x@n6H{ps-t}`%#lL=Ub<qs%e*pAvkU<Cwqm50b-YEbrYpy+)@ECQ#tBLgS1If6Qs
zn0T<YQP+f!npjaGV}Cc21FZas>_kTjYE|(!E8Uxxg>tOjBkiPVt|rpS&j|APxQSQM
zYmT4pKZGGhW`R6wr31`H_K=Q2hwGr4nD(mR*d5<%xJu}xk`Vf%EF5V)w_K1?Q-es6
zg6O;b^-YeA$b_<D%ybHZQjHSO%AB7Ur86<e`9zq~=8ezq=zp{hR2(}1^u;=+AUn!7
z^Q$Wn5_;2`vvr}ZO$?A@x|W-VwKVlN$&5+~?gOME*^_bdNzm2ZWJEC6;?V0!1%<I*
zPpFW@3?zCr_ZIZY<jpABp>zYLYoo6tokP?zncAk#OWe6?qo&Y-NlP~0^}}vp1<N~4
zj>wNf$-lXy)PMQabQpQLgG;wS`)#(wD|qAg2ZUm*H91qn#SicT(l)nXea{W%4dT^`
z`0J1=Db6UKm_)yY*NF}laj_zU=Z)r;?sYJCfC%;UkK$%x2k|kzQXRNpp&+HU@U{;3
z7AUO0mD6=N$x2jc>5fah<_jSodS%9t-Iv{MvU0C66Mqy}C8KJ3#xPlvA!`tx)=txU
z0IO#h@yD7UHXED6l4%`V5?sg>z?upSz|a(PB_ZMUJ@Z>J$O0wnzVFeVLP^2K%cR4)
zk7)`)tCAUwUpLOATus4J22@N>=VUast``ZsNiT(NZB|p7l{D8ie@?O8^=>Tc>s<7|
zcjB?`B!B3c99<NzzPnXpYfR()u$;{FeU?U;zzZC?CL*&UxuZdi3UMwaZbOac6PbdZ
zs72#6QHu8zV>K-uXetv<KB6V1c?09Wi}pSB#PC_Um3VjfRdJ)03kD?mCi}$jhb;Bf
z1`}Th-2Fz7+bFD*JYzLYfX6K*9(@cnp&z+Lp?}VBdcnP5!k|&sFtERh_JnDIw;ylb
zliaI2Tz9C?vX1bQ_Hm@_0CA#MNbxyW(Ba}VPCE^=j6p^R);4zRF~%CkTW~R&OCCQg
z4|?_iO+xvNYUp#Fw46fl$5Nl(q<+`UkU>yEi;eQ^MPo-$cB#IImQHs#H$I`K1F|F`
zHGdhLoP|~|sI4O3rKO)YZBF=rNwu&(!$~noE!Gqf&!H9Q&bjhKfN}=2(KjGl?hZwc
z*QpOd^s2xdIjm!8E8K+bs1AlV`H%??VdTlefsdTVhR6UK@)n_kwhK*^1tMS9xaD^*
zlvS~N+C#Cpy2vXYF9$K^8JP7f?zw79?SCO5OgUEPq#@XR-4NVgSS=#}Hq#0l{l<9l
zN%^9Y!h+e7P2#5)pJqM0P)is>*)u;xb1Tv9I6hC3p_xA$(ObTPB?~5#ia!>-2->~7
zB!>z&3T<d1vZB^~CtMuV#w+ixK54P~Qkn@YV7Quq7L4!cKV!Bk{ElD6B)sJ+)qiP2
z{B!ha1$^?$O%s24oCZDdw>))K|Gd9#R=dgKPyI>FFmp<yMtL4(?~Izt)X&Z6tR*z1
z>@~GZL`MIkhJqyOgyS+}Gah6?I&magyzVfo?Nc5+kKgNardr|nmsiQ&WSL24e(MOo
z{f_hIB-*ZmbY*bC@2sZS>D7{#^nV&dzQCVN&X4r*vBUg4L$IwnlN+0PMHvi{NL;bb
zGMPw%41=ZkFH%@d3o)}^MwnG4`>woPi39YhP50*)2h=t-QnYrHvr*7n&!`qNc39mP
zTWsg;<`R$o0E64_`XTO6cgjf68`yC(4Rtuh$TMr5^?{YPJl^i=kB{4}(|<%4h-#0I
z_)6Y1`-2&f(=2innrz4Arub7RATbt4%t|uyjOTRw5F`e!GG2<JMPw@|Gqgw=1N4n~
z-2-G~I#rmX;=yDyZO+&tK(qXrV;B9|IZb#|+F8#^jyUnF<VR@Ca0Pw-)9L#A7M7=B
zh1Fw$PNF!`xh9u$zQl-`!GGUg_P9ZzGvkn#L64Y83xO4EDx6K7IeSVnPp8e-%Ce3A
zk)682%wvBXri$<8zrk?sNoG4_Uoyr|^qb@Vp`$QM4~DhU(8r2I7B@LovAj&@GB3_G
zuW{%JlWKHj81(R;?m&<CDX2n7l|tE=gcEsAHNXiy5(tu6lQ>pP|9`>OV69A4#~#Mf
zELtpzthA@T^8z7@I>AkW9er@!5QBQ&44wH<Daxo#*B3<Vu)Z`b`&wida}cW%tbZWy
z+U|$TP8G(bB5-3aT@7YwcD3~MsZrXqK@1Lg^1SL{TbAV{;*{b*MpMlto6n-4t}EPF
z(0m^jks})2SQ^r^7k~CvHdHQyE(|u^teKkMI(LZ|mV_X!FVho_XMY-cZZj(va|`vh
zA<0T-$$VRH$RTJ)Y}4m$$lV@1XUKfQ%UR{~5F5P)Lz(KG@EUj9#?a@<xcVfYW=QLV
zqhEh&1lM57<WT1>o`dh$_sBy!1wE^pnzXE=BPvGki`%`4;eTo{6OU-I@c2<I{7aw*
zr`+ozN)h<{JG*hiugu;5K-WPHQ4C0E%=C?>&Q2(5lH@QQ95T^aON`+N4je-&&!}AZ
zcZwB0^4Tw9xMhX6H7eC(1xW{Y9LNY=7Hx(dKCjQP$yDp)1;kkM=BeN2dv+sAQ}9rZ
zn_ggPRq7y#J%5lnW=6M2x#J|`RZm3u>E^%-*yIOs-<0<uK<kzP$vFrd+fQM;xS`T6
zyw4fD90l=TZDs}NVIHxpL>321_(ev=#qAox5YXl<G%(3Z8DLpC<Z#0SGm(VzX+&1;
z0`JmJh!RY41D7Dfpw(^ge~Le`(-Q=!Wa)u_)Tmi;-+#~ro=%s0OQgW-LWcfc*V1|-
zN4$NYYlm*@0qx%VyFNlnPr8}~qJ#zb8VXCaXT(6buqu^mP)6Om4C}|as76$7&2>^(
zjBpz@XU_vX_IcykyrVI$e^2T=h{i-=h;8w(!3J3)4`17@@k>Wvq$|w3%lm`UsX6*M
zgnOc>-hUKV_;fIS<~w!<HfvS%<F)>Y1j_2-W2o%)I@XN*+?xNqaIuxR1FN}@R;Ba5
z*&TzrXeQUVxz2+TjgY`a5I0B`C&%quAxlN@+9jn{&0j^^nR8afYAJkem?v;$_2oLI
z3rx;3v3R=kk8VpCmQ7M7M^uhe-PexbuKXRXlz+`frZo!UBoj#hv#E8gokb9KXy5#6
zeMYbBdpB8T$5mvG+l71q9EsrLLrsPEqrzX<ZGxHR*QMTymH-=l?c&X{=v-E+^~U7d
zhwb;eqqJTk(NU#KM|<trP9=#$#^n$A%TM%e^33S-lHpmOpC&L21*lSJ-vffQiwHCf
zQhzh|WhGMCX@XmaNh(eI_{xiVTQ+KL?mjkstRw~&!ovoHUpIyLJGTu6*k0~Z<BR_&
zdlTtXkwLRt(XefT*#jVCx7?#WREZ!Heo(Ubp^KXbn!X|$s13;KJ2C^e#$Hg-Iu2&i
z#{g9^!zI)y3jU`asji@)I$5K|)v}<ZeSf2^J`S3lPy1WjoOBvwX1<y=o;B*-<?fNv
zm*O**CeyC9>gV~JGeUe)cg3aL@WF(vQYiEVOT9{URN&Vj;cym6YY@Edl!Nj-WOuEj
zz|zdeYocKJ&)iojf>0@qB!ysHgBIM`{^y;k^8(WDKwC`V0)=WHlaSzB-d*cwQGZts
zAD&trNcM1?H$GLvNvpAMj3Gtpc}w}X4z;V5-J*XDXrEP~6c9Ay4_~4ECGv^VGRc*K
za}LH`>&L}?h7&_;oEu&Ogj*+Gy`|yTk}h$@hx=4UF4pigAAYa!oJExzMu2vD)|Bb+
z<>1i!ZqH#z4Yd}FHAH(9JAUA4U4PFOPf4Ol!4E!yL6>NmTVYfXN^lDY1sOELYU)lQ
z*2bXPRcw8_fG2b$nEaI-r>Qti@v-G#Uhc#z)r3ZqgwYfWMLiMn$fQ$-?}UoNk0CE~
z|J~=<z(OOXmk9sSkTjwz`cuK*vpF09zW`k!Y<GS?p`XBlIuLBJR-GZbSbx`RO`3Hc
z<B=we-xtM*)M26w@vvi{3q7q*otFxkpzw3FcrmNLqz|qaUUBKD7uoyeyNP_8u{Gtj
zXA75(ro0zWCp$q@SOW(!73Od9caD}fB1_z|^n;i>wfUgpaPs%^mNgt_4ar5d-<xV3
z=4>&zT~4g{ey@L@=XZY~<$vy8J=do`tvYkrJtd5_`mK-TK%Z#qa1|Gh1G2*{oj<y0
z@tyj@Prl)!(h$&xC5+$pa@bhkAYc!;O|Ka4NQwWbAfez)Fu)SQGx-hqRo7o?98P%#
zTb=&(;dIx*oI>e7IdgUFM=()HfM}hdsq+#oVL0>>n4hFx+tk#Q+kb+A)h!t7KVmy>
z3G=I22MqeM6{G=dG^mX9p+XG7JXsp517bz?_8^&$Kt@x_UHL^X`hAQG#6F*H4tARr
z!YS?bND)q>itW3+?+<?e*sd~0SMSlnpqI;|J3DsEDVc<g=f61+DpqM}Nx4=dXq!^E
z|Ao2WQ{}mLNVH_}sefDAi<A{i3{n4v)88D&&)MH71ENB8*l78bPp<H~%-*4&)G5^;
zmu>%G-;GaI=$YO8yV$PL$}_u3#WKkMm$+rD?4%83lQn*FU7^~_(1m#YY+W}P<^xuq
zm>7kP%k@G`nWHJIJ@EHL<&34Vag)aYP_Gs;XXHT~Uf#Ba6@QP{0>u=!$ho09`x!R?
z*0-{65L6L?oIj++c`WdBkE^F`>WDaYne6!y3#7_8EOVQ1Ie{@V@s(JhiDI_Pp=u$-
z%u#G#W{%rXSvP%gh_;f>5aZjhFg8&rJ>4*s{)L4~u41jNZ88+;bS_8wW2z80*Kc6)
zS0bs2Z-yF>l7HsVI*hZ>0KdHEr+01J*r$nwn3nhz<5BqXDFeEdcPC(agRgrJD}o?D
z$GHWMB>v;bTPS1oDNI+mKk58g1|)U%(AX>o8$}{nh1W`0C{ABGx>e*>6fnu?81tY(
z&<TIk^0NWVo~+X^H;Q?J$f55b+mZ}Tlo{%?#`!OAOMg_Z-6N33Je2gTG14}$ew5qG
zLXB3cz4-o|pXx1teEq7e2m6z}e@U1iX=QF@7~Z3T>eRxrR?*RTPC0L3uxU;gBxL6n
zRs<QiJh&5+R!TX<Nh^`yg<_icO^UF(nyUj}s>?KEQs#Rs+huTJ4J0WYFDeR$g1w_H
zfO{`0A%BE)o{y2BmH1H>d#>VFikc6@;%9~SowfVVoZE|&CeM$v+Ed97da<AMiosN`
z9*+@)6HfNqKmo2l%?&U?eW?}&_StpKc@4i|7CEH@08<p>jW|<31`#GFb4E=)BlboD
zXTbilX$wj_r3PA*ig4?<otH8N5j4~&RnQia_<s-;Ikx=B!kq{I$Y(mO*pD1qv4`$F
z!4Xrd)O3ra$eBnm;WYb$<L#{!bD}?4tr(N129)F7B1A~kI$Q6uI^K*xyDlRLdUFT|
zslH^TQZi=}e-C&ADa8klkP{c>=F2wT@X!#kjnVN4og$9>{uQ&op|ar|jr;FoA_DUC
zm4EGNZ{pBKgjK)5wb%XRxWy8jASJ{~EXZ2-7tGQfafXR?(I^0*C6hy_M%?2UkwR3C
zlG};HD`kCATZOO)uE>}udO>xX#x5Cxo3=dVLyEFX8W}2DEnZ|(hEzZ3H8Wc&Bz*I=
zpc@+iSTzB=ar%Afxr)a&l||Xfbzao*e}AF{3%1B$(3iGDscx(ziU_MY;}u#ouJx$X
zXX8o;=a3gq(2kGr)q*_BYvTUMScy_6@XD)IE|v0+uI9|{3KT0=4g2Cw8}K6D(6$CF
z%QH)<>0UR{{4y`Jjp7k=0JTOYu@qfhQg~_DY|ULj+ksm^$rLgYFQB{<G;_4P5P!Kf
zXF5?fq(D<WiCUgsxkA(c9`n4f)7?{Hy`RsUGNNpr>ukJ7!r1E(Z=+AA5fQXIdwQ;Z
za!p7HQyN?ktg4r;PqDI@MVIqz(ps)?AAX-loxpH*L)dk)t`%>nmG++hu1yy-xpV*X
z=g0f#d^h4>WPX1&)=gXTE-KO?1b;$BXl82)J?RL-)ciV=mk~>ZejOXkI3bk^%@n+;
zc@kP;kNK2wkI9n}6g{qXK9ji_=5Du`tX@ZwD}7M)czl$z=R&eU7cE9SL7}KK5LU!}
z<n&hy<CE6>)AIB?OvvoYrDnqx;#2m6-8wi{oim4a;A>i(F}2eV_N4=A>wnT`_xq;5
z2<My5xq>uldRLygyNAkKY|S3_B%TcSJ3Dy%7LC|?hJew47-6zyY^2A9tZ2a*qg%aZ
zYxZ#X4v{L5CNEx%I)vq&>+)1GM}e1X1gXvU!-nzkcivwHugzM`@<jC#Opfh8)9Fnf
zGWr`OgOZg|^6+uOQT5@W9e=w|wh4Xi7Z<T4-ZADnBxmqA-yLGr?I7FyV|(q+`1mHG
zN|lt4hOzPD%2dK<I000NtB$g#ahKuOSHoyzrfifn$bhiZ?Sv%6p72oE2M6h4dF<CD
zFw!IIQ(pZNZH0U<%-J>2Nv&U*zhZ!_{wyx^nM|o11)k10b4xP|Q-9`pY?vtlQ}I8J
zFbhWDsq~6)QK9+qdVOM|+KcgJrVoRrmTp`1$OYvZzCj`lE3X9M$f5M8QS`2$uV_Yd
zbr7<aORn>nN-%UPr~XN)uNmB^j(RF9NX|Hs$vxYzNA?ppdUvsksfM$eU?9Ed5gE4o
z^9e>$g^XbatWo#K%zqfw;8m=|20(3tayHR;Wcn(O%l9)PcO27*w>IQ{7v2l|j#L^%
zDhcgaA=_z7hssm2>>F`&xnc=gDu}<g!m_`)$~5oID2q)eD<&j+D1q~sW<8Sqv#!F8
zTP4JsG}&u1;NmUq$phSi#)OSmHL3!Fm2QU2i?`Y}P<+(NNPo_CE<vyc!}I4z^3e3c
z{ZoqOxfiB^DKCltnr9xav-<Lw)lY7ZUK&HoC6hc}6`gsx7Ko0TTA3VovHYBh^`eWF
zOUg7{`iz*1okpXdF#K@>U9!%yRfv7lv2HQg=17sx(s*4b&-)}7Mh>RGE$L(3m-oqD
z(#dM4sg~;eqJQ!=od=L)w+0x^=q1s}nkH^Z{)$OS%1?3Mk?_f`Li+H_JEL>j24BeL
z2=e^xv@njsj*fL*7hu9#`A*8BdHu0|Q1f_b>Yit!^g<+6aUXuLZna&gQ%?OuN}Ckw
z;_*Zlbp=ryz28WaL7?LF6%;tQ0C#{CXzG!<KE$2s8h;dPA?PP$MVJ~SY$YaTr57pe
zXeEz}wLI|S8EER%Ck9L(EQJ*+^dwJjEv*t0dlutHGheR(T?9$yMMeO%QvUH>hL&2b
zMzsXe867<dv`;q)-R2w}7FCA1$EZ=qfVw-Ej70!R)6z*?+AJO;vTyV#`YaI*+w`v2
zEdZP#!GBqK?^Z?zR$3-ND>(@{XpkI8UPyckAMnD?1SxN3sswJ7H>xK4me`r@LVS_A
z(erx<#3+yvFxV`@cm^II_T@X{FTtk5;{#H8n6QFBQZ!U549b60=At8C<o`CXF8&dx
zk&yRe93o+Y;rr;sbnpnJ`EL_U%|KJi4CWMhy?@Ck7=TR#HcpGYM`p5$+|LX~<jz}~
zj>xhw7u&@1rc?3clWeccyzn6#6N4v<rv%5b$inyY#yoK749P|lUgPdZ?4Nw{&Yx+3
z7u@lXc1U5%WB*{}?Bpn9UYQXe6=78D^AIEu?i6C?>^{9rhjUN?3Mba$2(V)0XN9W+
zzki{EzmCI>?-BkQPrW!3eEreH=7Uz;>g`od$yyFL?+QCQ9*S>p4?^@x^y4tDvkQ*1
zlJvOerSTz+O}Gmje8)@fl-;NMx(~98>G^C*u6!2iOt_~@ZPxo$iUrSOdVLz<1OWjP
z8Arx}UAJTpY@k%&R?BvaTzwBA5x;52=zj`TRpW!X6(uF<g|+Z?wT=QSS^!#JRI0cO
z)169qD^Wc>2IRo@kT}UA;I+bSj0e+87X5if;$|D%!SN3>dvYg|=z3Ju+UN<Uuvy5?
zIabvZ>((a=y@b0P*&EJ`ip0gaEnMYFf`S1<lbakNf8W<FUQ$NxpJz&fWxj?C^?!Wu
zR|+hyZno`6Xiq!vOBX=Rl+bin``UZ6+ha@X>Xj)^4z<={B3WbFtz%iNhdcO`hhseP
zgbkiFtY9sLU0dPl*?st3IfW-*HQh6*xHGAM+0g7|BlWcbx4+}nA@ygV>W}mI%RY~b
z?+hd+JU`Dtg1)@oySFvaIhc?1XMdNts+o|Sl#ybL=i&jykZ}fI`Vk8VAXgE#qy6i~
z#aDUMuJ6kozBZwFL7|x%l};6qZ}&g$!Pel2O=?i=2%uKw<6rLYvu8k24Jq~?%{=#2
z*zB+a)yuHg$N}DAzeA9|kySq92`q3!3O;AErS-_@X2l8eh@;|3NCPX0p?}dprZHYB
z1|_w}D)~qn1-S7kpyEOZoF-Gq5~bm7bj+8aqy@W?FeYL$)WcF@>e&bpmpqrF!C=}D
zjxsart=KjX%C_cERLV8~@N;t@!z3Q4#R7IYVb~w!xm3}aUPPFA3hi9;=QD(|tj*c%
zN>m^-8<|dUmck5nxH~@<>3@%+D%2!U$<ajFf<*dpX3Og&xRSLS-kO`bSM2#f^>Awm
z#ycs`sdZKw^`3C@_2$(?5$M3m`hhdyG8N$~yKvQglMRH6yBi9B4rn@Jo97@^!qM{T
zi>;T9oi2q>WXQEwJ7NRV9~|ubsqPF^UoP|SwnME=ohwaVrn)_RkAF@(=dcFfFfm__
z(i^?*_sTQc_!@T%0}x1ikp7&d``^BSx&{-;+6TdWh4>^>hA|i%OJdco8M~RHIDlvm
zjFH15lSYb;Uh&5cC;(O(6pqm&6YSp~dVyF0XY{^Sd7vrTwy!m&dp;e2KMR3n51euN
zcXnxzeRpGM8#X#)Gk?tl8-)E1qQee2i}cUr7UlY{0R}c@DpMy2fNc`;Y!?JLOZ^{M
z*q061JuUw+RKndFeTo23s)K&q`!Y-Y#~kh2cRA`6T~Fa}tdgZObO#BLFNP-B|FT>D
z$DZ)j{QtIZ{v9IghXyDOL1Ehg=V1S1&-@>|{r_B=u?I5U8h<#5=#i=2_mXHT!mxip
z5N!`2sD>c;1B&I52|@!{_<RPx1dn5?)g}ZbW@2PyXJsXJF?DtU9A)ig)Xgr5$V5?9
zxofTT9Yia~J1(GMT)_SC$b22W`DGfkJ7A$3zaLxphapmZ<ZrrRiV<&=fRo#uSvSzR
z3RpADEa|4kMt`(1`d}RN@z)l?9qXhrZDXiJGpFzBP0TVxz9Z3KY9m>~-HWHhX`<y}
zYXAU^u5^=#Q*^gND`k{J6OVrRUhI4$17`0T?hs*PrRU@)ZKlT^91~aN7TTL3CREf1
z3IS7Ac-&5aC7ks8-$(RW7LrutF-}jl<6eL4J{O)R_J5mYejr;}K&r*@YerBJ06qg{
zgpXkYM80Ng(ka_iI8)NW+G70N32(hxyL%K%?s1j1K+6AdAm931_mYg8uU=;Rw&O2J
zt-#4%8a5ynDCIDtm3Z+H`EjwyQ6Y<p-utRb4SQ#i?Q{R)S~-QKcnHL=U)X>^VJ$Ox
zE2e&tV1KzKlwlg$?q)q0`EJL_fXY8%+$#J6IV1<~MThiliBnr!4?G$Z8J^0Pc^}WC
z{@Ax5vwNaByP*H_s`upAEC)jsj8xwz=;mM@U$|#dB&ZoUUrEdDh4xG`MZZM@pu-W%
zc0dEZ)`VyJYdrnj;_++!rt)L&9EdtBHttO@bAPbre{YVneIjbH0K1<E;3r@KzF#ZP
z2^jvI2A>9x(?cH?a;=`l^1$y%V>e;>g`}CGT`ynj#0!G%+wUe!`g5gci78x!SBY93
zBvJt`u@59BD++V;1T0YWe~H1mKr5$tcccDD{}_z7Nvc%iEDisMX_~|-H~|ZS`(I+l
z$A7Dl3qP*LS>bDKteHOtUq$<3|6#@*aQnV6O8-j?K3w!%6`Xq+sH47DXYsCQ=z)OG
zKTO6Gw!#TmFwXxHL-{JMCX^*rUTHgEkaS+i4HHoH@()u^jYIN<G5B9%d|W(}(W7H?
zZFh-}=b97M(#Ee;|1&o-IPWJ20f_&WwSO+5aBHTgXJSB9XqudpzlE&(d-CpO$Y3h{
zsM+hZUxD#k`&NAr<y}8{TK?Tc!o(eK9JPCuj6Xj=d(O4v`z)+xtT`{D{$W|1%HdIQ
z(!VHq({+{MH5r<Kl~^C>3hy@6V6LW<VPXP+LUbl9>?I`0)_x1~472~7T+}XbP=D&y
zW$&tgeC=+k_=xF>jiz81Zd=2Ew8w-<hlN8YD@n<|bkl*J%}jw|)**h`(1O&>HbsQB
z!F*iCeCK6wwtg`IqwoY&gsTqifCl3H*HJ}?3L^gh;#gj9pam1q&I=E-14{Rw(Z~PC
zsh{e=KDt1Au_m{n-Z6X~%~XHtR)0JnKt(?Tj&TtqtwS0Ah0B*#gl7NXJ=z*TFb%$T
z3>qr_@ATX#bFWf}@7%2X(qm3mW7L6nhth=2E2~}A`iXTfr>8){qX{X641hfq@n{E>
z^=s*f0o$KIa3FY16$ttS))L~MC#|xSj6}sL#uYte<XX<JC6YKJM5MWZFn{_|>jCn<
z)M4P{|J^_7MV0lHUF@Aq0q=1Ib8`gN3K@TCSzi66Jm6KY=Hj@}-^<ppRMOHtg!l|&
zcVF?rEtiT+YUK<;(AQaG7V}tsldh>>lsd1*_k^k(*Cw8U5K*a9_wRz*d|itKu=N=R
zz6(}M{(tJ)F$mF10Al`EoPXo}*9pVE`WDB5E&)6DcD>J$c3nN_K>&ROFL+PrRVxz#
zf{v^`RSBSjFA^k%<lL@wdY;B#OZ|tb^~2WL0hRE`v^KUhUUf)U6m4dS)|S!SJ1HDy
z=uUIjo#A!x-8vne#yJF8Sw~#h0d@ZG-2J;^&HVpiJYVGJ4`eDU`hS1m)@k0|gCQiN
zXaG(qxZfVA@E745VDpQx4Hh>Vf3=TJrAn%qM;Rj%Gfguu_m^aPYh=>qAA=wOd0`LK
z>HnkWQQgArZ}^cHhpAXljlWMqV!=oEAN?;RvOUm%Z~vbL<u3M-Gc7XcPI2G_y}YBt
zm4j4_{~AzTAuAn%1b-<1Pd7Og*<AK~DuF~vl$0!E3Vyxcp#VD5&}6$H0j~cP>7$&A
zs_uKd=l@lJ063~4@oWRp{uxIzO3HYygKrK@#JK<@pWw-sz*YM%*PQT3aumK97$O6p
zpMaz8f&$mT|9dtlb|l2X&=yH8`9{4QPmRcn*)(3Q@Vyg=IDdwQmk8M3M3nsl!tgZ~
zJix){Ajlv{TrGN5Rds8a>>Z<ORbp37_0sUHX!q~I9wg-xP>ruh8uJKHoc*7DgoLd*
z_jiKOn76NkozPvTIMU<1Duk}jsJeTKTpAJpVFO9^1T+Bti~b$3{6+5pA^ERso7y_K
zc>e#^25~t;6Mv|BX24a>uWDmprH3;Q1OVj)%EtCT_0=vY0P;J?IEzQ7^1n(8PzjJ}
z=z;RcJlE2;L0858_%zj?ngH+>HYhRHXwh0*G!{1K1i5&F2nm+0&w++A*R#yk&ZoDr
zw9_s8;s*_y9gtbTMk8YVu?>&ZWwC&j!-_(*BZL<7vVWJMKw-6)&SjozYbi&DD(l^4
z@-mq@K9xEDIq~Vs0tQ$~XUPCb00;)K=z<`tgP`J0fTFD1Y7DWPXp?Uj?>s#d_FuyX
zq;cp&!N34N5sZ44?RnSyG2iCyt)sb`h*;S+dsgZ^G@qUFOV%PUtX#JPvaUB8e5Y{p
zF+OTI{C@%n9!x4bY-5mS$D%JxoMH(z2x8ILZ(dol6Y*T%1ABf!_a*Fir6RQD_mN0D
zcilG(uC05PXWZX}pPXH##zsRl9luCaR!q0hFAh{|kf^IKRcyDM?n-F6@jjfF!D^MN
zl`g(}t~VnLpW!u68a$<4nA0vdY_bg7&O3p<;(x4<Q71F$p1R&EMm=hkS1O;dF<aC*
z6FyAT#NaGiST`bsWI~U(*I~YnM=e4%aEG4#UN$G)A-}g$%IlirF>25MgV`AWL;2L%
zrQgXzF|jRqCRo}(8z}^<scc<}oc%i(j=f8gZr*g1yJb7fdqO7cw6El2FsX6MxO{gp
zAAg3P>auVF*@4DNK|d>9J$*7dnppg1z4flp3gl}ZRBRy3z5|S2@bWH#2a?$i=@WKW
zK=qFG6J%9C!V`#JNY0M=6T(M;z%GM|1Dl#5+tQBLMhMcbnXD0M)2`bFu$?g!>I#N8
zB)k?$VHX)UO$f?4Z5W0(>i8PN2W@+h#(x9D{0i4EC?_&P1jXy;bgbx`?3Ab>MLq-<
zJV<m<KjuN2M%?=?{fMviLZ^EZ@!*;{vGYx5NJxmNi@zkaJWTdcbUr@?r`}vRto(m}
z>dHFkFrvspP=XhOD>!3v&_%;wnv3uLi2J8_FDH?aq@@ZOFUoztOj&rUZup!`&VS?9
z(QPbh6W0is$`8pdz>tCP`Muiu{pWy}P_%@bK@(Vuz3n;oxu0vlFl_x#*Jg@u*Yc^i
zOB2{3TFIz>8r9@5ESRjA1ZCO{Lo@6%rXB2cU`4{?9&{|Ad7-(EdhIfs1iv7(rmWW&
zV@H~P><r(p>4pC2D75($TGlfydw*~)vaIfxdm{7;)ZcX%Ei$30!Q?Ok{gakZIV@2N
z(`*FSv`1(&jQDrJuE;oNfY1kao-=|nJ)3V5{Ty1kf!^y0Kf7Nlvk%JwmtjY&Bbs4%
zTz0!F)PA4miNCudziv~2XYMLkxa4qNbNPCMrRNsfP+C2lHo6AI!v?<U)PEem{}%s3
zu09S7YzlY3D?;Jv;^9e!RZLqoFH5#QR~E`E?w05NFYu~F0x1lkc6;2fp`{v6qM$~1
z{;)OBu3e4&Y=W26nt5gVV}8q{IMrC4<vZ+azX7M>TqJVncaa7+r#jE!rZ7=wnROKr
zqq^4C##WNr0^8#SbYE4H#(xa$H6+}v+hYbDpX<3p8WtmV*m@N%b0I$s?h{XQ;Wn4k
zxywS1PK+>S!;uijR5Wp>jUzGynExC=9*L9BsA{~Flws81?DJZ?PLFs(8LtYlw+{s`
z+SA%dLYa_t{GY0bZg&<J@ESX*TNDMw9x_y#qG0q>J$n&y0nB=t@PFH%m%2}{0$(^W
z_-Oayb{@lBDC^z!2k>=yEbg8+BG3rmZ!M+2X~{i)-SWLa3m*BWf0}+U9QechKL?OJ
zs2c=R@V)2{I*|A)d6Zg6#Tk4ZF4S^NCI$c|E^Oikz~g{$Ezm0=5y3@+EYRoc?uI*;
z6HEkFghGT+#7oK3v47W0D0uL1W;DD(PjBaQZV2bNnQJ!XM1F-qy;2}-BF>GW7KP0B
z4wM1p%?^c@SeU3Q=|ffuH2B@Ro&1i>w+@#G=YT0`u*hS8Dn>Cn<uTde@iC(vE$_(p
z5qQTWaE>?!h-dS9;$ior(Kqhw#CK0lPHYlA(=_~P7S~$_41Y!OIvdcJPnO3cIMiZI
z5OQ*{(H1a|Ot{ezS~WnvXy{jwi5l84k*N~e@b)b<X(E5&AmJ^4V!<f%W+DMTHz;#q
zAZen*B{b;`m5(04-GDY`2|SbhN{wR;KP~&_faYeP0L%*T`3n%>itSf!>+=Y46wa0e
ziiUrJ50!taWq)ZrRbDInsB#SvnjD~@1S}aM^8Ez?uKug;@d5jv10Xpd@#InHq-Sy{
zzdKhAE|iG}$4wjS-#ZM}8?q6d{%OVjaXE7w^R1dk1{;dCvDtjfW8Yht>=Utyj~t^+
zNC6GL@J{CcN#y?{Jo5p-k%6}V0RgUo{<l+okSLYqZhzV<{-&CYP}7;c_HJ*2Lvq=A
zfE;qT+<!uv|3?^K>HO$fnlTP3koe)#_UgOVwd1gf0H_j#|FaJUTvPe#ItyUptLp<G
zaf;|cLk$Ii)PBZ5@(6@28S4Wvpi+ChsSJ8!R+ls-2Z8bt-tBnQrCFD<I$e!&CRY9z
zVjb)MjepQX_v3zWK|Ems`%6FhBtCYOaz6WiLYSRnkC*}1u>RlA*lWKMINOL2VlX4T
z=&Y|i6RN~1MqHa)7E>)WB)NE}->S>wwtfwJ4&&v3Iz@w&O~qA!;tk<|xugXR+}40A
zWm$$JBdV8LK3@#4<A^*yK3)#dl5V|4D%bT(qJQEzQWGepSBvt09;n){Gs1WNQFj<I
zl&W1jann+OTg#`J0TtoM4e!A=0oFE-B;^0ZIISik=aFKu0dr%LjQ9CQlyhIH4x%+3
za}rnq(h5YTYk;Q84Kc^?*3d=*d&INMK@f#!Qi;~pW@HbN=mfJC90c?s21kOUf?gt*
zoPYf}CKsZbSDXlu+)?2Qrd&vhdd}Czxb%c8=O|~Vq6<tgf}QO6IU73OhWI^UnA+<@
zIn+6i`cHeCvf^#iHevKwy4`iR7~D|Nx}_7M^-t)jb)xUz97ru$&4VMCE-k!7&=4dK
z9F7}Md8w@}l5l8$@=HeRoc65XJlXVwpMM4YHUGmADEX}jHfJGVk@{O`A3I(T*kYCE
zTnJKv#MVs!#0N^V)U(rlM8+C{jsS>2cfSfz`D3^hGqmH|g(IId25gV+k*@Vnl@~f*
z#<%+##T|?6AME8rA&#J?Cl=oyc(7HDDbWf_U^g(%IdL%Dqbuzy&sqB_n@0&oFYEBi
zs=$9~)^0s+em&rToR@N-y#Iouov$pKKTPF`Q)6|0S=cCjxi<5BcTh%fiZ{iuP_7+L
zyfe44)Z}~+vi+gTrl;A|{r&2vk4Q4FO&|36D8oe%E(-T3ttw)9N|c;aA}pv>?st(~
zGR=22&7ZB?#T^@aL=+%WGU%EzlTs%Buv&kilBpbp4}s=d@bAeh9d13BulCIEpM9+%
ze6D{PppAzdV|UZUL}bU=<D+Oh?8$Vr7RZeqAafqdX=8nMr@UV0NcljbH<4rME&E9r
z%69-Ksu_|6e=WVNycQ6rEQ>|ip;8+trka5#6)R|6<f(9h^q^GI*#jb;vKPsD-Q|C8
z8t%&rXRuIM?vt5IHNU7<Cr%U3+c@tJ2*6e4^}V^M|ID}8{FzVE?|biEZiMv6jFO+E
z`(7HoICHQUsWv4GTm+yJH}R(xARR)qa<@ITogEehh{bynucG^^JSUO)s*nSKfAMDP
z5j--<WBL9?PpG@RA(J|OE}jw&w@QCmoELxtyT<D8h*a2Cm(=Rp?bhQK4az|l+Qhtv
zw`@(_LNJK6o3=FZla^x-b|{z`JH!G+zN*lCvZysmL{OlTt15QyQ%AoQqVUOBTd$$K
z2nSZ^*0o5DMvI!sHZ&d;*U`Kx(IN4qxsvhQK9aWN97Ir~*0%%KRjECHi_m|N?bxZL
zunQ<ico~Y?j<1cZQ{C2neFWvBN}uURz*5&*0@~{jm6OIR@~%t<2DNu5p-bd68Ye1S
zStrkSnbEfV&(7(;^i*SNsNl9|44A36FB2^F{gih~t;<H@#q+nlI}P8Tpm3hGNh|9=
z!kH69_c807s4?VEtwcW;5LADa%nn!=`?OG#7R2d76Px9?N2z;jQ3_q=W^CkQs3h^U
zB;%!6@^3e+h4fcEJOr<h{gol)sPX<-P*`$y@_s(wz5Km;*;`y)Cpj<7SCdC4D~nbe
zqx+8LCRA33xc1xGiW<ANl-)@n<X2>*6NKY{!SCeQr8XvXQVJp__=bOCZuL-n1W4%m
zIj9P<r+X!IY95cIlDdjI6`X#MBn=H*uu3H@nj=&pP<0cd-`29^#tTa@%No+I6sHF#
zrzVV&2?xQ8i#z&Y`GzWl@)i-zF3OGw4T|04Hk?l!<N12;TC1UUk<sr9;zws)_lbb2
zf#5;t`Z3g#CuTi+A_sqJ92G9n+<0^=_OP3g3DjIiH-@pcIQSakpJF*QByYXr_10RP
z>Iy@~I0q7?^s8eYp1hFqL?&l{x0{LIE{FuGbL6`}>pQ!qmnrY<pBe14w?e?RbGVjw
zYL%@j%8rtlJ2`u8f6XZYM@q1>uy}c+Y1b*1Lfk@`e(xU7<Ai@1K+$E!B&YbbwRw!6
z8x|@?yt>2u;zoH&J3aeZWlgL-u5xHCEw1C6H1;e#I+mD@u%*)_4gwiWsc>S|kK2BI
z0%yM10gDMX`(4?S9szLbjL`WjpN$z3F~77bv%?*vi`C`PGZbLpM7{(zxM;U4e#r_^
zQshN;{6kq09e{sMZKUQPFpJL@7OQd#@j2XU71*GLKR$M1<tOOGN`&yj7ng}oF#CA2
zGrlwFuC~mqi;pc4vA|VehTw(8Tv(S4**aT@L>b0&b9K=6{VsAF3zlkPF@dB+GWbr0
zTs`_)xVr6VBx9`KAYOzVfmyqt7}IDJEpM8yxvCJNwkv-Ryh4FJ2HIz~{Wr~MLBRmF
z(GH`Q(j}MmIGV7js3$zL{>swbUYR}Z7{xRrST-K8XcQHal-j!-KFGFXxntL3p7?1-
zlU%(sHOp<9i!um?qq7wejnvce;pOiO?P;i^D3&`9DYT`G?1)mZr}T?s#;9;zv!9RQ
z(q*I3D;<CAyVqmT+Q1Abyy&aT@JRn1o%MMSLrZG@o|nA`4SS&0a6e~DZHCW5f;}Kk
zd4>KUeWKNE4Xu5Su^zq5AS`?o27k8tFU;G{y4W*1Cx>i+Je-jZnhS{6Y+KJE-PJ*7
z-8y0(M<m!Gh(FVzdap#Kkhfj4M#jtrFV*WZ=!Abx*kf9@oXiUQxk%3c1rI>*zsOeE
zq5lwZkIJTi2V2qa*?lB#Ws!fZep}Ry9=BNjz<zgE_1U${@S}loNT<7IgwDI}Zar?s
zMwmoiz$q8+H@*sOBij=4(5QmP?VGsGr(f^hCB)6pURz!yZW<7joUAbD1uM~lGmN9)
z;Fy1Y&}OExjCYH}NVG0!?|=!xrn|tm+=sW^7lFu#AYN3#!3n*E*bf?8Qj1uGIrR$!
z8$m~H+S$7SB<C;UJ-k~XXN$qbxAaB=qRe|b4FP>58QavgHb5$9{)L|FDlb$TtC_|p
z=Kp436<|}g4t--|fPSonwvXwC^+$wl((!-&v#rP;zXgS^a1j8%GrrE)>$?TkKU#o=
zg})?wL0gTLcEP8cs|<tQ%h+JXVibj&y2<q8jXLH_wKM>;^a0~Gt(Ct-ncbKU*hArf
z_ijM6fKe{+L45OQZw*b)Rx;Q_W+tpZe9t$P+p(=vViDY@cJzoHtotd>{ymJbl{|mp
ze3Sw(u-TA~f2r{hBd}k(>tEfxxY(@tc0X+4sG;6YZz;dh9)A4EJ-pYP6M7Sa_Hi)?
zK!q6CZa+T{Xz^L>(QP#GUD9u!|03*@PlnxV`3ZjRj0NSJ&|g(HZcqN!rSXx?KZL=#
zeWh=JKd|xd+IYO}uVDY#aa`X&_pyJm$@l;8*|8DlmY8&P`hpLV|CQ@!$6xXl^aOu2
z6?0X2GzNorOlsa;`ZL`6>A~OSejvH6MW-7}ysi8RjSeCww1-bqfi-bL$<t&XbDEE-
zYRZ{3UP_wVN0QHuu-KI1o|TS{yjsv;GAk!b%Z=+dE9iTZet19U&!P%<1T265dIRk3
z`#mm!9hKQWpIxE9kly<}8-BFs(}>3Vy%v`TZ|JI0buD`mw>F-%l9v*@opM+)B&1C5
zsg-ym4RVir4`fPyzU_9tD0R*|4Bq-(W}}PS%{qLYuwM7OeTzZCFDm}mp1qKDq1W_E
zuJI=G54U%C=&)F0eJ$xnt?z#zVExnWf}dK6mqs@=-p&1gf66FdiaXq5!rt*+{Ojfk
zUUWJ(c=nG$_<5naiqwnxaE@(F%nrfA_j}Av3wq*g(abzM%M*v+WxX(RJ>HgQ7EFJf
zdZ-)ReD90^{${6{x2Kn{E`AvYWmaSjAHo_NP8F^0R6Te1;!cdW1igQ2f50|A;EUpO
zc{o+-!j2z^vYnrBH24otBgk03g9CV+<ydWR+cpsXu3teUAJ!(Z<2Y|^fC0-A*J<21
z&YU1vhQX#q+EOf&nn=YjL;w4Zq-e{sC8u3G45-lt7VqKlcz4g~&fDAQlP@HS1&xH`
z)Tu-m1!{%&&0^WP_k4eW`Z05*&l!?<NCgV<z-L%5ggC+@DglbRuuQuMGXh^})G?2e
zqlgL_`L0CLqgnM%x7&5<wR^O0j8Kp2&$Hmn`NKizZ}^YUmQS|M(7~R0P*4WP!qV0=
z_gF`aeCcf^QaZwZU`4;s2_A(3ExUYV0e0&($Wc4JRY)wg2*ZDMU#MMnBMR%JgQb#H
zD+kq5<+xNmysx}Ft5nX8-#uAm&*uPPaS&LI!)d5BT14QpAtgRbugS45!G-B1I)Me&
ziS@npP0CA`3?u&H6SK#D@92wJogN+0&XU_)9`L9m+=Zl?NPN-3WTcJi<3Lcm0ke*|
zduI^wI0UhbVhVr4&wH8BRM&+DyTiQDfT%DuAkPU+FW@eC@Kv!nfPddTHel-ah|S?w
z#b#9s7Cci|ref#(X=vtbZ=<%)1_9M<v;{ycmR9(=f94ajxiu&Lnd9uDu1~t?O+^2S
z{fLscsD$o;>>n=`)p4TU(pu$BiqJ)L-azI&Dz7E-YiWO+Nt8^QtR~4yiWHM~Lwc+v
zM;>k0CPq<O6ePvxMM6A4Fpx$;d3T%(jTaC@M#m;6P?JlFI18popAQ8+!ji)q01hx)
z=^SlKeCxKoT;sF(qXWH^La9#))s$G_LM+6XM`SIugj$JML2X{;KpzsZg66^uw3)zk
zDgcg$$~%8J$QapHVn})gwP~^zU4r+n5i4ktKkM}Zk&GB&*X1#j+6g9nq7(R#SDB4q
zZanJ22B4k=>R6=Wqdh*)(lsZ8$F&#-=)qjbrfSZG=5UKc*0(^XJeS!p{3H;d6j*0e
zzK3;oLPMfq@T{^I@yHMWy&<%Z<3PSunzBq-Qks8gN+qnuh^)#KCq|t3ecw%ZaigCU
zF&Lr~pa2xjgdVqf%<wQ9P=qBkL%mp$YF-;HNgg)W*@P!WpA=gb6evKi5@CaSONRv)
z3&3ecCj2SZzMe$|8`g5EXu~TQm2(it7M!Bk^)?YJ*M@w06&SvOWW}E^qFQ>(w;<e-
zzE^+HPQ7FHfGc`t81WHh>MIxmT<|zSge(3irbSf75d$I}RgMsiA|CA_`a&53dtEF5
zHuh*lHQ+sZbHQ0ZfMMh<8uK_HPEV;<;t+fqMm|>O((<6JwlUBS9#-5|c4+Q_D+o+9
zLQOUT3|r8jGg?j+3_=5Hip8b+reC6_LY#lUfF9fnlDB9duDx6@%1Do0zAi5i=TQjm
z01wzXY!7Z01jek?c;;wLh;dfZMyV}2KYfD0#2`qVr7Te$QVfNO6N<XkN(J2xyJg+9
zUAJ$WYGw;LgAhZR>#OO`?`s$6PHv1rgDYdOVQQ{PiHaTLvWz}54Fpq0aI)Q_?qYuq
z2dEkmeFF=yW}@0@YT#w<Q|-R?Z`#ednK~2~aTHlU=nfa|Y?f4&jseW1s_J0Yx{h~w
zApH+IS4hDZMIp(DRQH&K93Ejo4i2&SKlHZWxIIP1{V7^)Q1ves?e-NK(&@^N7Db_j
z%4fG_qr9(mz3R6I>C1pByS7?!4+?*Y(r4#7^%Vp{c%?{SbPk_i51TW5aWHkOHT`&f
z2Y)}-&i&To<mB<@DDbYDl+^pbhOO&nD||YgHsW2v*g^Z6J`d!PV+!Z-`;+0x@6WS(
zJ@l@s_YaeCV>WRcmyHIUjfY{MO<KeD%`mQsu>PCe$4^i1$9Q-5)<3-)M(=;ZXmI6P
zRf0dBG#`Tg<t%)8YE-X|Yp3)Mw`uFJeLk*_?ivptt}X~^;#TF;u6J`f^FB0Z&mUew
zd^5NnSAN!)U77z6SKM<BoQA@SepEN5C+$r2+S!2;O?P&dM(`vzGgCOMtWDOh8^qb-
zl)RFbo8Lc5*ixnDoREdXmM4E<SVDZPUS_?ie&f92rLvfTQ}rerKJEWZ1l57cjs~4V
z5z`0-1tmE^ZE)rWy3H_e!HdO=zH|=}QddniWJ8~_GSaJTEV$E3N<u#Z>9~B{x4r!1
zqT5MYO``7#uQSi~KG60YAma-#kTWD=*M-h*PxUfg)lNbe3O!ZKjJSUf-Y9Fgnucz1
znr-lwyjhqL8yY;_BH11fW)8KYwM!3rVzn&*yUEm9?!FM7M+_P`Xv~G`v*zvHYKQfu
zwZ!HU%q-478Cg#5v6KOLoQ;%SPunmQhVSz$j_58x8rjC~DuKkHLKR;Xl-(edIL=Ap
zHHkCZ=~_+u?>lKe8Yq8pxvHA;`h3~%Ga8NHLLoB<3<Aa%WdSk*lOS9PbHf7AYi%%d
zoLZ%~GNyPUme|TeW>~;9<;g=}X$lGpsfCeBpmt%N2;Cj|BPbpqXtu-<MBb$I3T6^G
z)QI47nMWf25(|^Ez+Gplf8;uIsuPJ6bCzwB_|Nm{?c{NOemj4A@oK$S=W`8HepA|r
z47Fid0i5O4u~JiMBdJYIJyvalE|7~z@TMKtaeUu*f-(<N1Y98p#TE#4jd3j+^Ll#^
z=hK@<Z*uqZv`{iExe+qo9Cs!ws!9oDX^PT}>!g=uR){>_ZjF9Y)MMuu+R)|;;1YAA
zUDt)@%lo^@%Z>>^XkDUBu)wt~w|E8vuQUY!8bIa0w<kLTS_`+l%?Q9W1pq=oy}!47
z0tp5Xw^oY-JTL_SH9*S0x5GjSH9!Rb7C`C0x6JzkxCXbdi3vjqx6um(8!!a`5<u<0
zx9mFvUkkT#&Iz+L1pp{O*T1(V0}B2Rw+oB}DlmWSeFvNq#rys#ASfUT(xk-$q=)V0
z=v}1uCgPUayOmsWmvVGO1w^WVpdv*DK}137(WFRM0TB>sQUw($hkzjMKeL-;H+KiO
zharEzzl4v%UGio(JM+9V@4WBxzHjl-T<`ro<G*LIxan`Md8bi+{`RYUzT4x5|M2mo
zGe3W}@7JtYxzE3tllT0@-L3yh|Le?xCNpl7m?v$z_<Z3{T3kCfuJLogeY4i=>SG=`
zvhkZ=o8(zzyHcR!x@OY0gi?jie|@s|tJU1&KRY|9z}CIHO1|+-xy{qdGQSM`r{CKS
z`?AAp+AUT~{Pp{~my*fW8J~Z4sOhc^Gj@MrC*HpCVS#xco!YT>$Vciw<(F2uHgxsk
zUc042pZs!R&*>+>{A*dAQCl0Gs(yG?9-H?>N@|*NVD+k!6X?U|#ml{Br2W2}tN)Yz
z`IW<_P9^I19=77A8si&-hF9`TetqumLrbh&+;>5*n`3&796LXKbvxphUklV+y!C(d
zipquB58LocjdwPm=+&v&{_`)FS~l**3#DuQd16u>w-bNLeh|O4zG}a`@9g~htNh<K
zD!#jKvFoq4SXIC5inVVH>XCk<zNc>ec6rpiz9aoBB~IK)t<Iy28~4+gll4Gp@7=gR
ztNs0Yy+aOs?#Tsz71;1g{$Dn<cej5$KkdMqokprxkLA4^rylxaLf-E){&<hvQImb@
z_^jtDazA&UKC<AX#&2|%r*AINuV9U1>cmZzrU`F6pPrs{_<W(dKaU;$&RR!@TTMsS
zE&KFKcV-MKzGSJI)@|M3SKG~aq|k3)Zmda=<5zAiaB5+xuA{mQDu4A-i<5sd(?9M%
zw@ATPyMEOS)O`M%y`5IQ|M;IDO=>W$Hk<EY@$`h07fTE-G<jC%HP_YSZO9!5eR-P9
z-7s}ti>1Tc-2SZ1QekGxROYuGM^e_5ai2fb@9E3ymY;mASD7I%mO4MYK~jxXm5CF3
zIz2i3vl|7@O`mdP!|Sil9DRRk^MQZ9etXScd1k?tAB}ryXSG8cPRlC_KK0$i@xw<n
z=ro0Tecjn**WYY&>9GaJn@?zaY5$4+b0;xVe>pUucjcFu-+Oj_$2GZ-`_q9JzI~@g
znG-`lyfpgU)+eX^x%ANC9rMWUP3R@d=Olc1eCU~jO&)(F`P*k34=;a__o;?G&v@65
zO80#AR@W6HJEd&iL6xmEde1xZ=I0vz`u3sx4Oi_dKve$p%c)CmFWK3*n$m03M>m^n
zJ@>>&@LJ7BXQ%ZOhPMBrK$Xu=AE-H|?Qb_5oTxDF>bNFb_WZWG_FI20tNmn?SA`Pq
zk;~E_8CGU)!LmbNyqtgk!nN0D{Mmi%W5?^)xwGe4*QVD-c@7S)ddc;c^WvAw_O<l&
zTHu;?^2E;{bSYCtDzL8m4w1J#TBhQ6$4)-B=)~<ZZ4P)3PkCxVsS969_A*kQx0>Zi
zD{!{_*p<cSpIAR~XvOPqf42RX;s=)9Os-{*|KNhX>B}8Q4*7qx`03`UE61KMTRzXJ
zj}Pu_yQkL&hYmQdRCu#NvASwT(L0AX9xFy&EZk~f#l)LwAJqPt&R6q~p55NSqz^tE
zd9?hd;WtKAN%oyOG4G8p7P;2GP`}F2-Sdy!Snpchzsr=9g;M4=Y*}zbzOP!IIxu(J
z>qE;W{gU7N!}5RiPtO_p{2t+{?eTk`tyWuVP`~RRoeOi-Q(kb?NZq($;d`48^%*he
z(un;>zfktfE3_!Jkl5`n$z6QMk;9iZr1VQ!nbJ^r@yi;A8_#Y)ZoJZ<fZFxYFCQ=J
zaqiHH^m=n%manZI|5;C7>i;cK&@r+4taonRXxybyl{0^@{<G#p^0rO23bxyGx_q%q
zbH}{)+2Mktwu~Ddf8v8@5^GNUEMGya0>0t*ja&M^HN3)R?BLOZyHC8e&)3N*Cnx^d
zbz%z-`BY=a1ji(<G5K!yeIMmrb^2<zt`n2eCiWk>t@ZkSi_Z66J8Iwyk2FZA-uzCV
zDj&VF`Kf;b_2S}Bwd!2hb72tkE>)z!CQp?&uJrBu;kkot_D$;jQtQ2x<IinG+bze+
zOdnAwu4cn7We-$7dFJ|?88tSo{&dRvF|*oEA{Nh`F(+ROQ0MXQ*DU$si+M9PzVPHq
z+nX=<OI^6T-`oMe(EFB;E8X$;qeUxCQp?wSd)$BbWu)=Ut&OAVUGDqC;5@0bd-d8~
z{L|%Ew|u?phXX~I+?YRo$?W)Q(&dHSQZ}BSnR2SnjLV<OPnP}sZ2pX18=t-N;oZ0b
zyT*O29-4k=$c3Mmk857=pySVz?MF2H;8c9Os+V6}U#Rhgex>vOSoiVc((&3w4*gzY
zU8R5h7llcOccibF+K#E@xJYc>a4PBa*nDwCQ`*rbH{Y)Rck{1SBv)@$xb1I8OO|-=
zZ(_vz4f6f4prWhMxH$>0uUgW)NuiP%{YE$k*Git(zs;if<x@{4tvld8G<<%cG38rN
zT=vKLs+AsD)uef^y3;53<Zsql;`mi=bn1WD$ZPK<4>>uZ;HDE_*6H~MUU%}`DbM`<
zWJ<R|^V^qRG4|0O?t#S*6&}!S)9&HNid`LW-tHP&{}0dZeCIwKQoH?+T~2pxmZyHt
zU3=D6Dv>uX<;j%C7j?KK|5JJA-rI{dvoBX}J%826f=ZsWKW}YI<MI|QkpJRS)$f0N
z{N>T#d+u3s>9x16mfl?T#}7A;W)F3*#yxF&A!B*NxVY3GX7-&mvzBMakYkq$^F0TD
z_T~KFDm-~*QL%&S`e*8w{e9YJlXmZ(+<OSK`_IpQpS~b|GP8GM{xU7{#GS6tXy4ja
z6^5^^P<+Cs8EotEbxJ<n>*sCGpMHPZQH<&L=;w`3&3ZS_q#{jrrrvq(QiV~2UGs*l
ztMGgt_cOble-CY(xBaKnl)T^e&p*7$?wyyXKYN-wR_yxQ7fTdw(Q5r(@0#v%`hd~P
zKHoC5{hrVG_Kk;?x-{U(<BqNi)1Uv&{Y<fXeKxHxo$PBlZ|~Yxa`LYwepr8V^zbv6
zD^DE%(+gW$jT>I%gAH}BY}nbd<Tn)#{?+BD3FA&)S5NdBC-!K%w9oC)e3b#uTzIm3
z;+aB^96d08`L9C@?!A@wpUdZ_(KD`|Khm#Fwb~V%e1PxTaD7LIzZ@N=FN!;r=Z9Mz
zUOTa~PScTf3Kn=}%{1ceN1K1|se|wSeooaR`SVxI!=IhLzj-<}V4hN~Nw11~uDnkr
zKlvz?%<OL2vEa^fUyj~ZXh~dsvHfFQS$X`B{+~DL{ey3Nx2|m3m(P&JNB(hiYLiJr
zlNUaDx@=rZqk4SaZab$r7c6T1cgchWeS6lM_Hk0>Nfm33cs);<_Q!u^cfHgNbr1Bu
zbgtu`iU;k}oW3&R_vfGP;3~c0#*Fs2c0E#X#zzyUUpd`0wXN&Dshhm-c0D_1aQl6=
zFAjcsNQsGw`TpGYZlO{m`R#STy1c#a+5FQMU0Tqn*Q~;YTWy(IY{tvKUpun(?=ODq
zGPKJ}MIZYyVNu0-FLi&Nvh{<xzZ@Lbc-QY=e3IV0a2aZ}l-B#x?I7Q<?}YI!n(XDi
zy)<jo@`4LqSausPdf{ZLS-ZF%B|z8eJzCy8y!iaz;|pv_o>%W+17-Wg8F%BBbgs3m
zucK0z&9>?_)#+6mjeWG?R_;!h=GV?Oy4i^B`pWi(+lvj$Uuu6_!E@6#O}%!0(bONS
ze)>k|_Mmft#gp2V9b2^J?y}<xcl~-_ecPr%_2#`*<<j)D72n5$)l=rY-*Q=rB8$J8
zH}UfH3cHr?SupW)08Le?-N?y@znq^@WlRa%;iFyuIQ;jxu|+p;ob~eif258cS>fje
zCuH)*sXiI4%58tS<gC>&@6Cd#KTn)6@XE!p<*V$wRqoQUo*5;^{_)<5J)7TIQ?<^s
zEj!lxxo*)slgO<N7nS(oYJ<1y=glax>C0XnGj3l?S5K{YYxmOC{k*S(q$Q7Ct7vOG
zr+C5b6-%9aZ~3g$pMO7mt>N{wbI-o^O`DS&CO*}r>xF+KPbNHCcYL!B?_H~My-WT2
zdlp?^*fC-1n)Dm%)~~$t^4uj?YfKp3r|94=r*6(Fa{X|py~Q?8+EIMwhp)(Y<62d{
zJYsB32TKm9wWClgtUbQB-^n85p?(eBJuPkiLT2_S*C)4MHGY>;;M#jn?5Hv0=Nml=
z&aq8@<*9#PMo#Ek|AgFQ`R$48%J<~G85KsqRH9~=3(4EhrvEd3O5=;yYF_W%|8mPJ
zqw0435-<E*uQjiI@cOO+Z@zhD3wwOlgt{G)k2fekV^GI6`+TlKH4pT?c5~#6DZ^_I
zKhm;JzsCLZm!0lgnAmdisfm|XjLYye$>_S~Wa)paHEtB`U7|qcne^Me_bndRvi+Q`
zEefpZXMeI?{$;aLmL#a_#aSaucH8mU(f+fxT>gIS7pvB;=rHAU%2!X`xb^$r6&Aa`
zIk)C_Uzu4?_g~#oDOBm`YmXP1e)%=e?w;3*l>VdNz?a{tymaz+lX`Tmb?Wms9_Ri(
zxj=uNI<(@CzU^O}U9)CiO=XOC@@o8r!}-T;<M=%5>%8^Mpq76ZtoZGP^$;4}x<Z89
zjT`>fmiC8>e(}oaGY8vGY5&p0)4sU!18?p8`Q&FkhJFKgl`rgtcdl4-a>MDWcjG2}
zH}uW)e@0(P9kA%--}?=%+S{45Za3G+x$1xC=G|7LZauND?C~G3&+4`B$Fpr$zHvEe
z@VQcn^7!UmYPKr2fg04ZU%`rRthqg=eu-Z`Z&4c#@mKR6t+Xb+;NJBYh@o@NOzi{l
zSZr;_Ca;|=<0<-ik>>}czEjWhc$Ip8{j}y{sdMSGJB+AVd)^zkb9~c|Q+7CJrVM|<
zd)FRwmEBAAc((KtZO8(%0avl^qy}+G*Ct;m(PDgspVLnaYHOD>N{*>CvDGt)t5=90
zlsbiZ|2g~7`3@~_J~J4f^uxgl&s_gnAO?Ke^}@cvM?JHap8YKUrA3{8dVc?(4sY*g
zYuu1#-dNG54sS0tWp};Q_ipFw{eypSVA<`i=i08H`o|wTiVg18aqGq&E05#5#r`{P
z&Yhq_bPafB*5W?j{O<jYT6(fU$uGZNrHrk0tD-7CUh(g`WygOq@6BuMb&yoB$GKms
zy&-N~du4U{UyYvVEQ~E(&U1C)?iUZx)#^6Mw>kYlt-h(=jL)e#^XjcQTh@Qs^T57d
zU4L5JF7=Z`fBe{eOWy+Tl)B6oExE8jz3xNeW*=&|vPbzwsW%@VY+L<KE9UEyKMmRV
zcFD9&l@iMq9Ncd~;hRqnuHLU$gHeOK|G2R6s@2QCeFKYo^jMy_>96$rr`w9_;*cI!
zmfl`f=!f0KraadFThbOclFxs$Z|u#kZO6>-)7zeYGEZuaO;2pE^+{so#~yL5+51tS
z@AsUqe5B?CI{o(e`p+It#;;G<GW_GNd4K73sdF5DbA0{Ze*e8ouWOaBw{{l(=G@Cw
zUuK4v?D}iNda3Ud4I1nneWg|Yj6#<Vy*hQKt;dB6vgiOa+CNjMc|(6C?)3cQPu}Xc
zb>v5fRxCd`Xxy_0x=z|!qkGd~+h)CgV&L&N_pgtudTkP%-S->(+2_*NXNs*KHMnzk
zg2}i4tGu;K*&1!H$Y0!X=DiNL#||y^hkMT7<)6#wK8fz!I6dup=dRlu&V6)Ei`LSo
z19y*DzG`9J2K1eI%*B5tbr$e{eEUrKAvd-t{p)mJzrSuH@84|4Wv+tT7ngp#>5T)W
z%5nLhS@r9Yv=;B*joVaoaIK<q&Yg6XsxfEKBPCaC;`5ehu<{SL=kOtDffZ_SY5H5;
zGV+h<@zPVBf9du5t~v$yZ5^gxrGFFudim=g7v5@5ulnR>-4lQLR-aWKfBE3HGH<o_
zyxDW$h&%Ha{PAq_DXyW{Y79zxk7!xKGwF+Vjf-5mL?rClZHJIjZ@+z~_>J#~zIUoO
zUKuxJ?!i0l*QDM0q1f0*ZWMX3$<3m-YJU4}Mujn*gl}K>IU3JQ-I%_u^p=6!-dbMR
z^X-1`)r##`?frlC>CN&x_PcQ(y#5{j`{l_ycE9@N+`a=x_ixc*|FgfIUfFBXXS@12
zj((Lob5p9#J9}pDE%Ja`+y1P0Y2bx}U)rx8E3vcKV`W~x@bmG>GdBHQw(O}{PB%Hz
z)#+~B8%aZVC#>$VameTWx3(G-*A$~~uW3%StXsyj@0EWcam6n(d;2D|IWxw?EUk59
z;NspHFSIOBYEhm_cjE@T@$FX@p8WLY;c@EXj6-+hHr80L^ao^*KkC!VYkpj+$^2fQ
zfun2G3W=u^^UrG3)%Du(_bV)2emi5IlU`J&%vh$<ogY>lDL1O-4d#Zz-KjagZQRZw
z{hHZ3*I9phy4_Roa(Ml?{_mYxpV46aj~zem+1>TUln+a9toY}g#93{pd*>w`yVQEZ
zu%5-{{PWt|Yg5wk)1PkY{qqOU%$!qUN$TP8)4hupCVjGF^x18P&fY3|_S(RV!Z%;-
zIpOWW?kev-Qf=hrt(7M~(xU2$*SpneRHxyNDnEbCs&cF7#cQd|rA^5TuK#lI#JQS1
zr>r|`%a?a#dUdYCH~BkF`{$q0g6G$*aKL<n3s0GRKH=}NXU0iS*KISzv7q|v!^(h_
zbq76t#9OH|@!CdiXx>%>)4yEQe{a==Z<joLens`wQ-ldiPoJ%Hx$Tqqu((B^C(ZeB
z(1?HCQ%-4@vyJ>|P#stKTa8BZ`zw}Ox7brQsX+IAUw?cfWzgBWi63pfIHG~r;IZrY
z**lxBFF18|^vM@~o%DFyiIpnP=377a)i=k7_1`k2WU~@2kMv6%@f);K`n<L5@cc!-
zBBPHzb#?ySnj4oF{I2NJAB-vSXpN?Me|>-ZyF4>*6d$_#TF*;EXFZ<2_m+C9M(0}V
zpE}m?&3>NOpDfa7U)mz@$$VyciL>wjc5Lb2!<Y4KwV>%wz6DD?)!QyPS|+i7rS?OU
zG77YB>vli+>Xl=UEn9VJ8k?c4zj<Q%ge8L~tV#8iZ~jP?OL1{=iyD41Y40nmwr+pC
z^x-%7ma_d)+6@&ezj<`Y_EzQRj9c?s>n>lnIB{U)@*34@zWlhY<vY#KKhpi+yM-Qk
zCqbGpulU|Q4a>HjeQ8$H{hOtZ`>(co^QkhW@9h4%?a*s?<7%W`IJoPz^6I?fcjNZ|
zG^pa<F?oh%)UNjT;>LFA+kfa+#maxpJ$&(zpZ1(A*h}xKex_rARh9aIcc;HoVrr)r
z_QIZWjomxGt<+lh9DK7`IhiuB%=!j1yIs2VSIZlXFS#70=Wlqa<JYrF#^s;-{&Owf
zBTAlFR&`ALhw@Kt+xB?=&=YHfp%-6xd#bm<<wKoc<|{P*MoQhwSBt-V{M&!E*PgGI
z(SP^Oe=f~-c~7Pe@qDo0(8z61lqQN*TzmZb(LK-gezVt~KTRC*Q_F9;j_K)-`<8Sc
zK5h5lj;kIY+U+&`*JcYmH2+2V)3aaNy#Mqm_vJNx4?Xw8)YHX(+h1z<q1ydUm6=zx
z)Pm>deKx$!k;^+*dzOFP`NDs!N9CG#-zrhayR83;orS-xN`9WP>eTFiCV5Yto_V=v
z`l$Tnwh8MhAH6d|%(pUrc6!6I-;`R@dGpQw=_k{QxA^j#(ib+|IC1;ApP#Lo{^Rk>
z?;oVLy?1B)GGY1ME#JTTQRP2RFcVs<cZN(@S+334gm$HOymgSC+p2$PrGxmNU02p)
z=#j_DpZMpO=aZV09{${?8Q(9Rx$yntEt`_#Zg-rKw))eSLz>v?csCy|wffMmj0=wl
zRc8FLw($19_GWzDHNDoU*+s?lAMge{N`KQ$oV@Z{p{dU;{_x|OXBw9sJOBKELcPcj
z2VeXLn~^@E?A`Lq3s-;raQV-}@*UpL|Fkpz{u<{;_SsnK&bmJb_xf8Na=86>_3HnW
z=hr>?+J0AkQQNM~{u(%>RKCPA%W60MhweXc`}(KOKE1KQwpH_A=HHq!b#hbZSi-$y
z-H9!IH*ZMaI%WT3bAPVaC+$Ru4ok*8*7@-5an;?u92MG6@4A0HPZ42U{iRi}Z$7>5
zsjt6j`0lp9G6o3~C(m4s*BD=Y)ZYWig<n5T@0s5;dGX1lzSZ0PvSYyP&pzAmS=kQ`
z+s7^Yu-x$9KAlo?(1C8v-`lyY-^a_F#d}9xKal>Vi!6EQm+{}1Dz-1K^*dcU|M<DL
z%$mf}8H>*?{^@^Kiub33Wnb#;kuFX9<h3N{qAs03`fx~<p3`?<bCyeP^Uj#PpI-WE
z$<e~Xta><^r~Y!O{Je2@ZlALa+Om06_u*8zp1USD&bWGY19<PEZ`bGY6JM`uS9fQP
zBai*KP`dk7d_tSFk+0S-*mqK&eRX=g^unUGb@73Tn|gnKa?1YFw{!Q=d%D+bz9Z>w
z*$GDuwJ1`q?txYI!jqPMka)W7kh`brc6z60yFDjf-2C1CL%l|yS>CPj+0Da$9Cq$z
zyNyT3l#^Q9T6ilyw!KrK1KnngA1GBU^7QWO^A|5{cCvoCY1b3;?_aj{iBa`>uDv$$
zG4awzAGUv7ceo)vq)h*L2jVKP+dQyF-*12UjOgCs?#g1n)$Q<o?c2paU0Av7C+mvl
zJ5qVpFWU>W-_xhxvE;MyTLwR0y~6l99k0z0e(LnwWs*(Za<XIT@A3|NvhpX-fb(;o
zF12O-?M3vf88wQ{Emr8I=1=#k@Xc@YmZracVQ_!#Dd+laEjl*uvEd8V;<e83#G{Y?
zRqC-P*RfODjDGL!uRD+UbZ2Vb&);|_Pu^o!#$R!K(4AlJcx~C4L)F`CdH0!KbDDj;
z>5p&D$Msz}ci-xv#ScqmwkPdQ7+ka854AoR(qPj0eCMz4JXz`1W$%ZduepF<`Ekhj
z!N-4|un%4R(jR;JEc&%%r3z;jKf5UZE@tS@?e2X4VY?>NeyfpDJg!ZF&jxpCQ+-U8
ztrNEPD3bQV2<eq?dKcZ;q-()b((Rv5c}CRTc|1>%@89_1QopwzsWs%-$oclUlQ)bl
z;44@7f_m8d`yy=Mrpll1yWEWEfBo$tpU!{G+pWQf9q&FLC%u2J<GMGiZ)wx?ljPqT
z{#In}*3GZiDL4Jqm--Z2HK@_c%Y}~@*HHfIO5R%8?aHvjzn?kPi|y3n!nqTLK7O<6
z_M<C)9+M`t+q(A5lT)r`6mMD5kx?#n>+??^Y}9Xdv)v!e>T>ME`Nys`zLNe%jk|v-
zx7=wNtKJ?k?9|4V)q0Jpy8KSUm7h9JNiA^uk%QfD{*`g$k(9k#njRQ=@R(!7_@&Q0
z(V+OSg+F@kj%$1&?fq)wyRC2C;rgzO<3*_(;|nZ0({apIe9zOb4&S_f+U(vpUP^K1
z?XkC7lSg+S{3Bnp>o+?nb*JoZMSp)$yJ&%<YKc0%;`?0e-J?^5=eaX4GP^3)y8Fwp
z*Q&K`|3uGYsj0^<?pZeV>(Vbaxb*#|g(-VwC!-cz0nS`}UjB7$ig00P!-VtCmSARH
ze6_%POH04FY*_8$pI0sSLgnctM?JQ*SA}|udhZ(We2u}cx383P<-Hk;yH$UEVfT*m
zN8PgvKDTmKO5Q=IUfD54*|mSf{+e~)oP4*^skvtkOi%nReI9-J%|`o$F4!6SMz+)P
zw>s0#Cz@7!;p9^p{GM+vy*1|33kq3zNS}${m6wLBdL&QhhHtsntF>Po+3v=R>yPYy
z<MFePzCOI+pl4Pe`=`CFS@VCQ#l9O=;>kT@;@kdOS^c>3qH=S~DMy;tw=Dtdhdi>p
z$KIi*+x5#cJJ~k+RoCpewNF1^VcLPcMJ|1f4QO>C@B6<}SDdf)?l5oL`7sre-{|;x
z`@gquEPi~%?<qGiQ675mNbRP@mOfwc*bp3ZyEC3I^WtRtmrsu_Sp9!Y*X;r}^6UpQ
zTQvIk=zw9v&a~?_`}Ot(pJ{sa@fsNwT{9Lx{^Q7I33<=-Yj>c;h~|yn7{p(^vxl!Z
zw%wo?zdzZ$_BWSZD|;TgHg8-Un6O}B-sh`tXmxkdl7r>%7Fpc*>bHG{ZUL1W|Jb<H
zW4D^L?z3t{=Y;gH<Nkk_{@81$xLvOT$!mYr<8nTndteyGP?Qb*%QCd~FGlFUV;D`+
zILX>@68*$r415E_DVk#9Y}mv3I-!4j9<ShrUu7Uj1nyUbdkz5<Q<VE$M|&8|7X0^r
z<j<gZ8F(bOn&MSm&iDkU&*7*UpQd_LQFW-^0eu`G4LITxu$q7If?G;b(|{bGpa>2R
zs2T4NMBwnmC-mxFGd@{$0FT$@1nBoc@hX(!>6-Cg)$0K93Gr!|9gD9S?}u6+^fp1(
z483)H7q3kaZSXQShbjWM%LcqQcypUW@X3PHtNM~{K!C3}RUz39uO|CIA6f8%z_*s%
z0RATT5xnqX1crYzwQ#H!&UMC_1d2`|C_BbsJ>lnkDY7LwIOFKz@sh*kfsb7?UPa#<
z*WPe@Lv9TR$>}v7oueuE?1DQHc>8z<q@atm2@Ftt@SRM9q+bbMRS5i?dAKY*AYt(V
zH90}jM;<<s;PWQA+}fS+A-#vx^iLsWnnK*d6uuz1;Vgf_FE1q9REGopEO<P=WZ;wq
zo8V3L0W_hi*Omg@o)lG51+VIPu&LxpYfq}nWj~dKm8m2ct5ZpZbt-w*;sdhQrjj&G
zC1GJIo4A}Fkm?h>KDRB^=dz`^+)yPY{Z*e6z3hBIJ;ZTqdgzzSe)32wlSgq@Cyxy0
z<l(qAJ@kLe<ve+WY4S)5lh@Ye@ghxhgA_MNg6ap{Q0oMT3MVw#mZVBH&_BhYN`mNm
zuo)yQKA<iam_hWQX&^9zl%*Nut<E43))^#diw{U!n?c+(gOr6C>;z<+>~gr=AQ>&2
zE=7UMrRsq?=k<APvQJWz;lF6JO!cXv&m+Tm)#ZO}UYGYl6_d2K==kNbpH|w^21#0-
zRy?fJN?BZabh&#^D>QjjxU-qe9o(#>o!}a&>hobsb2)q|Xrz(@c8#ohys8sj+Xff3
zz?!P9v~D2zJObP<15*z7O`jJ{CfrEXL?-}!{6>iR3*~ksDBMo7IRczXpm{qq-H}|1
z%cFnlV*xEe(JeToq&{%@H)bUZc;WNHEdxQ*=7Sr)Ly+J%vN@q;X*WT~l>(e0wpHK@
zi37A}@dMZ8R8$!vNEIAG^HT;65cul;FOdoF$u4NF_VKz*SLz4U#H8Q{94@JEpEPKT
z1fI;RCWAf@hHA2Q87!Oih|LV#mIw!-pv8Z~xEP_ivSREQa`!~7{(KHsqTgtDy1XU;
z6CA<mub~YD&Zi1@d2NL;TWdB&kl@L{V^iV5$%Vja6P#+YYQWt7lJg~}t$O1Dwk{sv
zw&6CH)0P!koUN*g00JM#rnp_nwic?lH8lVIgHx?~@G6NeS%Jv;|9O>oeOv_50WyEj
zdZ-%SZz|D^;C?|_P`jX9W~c~WuOKBsdm4UjP`uCW&@nXxLRoc!K9Wl|7JFB6vg$QK
zKllVegn9;epq2&|$^(~dRZ#-+*=nM(4(Pu_4Np{^f}>C7H{8JEa-@Z*XYi$vE5gMP
zE_p};wHM=QUJ!YjqDYZt2!iK0jud}!jAwX7A_)Ru3@!qJQrLLY%`;s?zpy?tXq@zg
zDqe>`{1B5;2lr<Pem)5PHbSCPwYid=2=2Z_xCDEk3OKa|+T(+Ykphh=XhdZ~Iu=8^
z3Bd%;TSB^DE{61oA)Um6Nbd}HU+p9Zl3nPq;FDF|_<%nHc#4NGk|Y60aOHouY5pHC
zoFkX!mC^Ti!G^YB3EHHFGBk2~li)QwNl60SvtvqO0z1j&XkeS=VoDJ?rNDzq(a2cs
zgGP<E5x`X+nmgL@$_5|TKzv7+8^YcP_kbAK4G_~5V+rhjxfs|Z2X-O|_GZv{w*g<W
zb`k~9n3uEzsOD|=>brx1^VWYphhS?j$nX@K7=yYA!VG84*zwE7pdLA>lR>C=@TuB)
zHPnX<kb5mzv-p!$C-SW!uZ<RKa%ZOlt%Lz^Qf$aQuF7zBVtQc0H_PSdsW8jM^dNG2
zKpDqd7xJ{Kx|MGV51<_Y{qF=4fOmstkLJDeAUBXlfCmD9@GjM-8tQ-0*);&}2=2Zy
z+b0CzJZ4Q1{Bkixh@2wOhM})*2Fb}TS+zOUB-Mc;nkd)_Eg^@R2=&4Q^hPq=9#oIl
zEqGn`z&almlt$<(;ler!5m@nQ2IXQ{f7n>>?dRt7fdkk#GeHM^I2MuozZa1VtZ=w_
zVgp;f$Ir<@2Ryjw0d#-pgNp(D(Fs=%!^a0<0|DsrAW=lL0uLkf$mL9J(e_3R$8#Tg
zY#VC4_#>_mksdRT%o7Yy3Vx#oeh8R7pynWD7S_;zK1i8`vy^R+GJHUWv&!Svf|P!_
z2M<yj-s@&w|MATY;8KLPYmerd_CbTzCdm4%`@Gfyl5AME-3WhOC0v^~YbSFix|ro2
z;^vM2M>lVrrv;pm83`lcF2gbuO94zG0IvuNgJC>JLVXjN@CPJuGtf;dvuP7{x<o(b
z^%p%Rwu!aqWR$|5%9+4smW#nYa<Dgi(;bavUGv-d94SJ;A?M$|feoGBau<7yQpnS!
zHOTwrVvvs<<PCoxbz6uM6`r%8y{*ElCc8Z5xH}>hs!naa7kt*^k5LMEhO*+92+G9(
zA34Aq&ge#>4fNN}V`zz2osueR{$}(S!g+S~OwM7*C`CL=Svgq;<zk4B9O6U0&U&O#
za=8&jS|doKs|tQnE&NqK!fLVjWHYzQTJ$kW;m?uQ&a!`gxfuK-2Y<$}aI`>+mWrfm
zi#`Yj*=@k>cDb#`7NZpAJYj`b6_ks?JaRB+gD`LE_s8fS8kF|pPX|g4M&E@{Bi=%Q
zF-n8!aK@|!46ujIu8ILZ7Xijt((3<e(y9?l#InW`0h|Wlmb4HXe4u21{ul5D9Md5Q
zjmC0kdNqISK}Et!%C)vB5*U~JpklMo2Zqn&fS-oudJ6K#qkyV#?|32kKM*gZEyWA5
z@Wl(^gyrmxG|N43ybw2WorNGAJ^rw-miPh>RGBQ#`vnS_uxy_Zx=Oe<Va86;xCI3Z
z%00wQ82gWI!U7LyN)QxWz!-v;Xo9C1P7xWBWjKF9kV#r18HJRka9f?Y;XZCD!`)LO
zFC#jG2k{!bwq!xoOi>|oB>WyB!6*ed&02Au1?6G@j~u`acX3On7VxtPDbP672~BYO
zML-YarEBYY?gD~Q3V0T?wxavxVt|hv;0<qaYsC8i8m%()T$JjA1}J(tK*Y2h4vbc~
z^HzV{W<j|a+#?5f!#mv7D3*h;ttZ`P;uGg45E!Lc=P_%xj9)H>^~hn}a1pmro$yE=
ztn-A39R~UXe7-;+9I64ddt9+dprsIrpE{nSHz+_ckKsRJ_%~`yb@mhR%}EE|EZK*u
z({zIMTm9yfY%EBaEwU}t*fh(<G$3*sU=)9vLMc^&HD5<N%CQ0t;DnC|p`Wax;0J&&
zd4L;j1d>mqeaouk6>=3hj8ZKiFl(W5zg$cU9<~<5QWK*&HId_uiS24$_e^z`(vWvU
zt27Ch1h@YjXjBo`nRA*|@DP#{2|GzyN!(}bB>o_AM1PDla|AWjCjhv3BUdKMS<`>w
z-Kz86byBJSfskUB3i#QD6it8#Lr9TiawL44<sLYsh?(fiMm-t5Jfv19Be;bFA-H#2
z3og=O*<mAem2mCTw4G+GFnoe?4{x7Ffc#JH(|{mx0RT*vM4G1sQf4rbrhou9ZHy-2
znF+9pB1(L?tyRo$MK|{)s)E-aVAOvbbBKDSB!l{qx#YFAh+~wZo?$pksQ1goP#-ze
z8y@MV{&HZbM40A!fGfJFBKyR=?gv+V!%{&-7IGJdj8ZjVbKbViaxpcCoEjL;>JBK}
zsFg-S$tp+@bYnGh_4j1UVaO;&JI7cdDhB0ZXpbD)4cBxRW0i-WG}>2)4#9sZ!<Ws`
z_MC%|QHpe)wvvt&l#3xfa!5D1po30kfGR+y`fH?Ws0RXi9X^Q9D;WGLeWlNJPGXcs
z;<22)gl4&z5=2Z1aHEivsp30OeI6SG2Y}1GVbyD7tE|Y+DfSqpn70;nVIF?eC5HLT
zn8!oUwZNm-0@P`=S`E}(g%y9J6yPLo#j**?#Q^@W0Uir0Ms-+`GbXlafGQCn&;>lx
zXqtk}yWnEy)`M?oc}UY@W`66}TzcuDL>OsHE#PQ7#bW<DQ&EeP>W$1oyG;7s_JK^s
zduK1w{{x}MEI7_K)X0T1)JW$XCN|4GXsD6p4M%hvE%FV5Bp|K2ss4X!0OZzUjAU50
z+X!7HT$?s&XIU$p(4gEy+_bs>=%%eOfEO8_q$!REBtcOMOOv8VND_&QxPY@fU^!lp
z!)%qZyy0f<sIN?@aXj)sAXYT`0NPSDya)Zvh$?3-xEQ5yw^k*FeE3yjVsOt4cf-%z
z2IVrUPTl2Z>bw=waaVtfx}ZAk{?0_X3o24!0d9n@5-#{d>z=W~e-6sU;2$~o8_wrO
z4pfOjPx4Y74wr7P>LrsxvmLei47Oa<JdIM=6P7AK2IXR~j}+`V%y2ii)c_99VjdM@
zK!7J!ebx8(p9?shllA<tK<p8PAzkQT>Wc**a}GY@ycHgAP%eL_2$55SOg?C>dV?tk
z43VMrv`1+8pb|?^GeVQXavOS#QWYRLD+k`7TucQbrvgSzst!I?20EQKP!rYcsdEC4
zq-vR8=taq^cWZ~p3i1rCM<b(D4M@()NgyZ}Q-jE<fw?!Y%inNUt6U{$2Lr7te4x-z
z$hkM?$Yhj)onn8iczuF$F|a>uuv?2vlBUQcVIeZv#I!{qD+z+j?aI{H`a$E5IVX;r
z<?dPY(n6I~&T9OT3~T%`=SrbwIV<r;)|kFVqRoY7$=|KWe}>b`oclB{2TGptGl`5|
z;~`<YhY)mRbMDTkd4p-5evqJJu$iLMU&Gg*En#Z8s8N5YMEEtKbsu!<-#hNe{13z(
z2}^NDHk@%smdG(>$Sn83aYxSZQKJ;cU^N=O8b_j24c6>8W;{@3GSvsH1tN*C?5+{I
zO1SoDxV*DgbTKIR5cg>IKe|T?faW<C;{}SBfdC{Cj*dbI9Dy-3MPdXASca4+G2E_c
zjyDA&wP=5gtQD?yyWw)JqQ+dhA6oNOqadX}VLNM+(R;)qqZIKJYsD%J%Eb^LF~swj
z3Gpew;Q&q<u}mClB5J%CuwMdXDQy!M5L;R`$W{90u2(I9Kjbbv8KoM*Slk1c*d7pR
zjmS!o){8m>F&->R=?ONc&?$CW6yKMd9LOj|KWBfb$VyNyhW^N*Kg2WbhdMf5Nr>`n
z(437uAUCndD1|%ES*i07l#9VVa&R|1&|O2CjiP9~&XPt$idb6aK)oVJzzwuG3~JPv
z+jwM@G7@o1{dj|NF*S&s8W;twOa-`6U!y?7tzea6&EnT;wgn<Hy1O%Q63FEK$tflo
zrK*2GVb)Z^FBemV$f*LI$^19Y!ays*t@UB>$*S%f()#q8suu>j@8qtpW0YzE&0Es~
zzg$cU9<~-(3s!V}cQ{*b*ndQjL=XfqO2poyW>K4%wW{{sN{v#Z9AfWHsoJYW?HX<E
ztyWu<(we1o(1BWc^Zot)ec$Uk*E#>>I@fdZ+~aeP=l;B{ao};|v{)WV(G9J*P)LB!
zl8aX1!v1wR_(7qE47F!boO;6U@d=iq=TY>&0<MyJx2MA6gunX53roycjlUIh=Wzih
z&^aF%rE4{xW|S!xvAm|}{`igRPtn;IPu5pYbq~}hvbIBn-b=5}>`#*fGCjY$y=yrj
zZTWlul*y;kl&AINr|rYMOO8bS&3tT*z8qK1!x>GEHvu&4g;I)(Zj2C(N@an!<Tm^m
zdRz_y^;D{ciTz{42wzL{-Masz=7lTpW~SL(Fqe#o{a(De(C&0CbO}L>R}Jb^aBsrP
zJjrrMU<6tmJW(-DM|c+Lm~o4x_p4UqY#VF8GqdC^rAWT}&AKFWbRR?#cq>ug=5kvh
z!F@J}Z&{=-pZbfnUsI>H93>G}QJCk-VVTzbIf*cs%tSc~4d4`o_tNHW4^{)DT_Y^6
zArP<Jk^7h&YV_6DW0nz9>)%hVE+nMi&u6MPNazlvStqv>aou`4TbF?kDi!>Y^D+}Z
z5}x@0^u)k5FqPK=_>@=A?~*GBPO|27Bm$IVq`YRCf$D>T<y?tqrE#_QbS`SGTY;=V
zstH*c+S`zaxQQg#T98&AQ2(O*I|o^3ZdKTZRM($ehs)IZdEYEC6QM$U@*(rpxXRh9
zzi5}-XQb2CZD*GO5<!W@H_(5ZSv)@li_n^wo33|sl$e~6U~p4#3!Rco8B@m1bmQcQ
z`R$v(i9RKh(457QqM&84Ixm%!=&*lJw5(Q>Wffb%4O5|e_3Se>Kz-F%KqeLR81~>>
zRxFKkTwWAew|uu=9!g5ZD^+c}z**|w0(9pJqQi#^UQ>|GKIceOamvSLZm0_t$=w{|
z#k+M=q^q>LrArkHj}-8G+_p$F8N4FbVH9dLv5^a&!`brapNo$yTdoX7zsU?wJQu&@
zSWD0}#7cx<EBN666SH}CsPV=|A0&=aQU{7gy2O#_c8M4*9xBMfOIq$1X<*)cES;cc
z%O6achx_Z@wxH@p4|7rY51p@?)JxPX&8Zl=@nNNRc>Qokjsvd$IgtCmgWteShci>k
z6$18TY$grf+;&vykE`JbzfcsYEQs6jb81&B!8CjjRt37z+guc(LrJTe2ZxTQv46KJ
zsUN<6RVD9{?K~`Q)nZ7`sJ0t#gE+_<>e=8te&D*~@I8A$|8{@o?uRny7fB;@W0!lQ
zh8Ch5{evqZyi)jf+f^7{&-D+0ZA#pp>r%FvPJIf+1Ck*DXBppsWSEu!E8|$;{qn*c
zr|}H7Uj~3J{k8*Xa3x1aRt>rV!FKmSe*GhdTx{uJL#8^>F>EW}-5!J2e!9+b3e@@u
zy6W5WY5bh0@zN?RR9TbYlB_!!pS!W(HDDu`RFZ&%aEDN$NRr#$T_p!~!@(~!k`l^A
zsH9{bl(BNPyImLv8w<uzt=1}W5kAw9WxuL6!T=02Mi@^4W}@NlFIO4$!^+ZMT(iXp
z{>74fIo6E43Mh-#<M&slaF()i=0mv|P+SJk`0LeNRa#4qC4;y^(+fhA98krs7G~m6
zrWQ%%nud<(xay5wHecsliC0rkz#>7U+)`^(Zo(h-$klv>=RD#RrmfbWpaUejw5fl@
z1Mc0GS-z}RZMNRC3@aVRhaXHJozIqwEL;744nQVal3w$?|9`>lO=ms<A>uJ68^n3m
z(9OpJ`p>INJ;<6}G0*HwZ+)6uv^n95(GZwqPJR(fgvf}UEcNc?red&5SHqAqlas3#
zpZIHZ`Z`s6M-=62@amZi5~ngN;u;H3QHK48W50sD-93dQMu={n0$WQOpU|E5#l_h5
z$x6}FEIsIs7)3TN7hl}zG6UIh!epSwEi1!vpT2$igKSagv8TizOc1xMa2CH(r`2SE
z__mWn2HS_U0OPbYmGU2^$02_ee>(^o_;zy|CkKN&<Nvcz-)UICFK))h0s@9Ot3>)2
ze;-Iw25nKSL}0>nnx&3o$S7ecEmF3k$P3lzWGN){$|xTnXE`W#0%xRiq++C#FA16s
zFvrK$SlkPI4eCn*LE6{Lm4!wg-c;8->&O%RaNlU^`EMpc8eluqN!Z9^os$E|B$BY}
z+m;!0s9xNzjdLiydAs)kaB0FuRB2|nBtLi7>j?W0Rs)+E+Z#j^&UU?u>R!Mdz`S*7
zhoSx1hSoh!&fHwks~c_8MNEb8spl-^^|9XfYQCjbyVU=O7ava$_C5)Vzl#0c)h~$9
zoJ$_WL4*W^$<Rf)AadEesiUFTZ{MU>4zUjoyZ>lf?uJ3kg@=KjmTiS}le>ZW0j@;9
zc~_hIWBkvqv>)ujF)pSg`{8EpE$diny_0`LJ7Qy0`~49ET`MP3OxSjrZ&#vq1+;9D
z+G<yUtYKb-nTGILRj|~xz&{sX(%jxlG}vVs=so5k*VQ-mcWyJaZ<+HGNO>H(<5a>L
zjr+CEm1KRl`GMO+yz1m*Bw7Q?G6k6~k(ADpPU+vHVo|`0buQp48Gia*?}&Xs+_9fo
zLkf-obC{2#NzIK+dWX9yrzU2@v`1Dy@1q;VcbSDAW)Ow*OIA$x)6pE#pyRhM3%nT(
z%h(f@*RzDqf^)Oa{e(0%iN<S=LuSr7dVsJMttwXvu-6u4gI^Jk5Rg(J75fR+>yLS2
zTtYrbYC2@T(Bc30JcBeHW+r9U@OJB0HxKe5g3xpo?b&1xkMN;^io%S(n`!2?lRB)M
zET@93DY?98UEGUtn5Pf~1A1CU-(Re||F7D8y`Y+>c5@Wml+U={dE4EwUw`w5CF~RG
z9`BPl;JAnHk2KVlVZZe&LxOM>q=5Dgi{H!J?Z|)jDwVWz+;W(!ds}zL;?I<W3l!#+
z{R-O|3>+jC$zyI_{d9Umo9=b@jI)kHb7S|FQ57QxM{)6-EN&}J*xP$tw?zIEs}aGi
z{A;Lc*vr!2-nt#ynz!-ucYv@jmnRh2t-tRYY66-b!X%a7t3(TUg6lI-VapU_0-q>2
zuHhm&Ixk4`{<V<q#Hlki8;-lzdP(?ad@_J)4)K`UwKLw9?eSXnxz=$QP)6)p7BXw{
z^}3$5cd=FVl2i$dBavCyB|i(5@KbzW6c<1I%J{w<BzF0S;n_R&)AFFt7beN;jM+-i
z*kpim`Za4=$o*1%<k7wAGwrhh*OwW4rq=>EGH3+JGnG^|4(L=xWIs>zeGkjVd)qI5
z=-St4%iPmSnR<Pbr>ZQQ`!F~{gE+Nfj*O}UwLzCZJtnL%e_gXE6d|j*1{PSrn;Mv3
zZU=jO_iXz~Gq}SRrH;Hj@u}eo$-DdQrHlvArLZq#==(<Tdt@-nl;=4-D?G%F-HMZt
z<Hr6JRKkl&#|n^{irbmHSVh;f47_8j1u@rpL=Oypl8zojDEo}5^u|{(jlqu%51Wh!
z?3(4@2{<K=1-jgh9oOXCv^1>3#qB`)^-wCitolYM#ypC3lCdmdE?foP*il>2Q4LUB
zskrn@0I3S-4_tYKY2Qox@I$Q5QHlE6kn-%#_5eI9MeX8uyT(Km7)3t&N0KuKtUaeV
zyOE1chNsbjxcvLG9mkZJ=^o8l1RnO-EhJH=l|O+*99FT1kgc4IXCfw+UZ6^~=r4a3
zs?)yGeUum7nv<Gms&6C;&x@{rXaU>ej9V^ja|qs6@YuRabTSDf+KE3-eFYn5%dEd9
zl~&`KjCo3ZTX_zfe82e-kV)kuEp=_^<`d^(Q4LXe3k?;gX1`|s2m{zXdf2bM!bp--
zBa4@z$j!pgd|l6Rtk*cB#-P9b#MyS%a>4Zc&jpC-S7cZ8agFe{;7@@XQb0JzG?$C`
zrl6?7&}?2PJCi_3wnjP=NC$}(p$FeeLQ%~2LwJ-S&f_>CkJ`|EsRR3<fgL;T*Ry_}
z<lvPFH9(r#qt<C5e-p_qcfN>SDQc1wMyC_8Wdp4Jnsb0I6O8y58uX+4jSxEiQJb(3
zIyV$KA0&H2!z79q39M2=b>h!eKS&SYWIPgHeTpn_#|814{z6S^MomS^+$?9quSs0P
zc&12D-LSevsyL9@r4|6z)DhMhG6hX#D(Yp)DP9}OA(@#_oPg_}I(_1#l_nq|lKjl3
z%*F+zxAm7cwT%op?72daVpEc2KC;EMA#}C!c1|HIiMeVi0OYA3N|V;;Nhtmu)|5RX
z3X48UUf~G2%i#Zwhp)(YfT|Duko^8~2<@TKUrX|Ysp!NU$Y={((+SKIn5+RujOr=2
zvsIsdicW}gt{-a((sv9IDgK#i3OiJOKSDF)<q!gnXKwzR@h0VvBcriHhzue1iE^0N
zUh|&J`>S9K08=;|rdZ$Uh#c#u_~CE5_5FNJ_TWC*KyjelyzT;7_KU3>wlaD~6K4-G
zb?Rne@1ytR{ZGT`klS-9J7AF;5}ez_gj1;TI(JDNg?rYzMY3DysU{jv4xU98tLYSj
zWs^BAEfI+$xk)3)o~lB6tz|z6=0D}tB8JhT?(&X+)FNC~&}yS%pKaNM?NPWf+a93j
z?~#Iktrg&p$zmhHn#y~baYLk~7@{clPW@s~r43%%ZhhzP<s2+9Dwxe)hM2<Uz_6Vr
zKUz%Uo1QrmrM|wXkU0)E{<9jsYVyawR(z}y?v%;TrW#^<RACe*@q^DyO<Z$7y8l)v
zSGodDI-V<v%A0m>qW^JXw<*$PTe=Lm85&!et)Rwia_iUr*2l84%zC*o4l6kg5_-Qz
zK$_^>hBy*>?v5PNOe$#%aS$?Ct|WOVB~_h?bGooXzYftV&T=B)BEAiO^i`I>?)u<&
z_)dtazW$%3pD|@HDU-QS<*UCsd=IUiFEs$!PnK_W5pw)YaUpyzMPqVWZYZm$UCVeU
zOI`C2Oy;47N*025z?q~><-?J0?CTW)c)(8x?q;$b7ytNIGvg(yg+hgW_4`95&ViE?
z2%@Ne$e~NRuE^7j64I;U&q67x(&r**o_AFsCACP%a&R)=nq{$eY3(2K=~ud*-2x5*
z+)3+n>kBWUQcb2H6j))0|Kc)4=pglphh8nJ&jzK&-nfmOhUJ!Jf_Y65vyO^VK^kV?
zXnKd=K_=RCyOw>~?m@cRxr%`Z?XO-W2fQ$zFT3$`$TbT-exWxrWi}|MYkcT3sGKT^
z4ILW4shP^9V;Qy51k!PlNJas!>Dfc+L0qwMZ%y=l(S81v^ep<u!j^fj)O@=U^HhM~
zSY$9~fH4%v=OsyVMm#0*AZzGX`6riaGH+Sh8HX5~LF`DdOu?tDJRLX6FT$rS6B=03
zIza1R7jEcr_DC28F1}$VeuYypc|(+Q_M5p}#QLD&TRH*tsV7+iIshF69`i*pZenAs
z%UG}eY0;k9yZ}RyB)R(a4+?_IJoQ$8<sS3ZA)?c{yRkwVoq~Q{Bd4TS%`gSR?b}8m
zopCnRNj3-OtXc;AXeeTfOn*R=-JM^z+|RC`OWnMzF^2(FvI|LaCQC|~e6}vN&co(w
z^Qxa8<p7PY(a7J~0SGiMWXZ96N&J?a2iG_REjh<l(ENW2D$mUdobNA$kGsU#l<}~(
zg4N$il5i){g%oqa@)-n@c3eV~vcAZNa9WwTs3BW<gpl2wLhBpcLj6rXKJdNs0^1N!
zD_#KfM9UH*bRTbx@AG)Jn5h|ffL#<!z983ocdB4<+Ex?<guc!^j#*g41SWgLjl0@E
zh<TymFg|I~O%cs$1DflTb8S)T7Ily7k6MfAkNTX_-+yHdR<BgS-YeQ?01vxXm5gir
zB0EcP|N6!=skHxS(Md7ZZe7{tB3Aftso1)17(dI(U!6R2ym#T;*%3FI`}O@#*!|Da
zlg1t`cLPZwfO_7y7d6qNn{HPWhZWYIN)RVV+&C>tjw@inL~rh|f>g}k3Q}&Asf002
zrpSHJI=#o$=YrYf)!Ty}``#yQTk<!K)VJ9yo@QKX8$_h?h9;Y9BvWvwz`9bCq<FFv
zhfJg(y{Y;Uc638es>l6n^hJOzebQYIJt6~re%uTQFwO{-bSJN|6opHO=_qO)y;3`r
z)XS1I<cH53MO8A#XL;yDgl}z0as=O^afduF+H=i(^RkIJ1J0%>@iF-WO66s7JsozT
zyJW(`&kKnu+&9$hZ_??_5nV*V_X4dO#gV6OXu?WDJ@K@e5!L-$ZTeZSgy$>31o}bt
zo}E;{FW=61$*g){mRQ(|R=quL)|YerQFe9VkRL)g5M5zSi%UsX(Q3qs3)d&c_OC;z
z`QbQ40|&PB3G^7)*vT*31w>JGc#llWp<Y0-L~i5VDeL?0yNBWD>k_xb+G+(<d><j^
z(?VF<Hb2r#5+RAEDuz3L7ri}L>r$4;HUu28@i|JyWg)cn_U;LMyU?G326hA`A(z9N
z)PNE)2z2vj)tOD0t4`xx7hg-FLPm&=`Re?qkP%7!uKj}T*d+8cKOGGel*VQ^2!Zhl
zWLL|>*Y&%EB>fZ_buH)>**VxeEa#>g65c;kD=K5qrFx{^Q}FS|G-prJB>El@H%XtH
zqQ+xn)C%h_>&~E*G$SLs<DqgL8}n-(rrA$cM9(0zI)avCr)de{Mh(ud#DA6N@1d~o
z3^h`F)?IlIFl$@c)k;6KnP;o8W`ODi>EHv5>)><zoE9(~Hx*Qs5rvPVAhG0>Hy_l9
zAB1VSz=~6l)6uPlfk*+hSitQuWc>@zB%M)6GDIqR=u*$6+;(iTSa`~1+mre}Sc});
zA;gn<dF`nDEt@zctp<DB+#q72C97^Cld|1KJ#bwhE60<%fZy=ToETzgU3g}mE&9<c
z9Tg5kukt4UOVn=sm#DamFa0Wt7KQjSn+9I>2Hu#$xBZ8f0#vt=K|r~#Om6&)Z-E%k
zQ*-x|sNAYZ(E^*4O@lWH6kKxm{pH%(;l63T>n#Io*GF6-amI%#Q(ynmUuzI&MJuT8
zSk%SdDw9{J&xsluxm(tHJmX)Xg$yG=fiX4_>FLFZ#Gr#8V*F3vMc@$>fpi)ejpvpb
z!Pf3RnGmxPAwSE;Mj-krnBD6Q{ra2zn_tiioB?JOi-}R?EX-Ec#7P@r2KQ>VA4C*A
z&fN2@K$a3NPpKU}m>9e3W95&hJ`@&*3l{FMmrQ3Y&JwDoX&M=gu`bcM42&&jp0IT~
zuHT1ZNsKyn{jRFSkZt#CYmHTG-Bs@`>(6P(0`r1pB3eMM0hry1V(J(F2~<Ex{D@^q
z720yt0?(yc@+Bm_Ox_q367Qm+pRI{Wc=CQIpb$o_Qs3U-^S8)W=-gI$@hieZbh#NL
zt#$`?{gb7F;g{SetXQHa-4%}#sL(fhgrn<<Iy4YKaKfb&>$AHsIQVoM9b@n-R*6Oj
zPk}%{E)BpR{C4wt#^`r7b=BX*BdM-Wg%i|v>LuEPg{WK*85L{v4h)gwX-Vg)rFHoN
z4c;Z|mw?kS;J1g^SEM(sr)}5+wKX!vZh6M6pVW<yx9GYiZakT28L&yTL2W+4=%ka@
zl{BbPJ{p9Ae;I)B8Uw=VdHh+Muo6n;lxe6d&@oTBOPZTbN%H(3*Tte?>(oZntiXpk
z9<z1n0^ki<i|X(3Sh7K4vi1=}bQ^sGgIXXFx-*dEtgjg29w9ukvtg>)Wq*f$p+v4r
zPRyYVjOj+496JHI%62zrxg_vhvZq&l?~nF(&vGobM2Gzh-^quyOa~Yz24<IOs$vCz
zBowHkF^P>w_Z4`fsb7=BwCt<)xbCTtp}&^M^k{q#M;-k^6aMZT3NO-eaoa8J+j}G_
zKLxqEKBvXv<bW&@x4;Kf^-7(Xqq0)DNok*-OC}gUg7|rSe^;fiE1aVD3du}P>iJ|q
z#Tha%UmQ!1MM90^X;f(&FoR;VAQW)dO<ao+#5(BUQDHk$qvG>4F~<^zP~-w7Rr)cG
zq1g{5M`@GQ^n?clq-sF4?h^R14e2&MiRKRRUdHqXqm~e%xAyYU<LSf5if&6!1UT#^
z=bjjMtz6W#o99Y6#K~)CMmqH`kKL*zTwdREfVJI}`EumTJtY%fu^$9=5&?8NXmG)b
zkWJ{Flvu7Po=nC4CuehG7rhS<KOgPlhaFS9UpTG&;V*C0Z3Qu1d%)0D8Dcx|*)Cw{
zFU!^Eo>0EWdK!8qA|qrxQ6Jeugbo2d&<W;ya%5VE(wIx3y441ZPL!z?frCr*3-s%$
zZ1Y&U$QQ?dwyyzcb9Rzw`2)hyQ;NLCC^0$|PsKAPsU%$oaidTjx{4z3Ns(_4qv89(
z4@4e86?&MQKSzlX>j2fPrWkyt_qo(9d+CE%KCbBa8#jJl|9tY=HUHmC*iyswlAX~3
zSf}ERFR)o(#x}Zk;aOv9(hR;il$6KSTK~xdDCzAxLmGeZdF8V|YG?qtKbF4N*e-0!
zKA<1=!MOgddNf}nT_<4=DZc$mCPR#@@Wu<&fDjK!#`NRZ6;FkCb0v4_FK$!=1CYm{
zC7l7nV-Z-Fsti3pz2y6dcvp-3TuAg5x22IqQcY;`poIxHsxaQ(r0WE-+2KdGtOyAQ
zroV186u)44Xvs(YV|oVQF3t0NdD7Ub|Jc8?;a%QN{J=!3BFl*#%SPGKuPUBT&ORE}
z1wep8kdI>+151+;n?z{Y2K=qr-0T;VyX++Ln-qhx<?RLr@v?tb-KnbE_W5X_C0il1
z;&M^P=rR(L4DRH9O1fx9GCFs0E1^tWS@K*Jf}4@gwPbL;w=5B8|I!opn2d@$D_<v$
zCr|}E%6TO^qXmL3e-hNkH7l|dQqjNr=GJCo^(38b49x;N`7XgYDlrpQFP`Qv+8-+B
zbC7U`3patlR@FD=SUjL#;Wc&C$oJTSk4j}VjnxL`^G=0`G{JJQIRv9zj*Rqt(##Y_
zendg#ym;(600$z~?CqWynYO)*zPuTx>q<hN9uOIl1%^*0qE!2H=nN~ti;AmNI`~}f
zBsT%Nh)l;QNL17WHk!5q_lu5<KbgFk8_DmQUhYA+KsgRmzYC7$GqT_o&s}RAjOCoF
zbR0z%$@}yeJ@)_o&2xPG*$bl+m>Q>tRrD@>h%qe35-8kuWn&NYOs&+BvWyMk_-(>L
z!Ao*e$-l4qX!dL1jEL3w?jz>q53<kRRUUWH_;nXkDXF!N?drEn9xtz5pZf6l-s7~;
zzzhXe@|uMsiCg#?<=AIC2J^S4>=;VAc?L-DCM|o~b%YBA2IoJ8(ENa#2^elOgcNPV
z@cEXQLn+`Xs2D6fVp6Q1QxPX06ON$&DnAu4>o;pDYF-R<obHD)hE%tG9zEiH%5I%J
zy8#tWK7OWScQ2W*JlIF>pU%zi5L0XAH{8`M%4HevzgQME#y>x<ps^D~p-RphUs^_7
zr>jql8s#>35-2VRWgH@jI;T}c3$<d(bzioRPmKe%C{dMp7^E=DG|S>a+soUJ7Hy|0
zvqIOTzlzy}OaGnG3g?6hz)LJbix?qs6%1OV-07w`1hN6oa^18G!Y#Zz5KoTK)HVxm
zd+FZQvGnWIg+0<)Y1((X{TY?H*0q-$!yIqxoaG%Z?bE|V%~R&oRi_NUd-=0XKmIn`
zP)P%L#BW*}$G+qjIDDkC1-T_i$MsHbD1c3GaG6*B>QgT&wO(^>+*>ba;J5#an^VP%
z=(;Bn(GSX|@*aN9>YKe}cgdT$zZA~hQ~oF+E;gI(X;%^3(<Qp;qCK}zQ8oIt%+ed)
z71pA7__8#|i?iFUdiRF{vi53r7~@3d(<TeA0bcnjJUt_HYie|BT@6XTUXqg(;EuS+
z{w-ZCnq2uUgO6)o|A5PaLSL~mI|CVcqfGPJ6IC5mD}Tho?$HuC)Xm2;HbC#6ukd42
z9h8zOXeqodLFjYZ4B{d^x{1B4@9{WTt*l3E&cnHNZBYkj%()WP_rkxq+`6^T&@EFN
z_%cslXA%MSqqKWD{r(f9?Lu7<IkC9>jHKaj8psZKZ2hFXsoqH#CK{mQmzu>hukV~*
zMGjZ0{PgzFM4&FbdP$em31TlO{4JdbzHh!_qn)IlonbUnBo@Nuv8zBX7dBfmYD_yD
z>r%`RVp+ytvgMhTtzj6wh)j1z&13_Px5%ARx1maf%oW!4cU{_V(Qr#AES}xIUhtw<
zo6+VzX{l>tGg%A~pFE7l`jc0KTF_dVSRoxm8YTEWMNvLZi!(G`i^pz|>fs_T!6Rmc
zcCVYpD%75<)P9yA^$2gIefv+3`3Y{?cEnz4{=S6+8!>Hf$?TzEXk{E<W&@xEWB9?~
zk`ywNb&w_~YeOQ=RmYe&3U5rubqFdWf9`mPoa;_SA@c2rCiKOj_#5*^j7IaXf}P>6
zJ#uBwr_Z#Ba9|)~0wjIB#az5YhOZ!*Ei+ldQu}FUk>Lkm(I(<N-_~V+aozA5vohtR
z8!3RMJp_*Xrargad>Qsr9@w4Gl)Vtp6FwrwD}(lB+;#&zaSpKPVNYWxm;ABNq~TB#
zN6+X?3wSe_o)~0%(mye();?8XZ**mPw(lIm_eWRR%3iPU7S?p$+gm4KzL88g&)Hq6
zjTrrq@MWUPVDOk-2MnFmm@!J!AsZ|mT?vUd;WtBy*>uyzGa3Pft4iY}&l`X73ylLe
zRIY9`mK1!x<}U9l+G9-f;QoJ|NSa*x8v#xK9-#2U5=BUh{DCr-Oh3*{bW|X^UqJY7
zqYc%2*2^bWHJI=*53igrG?#lNS7mxOLQ8IFT=!&7$`3rQYeAgiJX5HT;NB4x{lT3-
z2_mfh8)XQOiU!u*<qKUPDw_zgDLBtNjYsX61V#D5uo5;(6z#5uB`=c4uEPJmRjoxy
zKC_yO{3(6B4tHFFIOmQ1ROQ^B>bCP%PE|jIj%v7(8Kf{9l^wM;G<dWjP#8_hiIjTk
zceoI4Y9#)GF`8>qpyqaYh##HCjBx#WqaQZfX>bOB0?mfPG&(ZQjmG(H%x)eIiQHV-
zbk79&=s}r>rxUtzTF~OyB;kE^Oh4<c)EgJDW_igrE!s_x#GM{@(?p7XMwhK%YJRAE
z!e1&+X?pmVRP0;*@w=Y9-E$LnJ$R*Vwc=x84NwgPmuK=FZmTSqRu(ilZtP+1xe7f-
zp$Ik&?8dtd#^tyT9+|Ib5lc2wIp}HbShqp9>O!eIPuK-u@4XOIH#JF;X9X5ELrlH*
z=1S&Q`*@1(t3<?Q7N|mH5~=h3j9ei^W_r3s!*}MB^G3mNzCj&9N6DdAw$04V=35qV
z=E2ROCEe>b&t_daJANym*Dxb|^>|apsILg%*@FVDnUknHqx20u-!z8l-e(Tgy-WSE
zbDg5etoCaRFNa&I8J)aI67BtOsJdLAXYxCGGGBO>QPC%*lf|5C!;MFV<ITnS*!y|n
zrQJA%XzL_7a$++>VajDjNjD^Y+tLcs-VZL~@+ZTg+r7n$Dm`?ZT+^K)9)#G}9av`|
zS#H0Y;&!eQzpB!L)6d9S)`~Zt59NMmqt!H8M%9vhrON_*+Z^*#)TA@Zq{aLyu)nGx
z7MNd<TW#Q}B<>W`Y+=1r5CjiXzMM={lF?3gUHe$W<V^htc6eist`!#P#ORdsB|V64
zZN1iCC~o{eLlT+luE91Y5Tyebm;|bDNy3_y&Y}sGy3?qPfgv%b<9-Ea4fc-ol)0G)
z>RaT?ovmImLdvghK+(#AY;upIJnmTaA)om86o%8S3HDoUmOyqYjL0B{<pRZg@@j)5
z7#9$dRF6|5)m<X_4v0tfI{#gs6$FF$V0s#!XH4_8uDipftka<z7xd39f1oZK({q2O
z!cF<<%qr5P!kRBc@9!CLm38#3{Qa&Rq-@i_D!2!SgvM}Sp6hg+NwHB|XrxXu6=A*~
zsw;Cqvzi^u9;FaHAH}burv)zOIlnX$u{5~<EOB7-En2vHr&dGO_S#s3*-v8jq1*%~
zoIzURY?;65Nyj8lEHoGcgV1<VZq8p_!vU>P37^d+!vdsXUFtEbHka=+|4Zrjg}J)-
zZLLbG^Yr-NYF_mFS&*W(>a5}h$p=9wjtqq$Nvh#E-gQqB7OmVmWG;TI7&@39UdeUG
z`ZSbeMTyfIPucNR%@!uz*}3>^<r^nOH?~frlg_A%LClzFI)OZppJKJ2qR=K}RBc!Y
zj~!B`6Wu%<Pfck}z3wLqy>sW74Jz0bgopDAs^mlMcqQ;d+>9&t5oCULjvjDnbd3l5
z92>orwWwrc9n37BucEQ8?UMwhKQ^_oj`op=hESsspM`?DIIV}~lm8M1Pl&{F{V26E
z217TTNfs2qo3EFs3hW&~_e}6m9FO%rq=0oF!g_Nb!e%9^P2VrfWVjD;tT4=EtfTk2
z94niPT~*m*`a45)gMvL#KAL_pku0*i9~&FbO*PE=JvPwI(8%(s<cWhJa>K=NU0&8f
zddkw#P~5VReqwGl-}bWjq(XQ7Pv4HNRQCEJwhXYL8hFZEueGnmFZt>!ekzOtfik#m
zqdQ=8N3Q{1@?ukp^PA#Ke9r9ww?)wVeDuxikFNVGj*2;j5Y8?hGbe*zJ>OnX&*@8b
zevz>%n;^Ez;wkuPq`Hm1(RPGsGO1EErAdVVLDY9LKRB+Um&>g(c}Jfnven{c)I6py
zI3)ooO_l_lo>i+>ZvUWDhxVSJ<BLlPh0^Q(bHXP`AK;6!*Le<BOdKpmQl$Bv1efoo
z8HeHrt<OK=A+s9PG4CI8J*S}QZ<oi#XJ<+|gxlJ6^?x;FXAOuR9I^MSFaP)9giB-H
zk;IGGm*Up76})Of-}IlSvo+_|*NW5U0Wht__+;9-RQ+FclLIyW2qZ*hptX5u4Vu`I
zD{U?F-cEw$cf}NCH<y}*?4H0V%ibkXgTs<*U3D^z0CWk3aJ*jYxC2)E7fNwNw~moS
zh(zXFW2h(B=7OPK+hW&jols@pvy#!67y7}NLGavv8A5UG30d=p|5UsAW90#>dC&f~
zRLmqqV7`GHnyf*W%zX&1Zx-f-NpWU5CUnydd3hBNw)^EVj_17skp*;+GH%AgtxSWJ
z`Nl0yFUMZkFnmt*&BLI*1viMQk=HN2Jy(lLa?mt%<0%<&wZq6~@xaPJbx3%oKktR0
zBmACf^Qme1eaZGR@2KuYfLoXTXgmKca_xh=ux1A_ca|8T<^nSumQU{IVIQ+#1jRAv
zSh;!V(~)SxAfvjvr?*~c#VzF5sXf;Q38+j#5u<_EmzCtR&=ss-7g>(zXg!9{?KN*q
zCxJ}m?$>DU?CNxFw}J9<N#uq$xM2nlO&m{~3oI%^b7;nBe7>IoUdLF<%LSJxFCj8-
zkKN>##D(no3tdh9S2y6;DH@kj6>mlO&(cQap|f_3Rh8mFifKZ^w@h`rW?snqF^M0n
zlErJc+=cIKIAB3W?#zWQ=q#);sgE5}j&8-1WlJ<*l?<%@<R%nL(q})_<rl#8BWUc<
zkI88*)0uby|NrrS+_(Pw$^Ui{xYjaBD@#&lhEG<WzVZc~ag%7nU)A%G;R;WLY9l2J
zZp%(${b-uYw^=7ru=&YTKG!2w<8P*KD^y++eo50`@B~yws&rOY1{iej>&IHNuVk$I
z4au^B*+A|b_t&27HF<f6Xn>z=h>7jV>IP5TjHQp@J7DO^yUlIU(-}JTN8$29c`=}y
zo>ZAQ{5P@SlPPQD3v9B-A4eU$>r=1jaF-R4q`NfDjNV@sQwjeKxmC<RKkqXN=C^81
zLX{|?T3b6CP_x@Dz{wllWm;DaS1rzElqD6{JSHwygH!DV;rE>1v%r9v&cNQ!pZfy)
zW^~q0c6yahE_<KP_B=m`P!+d;fMQh39UTzh*!VJI3k)IyJE*NelSWo<1VS^?{JRt_
z&V55MM-q*TK|}v(Z&g3$@_$=rAkH_C#`6f!ZqlJUSkAo!V%#I^-+sP%tk84$H%Gbt
zZ%^OslI`hP$DgL56H@&|A$TnMFyn=HqtWqjhZOj1jp|pP(80#Q)n#o&%xDos<b!(`
z2kuvPf_!(C1WcO$Rt3<pA~YW==l#C*Xg(vQkpSCWeqbU`A}0(c9qL~n9-}4Ti7b_J
zax5_HJ||lmkWeA6f&?Y0f_fiM!U)%&7B`m`OeP^oz)Lxk|15vF*972~ssiwfS(5_4
zt>Nn#IzAkpKE(|>wT%QUK^ob0^XecY;PY-Dd6Q7e?~e^@pC>jUg!jTT962wa=ULxk
z<ogqDSzP4_n(dS?4q=0JNCzzuG^ls?S%J4G=FL9<MH&~G!m>p<8w{OWUHx%C>SgPT
zi>w~?#oOle7(7S?a5LfE6tca#k-71?VXW`^0c~rIUEa``#+dX(Gsi^pP0d?PLS%IK
z3+4f3!h740NG~WY<lj@_5st_6Ox=AFvCuPR7B$FFWi$>!PEDFctQYt}(6K`sQ5I7E
zq96cch$zE=KycE_`)YFI1%J>_^Y9>{WfG19&Gp3z(53h833yQxe{T7M$;=lDMc_&L
z(I7;kx{5!B-`@)nLy8<;xPgGEgB(0C=xEK0`{YLB1yyJh5(;oI&ml*`*^LJgO8?UO
zUZj&Q<$XRSzUI9@2nrvlbVQ0Xk`cLw5-A`FBo!?f3@QPldc=<bk&R#@Z%&D9>*c46
zjtT$G8`i3*puhJM*UW1$t^7O!nl3~(?r(T7`{1HufgNE()BW4M+urEAY6z5#QdK3>
zfxjh1+cC*j0hBuI+z*0)$fAcZ79cnXO!xy~8ZUq_3DEISwcP8(S}4dj0OllCQ?@B1
z<*@Wv<-GY*Zy}}aF@&M_PNZh_m<keyCI>+r=8dH267ZNPoG?p@T0cl!ksOX6q?J?r
zQ%k>~*1O$<i{R$r+P|#L&A-L)x>(r9xV}1fsYdo{`OT>PNAj}uztdlpCol0bAd;<S
zsBqFSTdXS9$PYTtcnAXcp@ufKn>rWM7!;DK|Kj4ISbxz&F*<h|f%*eV`eohtbHUf7
zUOb=u<NDn%J=6PMtPCL!boMTVPbi-$`Eyc^3`Sx2$;s8#%7*2s!CLQwKp?zY%sdE0
z4jXbl+AA%6Lv7QRYwK8JRs9s(Vj6+<!T5Y^XqIZ4`AslO-U6B$PoVu_o&SJ^9T9jE
z>3}gBg@{rwM_Qm+Vn{F;kQ~?m9*uWS)QXBJk0$6}`pNy8ox83+hvtUkHPjt3Y~bXP
z0SPoWHA%cmG?Wb<58_<MaG!M=3n$#Uyj6wPW((%gkJ+{#gnwK=8uXBr5VhRA{tf19
z6VG2!w9*AU-e9}dm0An}iIcw!(;U8>$`hOU$&mFk4pZ*uCr)l~k&0uZj-yb|!a(8B
z{-J0GsYhM0;Fvj#Fe74sVE7QEfd<i$4+Md55=em`r!<?Sz3+bAQzyGR<cZ|-S|?7H
zHhABE^7zOHG{Dihec}WwJ2?8L8o4)Un-!xr@EOS9v5<8;sEo7g%#Y;D2a!i}^=r_g
zvprHhP<Ka?Kx7quXb4g}IW=a>A1#hNU9$jxC{5xywbabNqbvgMz6mwox$+VoU}BQm
zwr2VxTk@r;3E9^!{V6-j|3ZGo!|4H`^t5v0z_LIk6FY2qI{QbPi||*Qz;rpkF6N6^
z0A@97l<zd#SpLZwPrdoXlE=o5r;2x>)21|r<;c`#iZEvz*&0*ia_7L}^kLRHD_>M5
zVIGD-Qx2`qiOazg@DApidmw%YDp_3}8G<2C80I<}R&G{)jiWkE0n{g>%_;&k$CUGr
z&J83UR50YU2H*KQ;$uGfI%khv`A2p+&|JDVv9$B<YUDdfH+#jHSU_Ky5-8J8iY^Mz
zW#q?C9!tT+iIjt+MS+e&Fm$mz3^&L@4pF6c3+Pv)M@OzpDGmoJ&wTD)7ETw`oqC9F
zmZ=m|X1)D~TMh1mMO>K9{IpjLgGE`4{pe@vwx2_1;<%wC8kI&W5W1K#bqPN}4xJJ=
zgyg4ttN==i8H(}{{#BcEjtA7Q;zk;OBxLnspH3d_YwZ30fo6~6w4lS0DdO^!%tuqD
zBM1M64FmqxAd-QD*0>Dfx<wqyhJQw@g8ZOzEM(f8;uaV-@<Hcj9KyfQkd2#IfrC>K
z{zYT#*|?JP^JRuhKEqO4fnaXAh}5svRL()8mUQNdaljzQRZ6{Lh!R6@-}oh7kOt&n
z<godMf^=ZR$Ougrtx~3nlS8G%^pkj?$q{5*xThk@wmW@tnl8Q1%)gDtab;Gh7PBR`
zRnCqiHgq)`KFSj`tXcq%&VQZl83FuFqg0mK^sk88TLfl-Xbl2$9&+Sh5Y}(FkOl-%
zRp&%=L(}7(<{VHcYE35<<uws%lExz%tL(*o>R2K%#yj}epFOtgeO<}wjt7a6XIP>c
zkVC~K5Yti+##nN!iX0}p>&FO5a|nVBu0~W$Q0)89TR`MJLbX*+JCDHcr7V@x9=6iU
zF`oX~9}zylQik<Y2e;qutDAjYBK?`l_vae&&dGd85>-@=ETH63r0eVaa!@FVx{4eG
zPB~UKk5q&s)Ybis*w#z^Q@9p?GHT~YCOd?%QEO~2t4FnPztx1Y-LmOb7e}MXH7{5Z
z@s}b&@A*1Ull2hleivgGTLi}|#aN9(lBmYaeO2Jj!Mh7~&9Tp4IS7|B7q(YNVyVEv
zOgKR#Iyr_uERV*yW9;8x?cuE`YCfLcXd@3{MiSMsEF(oIO&nL^5M|4PgA`+Y0<AsC
zukZ%BQZQW<h>o81Gy_L}bzGMC^TozJiip=l`q({5H-2wZwt|?n1wo<GA!U$M3#Ww~
zbviH<>GGIFg)?Mx6`Yd5mC5a<iaMe|WDu0VCDK`ZMqTLBzH67id|0o1%a$%LBh}xx
zncO?c2_?{rJ4YjhQ$N6H|I$LpVrYlS)uM+b=u_jc4mb=;>=22uNB_^P=I58gNQpV%
z+LwDPhYyBc{rK6*=+=1qg?2+sOAHt6h{&{5BcXyKkHcf<JP<6TqveNk{OD*^E|uXC
zaMC0bD7w-ScX6Zt>XG8PJ=yt3nTs`trt|YgGHd_5m!{_D95MfP$wbu>e^)41ND#Cg
zu4<?Ve(P8lB~AzBPw949T<QOmrTu9Ct@V#6x1^u4!`_AsvG*}`K`*aIr(~8gcL9kO
z?(5i1WrPiPNh{BXO<es>hnV4DT-Uy}dg<@o3Y8??7M7M{+Snj=?yUy_;@9oYzxH04
z9WpqB8*`T|$`cdty$518LKIxg*Jurg7~beBHyy4SndjG+<Ei`Bcu##AJ}vD<Q2P$k
zH-6tbV{#>g)9_ts-{lVBNhZ6sLIp^k51Ff(Ls|@v>u%$87>1R#ndy!?Ml79Si#4P6
z+`UYHDK6=@9(*&6S;8R5!M!Iezz4-<G)NR|5!7>qM4>NUo}E8AS4AxSzQl~@J##-c
zTwLsfbqynH*^leQRI|K;LvdF6XMB@IQkjQkPa0bCx!tTb+k{2Z8vLKj0jW7$h9S~l
z%pTP>4t|mG$lj*|8rFENzUL|14Q<{P@makolsT>2zOQTaRToB4QTCeN9|cLuj7_SO
zrxzD0ebR2cjY~eZVXir>ihTHYpvwEW3VS$wOZQuNI_*nNr{8yNQgzBQ2jly`X*GTV
zc-Zo)#*QJTrkg&C!9VLo?0_uptB*m_r-5m`fa`njg?;Bc$Dx$pUv6}1WI{un?6(~B
zSuXoN1)RGD{%C7wn4#_Pde?VJg}qO0{}YH%JQ+FbT_xZ3(!(4J*o$s7Jb@j{EY_a`
z_Igd!g7xk?(XXm7t_KTpb`)<Ddz>z2xYbD8clgvfBXw()O}rriJiy-a{HW7Tq(j65
zKJA8bN`3VyL+cL~){NhukN)luAhrz&&<dLtAM8H*z969)kKyILq%||0`mCW%|B^&V
zsB_j-MjZ67wlTQF)G}@1imN{ExygQn?D4t63F+0iSJH`kGE(~rXgs#BzZA3jo@^BZ
z%Iq)~y0;#09JtWp9}n~}zpPv4uh;QJUQyTE7TQKYE}Zr5FwB1d?;d-)TvgmH)pDCF
zxJxZSkt@Bg;G3abbW?4(3`skl;eP17gL|jFE2gXWuvjEL5&G1UU8Db+zxj)Nq58sG
z;3}P!+U3^D*YroIfr03{Ir}FUA__-$n34p2FP^YA&M^^=-+KY%qxWVV6TeGzv^=o(
z*RZf$wrF}J&%WdG^|ZTC6het_5rGYSzWPzi{Bx<Zy=9cXdH6bHGkg2x&)+SjXZuR#
z3{GT1-*3zH{rRA>ZY6ci&)j3GkDUKof~vpw&K21ocTf*}sq+zd(P7_1XL3}{GEk?o
zVNOQIMlu<I*=^qdzBBwI#Jr}v`Z)L8=`U4t)z<|+hdXD%lz6TmP5m&ktLFUQR^7nH
z1<l4q>_TTEwpZ<aw72W|n+>sY_?6@6q*b@FTHc>RSct4MtGAx4bGwqKQ)`j?>CwU!
zP1%+5ymph5pf>)G^|uZ07TF*^i>r=uCb&J{w?mG>t3H|oV6rmp+IJl?g>QuOz2_Cj
zBJK9j-B1@J^A{obOZ$V`tb=d0Sp?-j?+Ema=dkDZB;uYA<f2S{Myhc=@&z;Qt{3l4
z2P19nL^iGxd>#^N^&c<{Fo2XdPej)3*?1LOiQ?GRC-)#;b#X}<$UX0)`UL>39pF5~
zhPB)hYSsc^&b)?`sHrNr(kmMT9wEEUe-PyA2=q$71wHl`S%&YNF|J9!GzoZ~E*I{d
z%TvJaSnYLIN^vqaGu;tDz@O9iUnVlcTJH_mw&?_Ycsokw9n*3u)P-GF=2o;4IuUzL
zkbhpfe#ZDt`mxzBIU(;{(t(Nnrl@2{!@b?3%GExg>}t0xa`eF4Om$pZ!3++uX1<=g
zWk2|5>|%<zfol_|=MsDsNt_k`L^i#Hn^AWp_aO`&E?xRb-yf)Uf4|h*YYH4py&yOZ
zUMNTZfCqebdg+#+t7ELUFBT26{%o^fxyn%_Iho(Qult4b-#geb|JJOx?!m&{n%e~4
z#9aX3I3YY;Xr?=|j!d~KG7zT1x217S{!;$1uYz6d%=>Hn!@lxr*7y=yv~_YKt1>Rl
zYE>o8AW*bE%SFjR(lAv(u>4`~0)SgNHvap_aC0j;lYpQ0A0F_kbv!pbZzJvsj!+x6
znNb!&`K@S8$Vbn~wA=hk2L$0IZT3wp3P4;)46Ipo6I`-i6T-pXn>~aliB0zE=5qvd
z<;8D+zy@3OcVmb*h3GZ5N!|GMz2k!5=SkyDQ{@+yKM29Hho3)p^OnbPR{Ry|vOn&-
zde}p5h5)0YNh8Ot+K788lil~Ryh-<}&6-E(>*nXb*ROq?b#`-Z%ie8GzV6@q8tCbf
z2)y4OgrsK}>8?oRVBvH6(fsy%18Yzh(Z-XLBt~%pim3q4iNs={Ncb>~|0UJ(AKCuM
zS7%hM9J2)Sm5g1#MWJFQYU=zdi=|nv)su39t<dG4^@YThg#w9=PXAWh^8ov;%cFuT
z;+f2>yejT+)-i^Fq>SO!Qnd(W!0}I#ne8`I{ZE~T5r3zRglLCH^M|!*gFukei>H0B
zn}79wE(vbt{jjpx8(zOKaH_x*+2u<*6!{^rA+K$$5t)LdI(nQC8kT(-esPZjTfvcM
z7F*?nt)HM${>Ka=hGEfuS&@T|b9tqpx&6L7)cojSG6+5K*eM8o1O`EYxFJ;+o2b&&
z^q=g8Rk^fIrmOxna^Zowe}7KUjvSsUcg^nC`$b&+1Gd+90?4z_hF#`awQ&_kXm0k5
zLA3#Kaa0tT3-^tg7z3n_IFuWSv!$zGjWJRMO4L6wy1HpWNZV;rOG1<2xX#4ew_dmj
zetK3F!;8{Q@RNgoRigmuV{gzO7aCu;-W2E~WS-MsJ&`obp;n$y(Dc%e)%oAgk^1vK
zyh0GtPIkF#D`$wt{na%yF{BQC@mpM$&2v<hwqFn3_QjrBycgvBdP4PYS=3}hlce32
zsB?17hBTaXs~al6O#g|!Db$1GdunyK^;x=YL34A9bG=-sU^j3ODh>gGNTq*%@-dEB
ztxLY~SB7oilNI5Nk@w3*<+kA6pFdiy&&Qvoeyz@YzI!47Pw{}ikXozd{@5CW%@_J;
zCcO9_x7g`0ME~!XhU&@2{2hXi%hH3y#2d?2E15<Y=OwDG=NUCew&#4yQ}@1e=wGOg
zl3kt`{n<Gk`EIWPnB6hdvo^N<e(&LE#u>2bioH?+p~*?rRaEG%FvxG8uLz}bAnY&?
z<~#6`fxu2^+|nat^sBXS{}2|ZBN=v__uH(aN&vSYG1PE;3`tqvO_j`V?C?waOXZ|8
z$i+PZ;TD_s;aH3+5`v-`<IEJ7LMBH?BcW(IxCZHf9{`0AXf|nLiQyb@MBqzTj_%_U
zP`G%PPd<~UF)65a5sa<*p&T;Gkw#EK`mZP8`&t41qbx!?fiOJ4tZU##A7qq~TG2PC
zGv>i<g^iuu8PfEP2w_sc(@~1}TeqS<);lk<y%5P@{Zzi>w@|Una<gmaXRUo7o9DIN
zKhlbU&VkSO)bO@FY%4io+_t~=6@(Pi6=y`4gI{K@Sq{9~IZkciUiM}V`?V~4zPs`u
zAWQ7ErN_xy|DZl?>p($jQGjuAQffAMZs4_IX^NzOP+;TNurt)7heBVa#w1riA;u+N
zv0>WoPL8L&4P5>)aHle&v;4@D@Aut2+bZtq7eLz8)GQ<a!$PTRLIt4(-BEAS^-Nq`
z-d}zD`7G=)w|qX2H8+`{Zk^Cz4&+N(0>q`-wVtFu;=Imt;=*g<?o(-gS)IVCFMb?v
zA@Kb<(=*()w;yO1N7xky%!KdcaIvrRvX5jv<N`G!hbA>1+rLV)oK%TVO7fo4k|c!D
z^HKr^Um{cn<gHrOZ+7~M*i-qpmlmyBd2rm1TFQ2dc|LJd_iUPvs>)@@@UegpNMPa?
z`RSU|rc?PcJ364wZc*bwXqk29&woQ&!6T;c&T?%y-zo;&v|&`RXhx{MeZ#B57-k<}
zSA+5zF3z5*wB1r_qUuha@z#<vt=A*9kg>}Fj;Ff!^|byu+&ScAmp8D!X*rp8^(vro
zR3Iqmj;k!R_cwV3#cn~|8_15{b+Q!%`~6Qpf4O68q0ilTh0>gTDqRn&KKYxRWscI9
zpawu*5n<l-{pyjg_1|2663-ck`qAo3W+vcl?W2^2{mZFdXnw=oyl_;nf@XB0`=F`l
z5O}ow-gd!3Et8UXKF<D+<%@=D=4;Z-q06;SjOcZtaM+VyEE}a;LnHb%>V5w2;+c3E
z@$IY^YNloUwo+&B1CBp_a?p+jeJ)P9^$1cZQ!JHhd2?WRQqVnyz6$j#=S%nG@UXT4
z@>r<Yd*YaPtNKe0L-oh&d~*<F6;DCqmq-A-enTTPS#w7F2lie1T5fMS>%+UBLk@gh
zHjhX^{BdekLipNc8RO!Wxq5FmJ@c{q#*(@v7gF?i0f(7CXr_}N&7jNo?|Se{b)Djg
z!?4N8Rt>gaAM`!PiIbVqv+J!kCR>Z_qR6M(LBo*oj-eBo#(elrWMI5VgXZ>?J{+*I
zmLZ$K`P;fh+*x25H#0rKzrkz9?bI+|ck*2ko>|f^NYRgR-P%xm(fsuZr?t$fn7}@n
z;@-DahTw;#<Wp|RRI4nRawt=lgY>k2)-pCd9^enOGtZc{l|ogu1y%eglo~I}%_RvA
zrCzF{x-86+Kh$0R(B=(V2_y7DEfs+GCt8ft)OfX|?0NP=1vO02NHT=9>g)87<7ry2
zNn?-Adt~*e2cN7OF5<S!&k91+sYR50Le-|VUwXc)&K7whnqy0|_lVN4rrmUg$~@L4
zfQwRC@!|gBZ2?I))mCm^)d%-{?Nw4SJ9x!#y3AX5z5c~N$8<9CxJH*EejkVe`)Owd
zZWCV%UHjW%y*6+t!?j7IVjSlCsAV+QwcFLKv!(2D0=Vuf9$(pqylu|pk@x>q57RNQ
zW}Lh$431kW&cCU*H1<16J^Px=RM$=CPuQQ{tZug41x~l~FPN?Bg_(qdg^gOu3;qW{
zK)=5^>nP^Vg?s1JEci2Y7{iv{!UYqMhs=MxW@|SpT&CWAa=PA-(uawaS#F+>)-Uwp
zk}yZIEgK`s<|5M}>O<6lp||9sba?=$Vp%olO*WFJG6yorfcfbO42Np2%AU4mp|+ed
zhPIY3x7g)e<>FOWxV)D)Gh~$R+5_*R;IzuJJoF>#d9p{Y=Kz7pTbx7RJZ!wP*j#_9
zY1D$<v@Mh69@M;LJZW6xY$$nPi))NR@R^fUOney0xkp=s-=7?~UKzf?oK|=0Nb@Ey
z+R5}?hwkO;-&9n0#5xFeyk+=sOHVL==#*8AP}C*Q+YN6@ya!{7qimlpbnU{2np`@4
zaZV~UGZ-6TYl}$A>pURoy(gq3sB3?Ozo0mjRb?!Z?STX9+dccvp3z~7WraTFEK*fW
zUP|Fl<Dxus|JIIl6XWHSOJrv@_w717gQuXI8Jn;SADH5tOP!?Sl#DVMj(<%P(YxKU
z-wYOG5J$bI=Cu{~yPjL1<qBJ!yQ9ndn^a$v`Le1^x(ubg3tZMLZ#ny-ew%+T;B8NO
zAZEDnjM`jkTTVy#GTZo&f*-E1bfl;L3~f!xlKgnQwckyr{k`7j&Z;R&&x_x=q_0$Z
zYP)Zcsk!+%Ic<qUh%LURA(|f2hbn_vRc1Id*yyC)=V>N$I3jm6>_-V!l*MvQg%V~t
z0l;!JnqHkL%?7hkX8Bmc{yl$be1wY^xEA-#r}TRF-8&~A5O!4Gr)r-4_?GKSlJiMY
z*KSUI@S(edWxa81=0dtD&GdxXgNrIU+IKc>*>`-;Y>&nBS|LtF1<hVs@Kx`(Nae?z
zMQIXqlQb<)a-3f7x;@yC8pxIMgc&Wz%#l5eeW9T5%qwM2`NndKsxE&T6{Vu|?A9qN
zXR`5>yl^TZ#|NS(<F(|Gba}-}@*!ZK()p7UqeUD;N=DRi{(Ns`V4m5G?R=id?mg}8
zxq8{XE!nVB$DVhmw_A%3CtABl0T&HI`VJk~r+zzhc+SZ(pv^CPFtv2}Ob?X8ZC;am
zy6hpnuvo^3vn*R9@3wzX(Y_NGlqE#Cgla94guqk1@Tzzd|E`@9Czk44MqNkP@sGrh
zzLW(`8e)zp&Ua)1MI5R)E(0aFc*Dq_#I+^P-YgZ6a?sYWW;^=CypL9SXCjTM+4YF#
z2-UOO^LHJgF)_=GSACdxT4@guGwiLDCIyB)D>sXg9s+>xm^6QHs%VJmQq1A(EslU(
zxV!)g!C_8>q{eu9?sBYtP%}$E=3a5>fCw~NplgJq7t3rm+G>g$-BRnX5|lHz!8%gK
zIH-$_XL1HGV$-9(iCrUDDQ&V@HFUd+7bDj>#=8Z>5BNQq9>pm6I>a@hGLGLtX@p#o
zUd-Ub8iWAe1qy$hYJ!6a0s1M{K{buh{z0Jw3wD|g)OH8Nk8f7*8T6~4&`z$)9GyFN
z0l7Uk3wOb<SJKbuaj3pThBHU7MT;$tDYpT%F=Q^pTggbw75?}svg4hab<iyH5s^A4
zt=rWW4WZ79C9NDh$KuAl8mH3Vv@msDX?cYh7nbJ_=qZ2mF^#(?bk<lu%8P75=ppaR
zaYe6Yfg2lAYBLVI=*K_b)4$X4bvmE<<!Oq$R1q~}3h;)pGS3+Xyy*S|-p5XQ#Gr4U
z8tl+4SKiT|2UaXT%VsxQA2>BcHOn#HMsxTbJlZcuenRCgWm_R7v-Agv+Sv0W+YO%S
zhbrdc4{U#-A7-9(j=A2xf5tQ5>T|TZ&E~+s@x=h7oqSYsE2YEWs}F^orw;hoS+E~Z
zk3#UO%Gn1?u)9om?=&sl2zY-`d{?%I(vve!m_}fNrO)3fsOG7hdnZ!eeLj+2-rV1y
zbsM6?Zt3z~yNcfYnR1|ka{qKd9hp~r?rh!s1uB0@A!zN6*+jEvSD(4%)++H3oE<ai
zxi-gWqdFeH%!OLINqO+J?5-CNEeD1^?}`H{JdoP4D6`NX*?u95VhG}vocZFe$8anA
z7LEz8^Ov0ZgB+bM^l@H~zG)G;IbwMef9CZ|e8(nxGmcMOYdFQMqavbD_rMt_|Kw^q
zR*Qf4Fg83EdSHmz6q|f-5`W~NqzPly+s8*#Uk~P#Opd}fKaL;J<bT9tc%f6}L``_1
zFzp@P7@OloSK8*&3rtn04?DWa8mUw=cT*_#Rib4p&OXZNX5`|yal&3Pq_g%if?W&4
zQz~jsn2qIf$xv=}jM8qtax+#h`eLQE71w{_XnbLC#hkwFg)!IgxD&DW4jR0remE~R
zM)mUj-m^n&Jwk_`yKxP)zBWhdjx(rM+Kp@*NZIyg%VNFHkukV{(zXOQgGGz1gbQz{
z&j_k)=5MT8h%2}ncEC;xa8LOTs!@72_i7o<W{Bc-yOVD$u|~ZbPt;3x2W+G=W0ij}
zy5!Qcv{+mte~d=7cG}N;kAF+&i4*UwC{3tb)SqsU*j|>2q<NiH?>a%lB<1ud?$)vE
zkzG%fBkn9+e=(oeY}K3#c<FSR*nMwms^PD6^Od_4dWLLj^z5ytZto$rdPe=&O8(w=
zarY#Ejsou)kAy{ZqA2e9w;vD!QMiAj6BjT~hmO((aXdKQVVWj>WRKmbXGLrfJRK**
zc}48RghzNd4fa{x;O+W=^y<{`Y&C1=w+YEwT)VevgkR@Dnb+&+3W|meXmcn&xeIBe
zQ-O<~W>!=@a<wVOCbY}!_00-IRoVO3n}?slp;^L5cbc0;^S*y=Z4L!@pVfcKJ^oZ>
z*C_wQImvieww`$p15{jZLR8xz!@YBqc^Hjqdcnhnn=g7stIcNC&Rl*HISPm{ue^R&
zP=LEG{oy^jhnCg{UtK#}Sou;c3W7St?#H<gsYSg%TDtF+oU=+~(X3~m;&{l8Yz5A_
z+u5OaHW{3Kw%gX>UHgT@Dhz)Df<ik&g2BxoJ&Q)%w5z&?<CNihA0AwtN6?KPz)izg
zucgthQo2Ob%KJ5i5$bXUb*NGPn%7bFJEvz~Jgko@v8))reS(7G891z}`}XM@p4-ot
z_8Z5=!^_BXo}marx`lo3%Xd1x?tl&5l+5?$oJf*1;aPa0Sy&(J=VX5|-SYxz9*LI@
zY}{TlCSKBp%L$7vdsz5vV~s^d#~!9f^d+s&&s-^L3OFn=C-Le)ntIjg#+h6r5o4;L
znfLpH=Y(#E3lbW-J<k_Ec~h}meH^l#6328ZS-Ur|IUzE`VTWEX*txuZ8JkyLx)49;
z|MY4mXI4bT%g33tuv~vSHQ`Q`SKIIRi*#8!I4%Z+DBQ{xj)X1noUZ}sW3_~)#q^3G
zPJT$6;C%h6x0TH!LL)7e{@c%@FK%9l+||UBUZ5f0*aW^B`J_{!)noLn{|2i;ZMT*H
zXzZax4Ut++Gvm4&u;m9DuMCRI+@G59q*V5)DA45Fi8w}`jTV2|R}Zq(UP>64O~3u1
zA%jqPH{^lUYmc{VMqv~0ru(aU9ojat?+6PXG`Fs*J;nxUvnv%Rgq?V0Y`_1+p8id4
zCygI1!91Glgzs$Zyz#cHr8?-gdZWKkkEN=;#Ene%Vz)a7Cl9lTJ!Wk1MW~7oXGZuL
zy3yJ>oYUl1z{P)Pd9Xhj%{-x!ynp)iHmsr7ha1zgV^0iWJ;TxdIShKKlc8s?PcNh=
z&esgi4As)ygx|a=8!}<K_4d-v8|*@uJ6?><B3yS*XWj7#t$CIvPh<GZz`L`jb#rv3
zq=<JbK|93R2%#D6ndP#>?EQ6le%5TC>vfCM&4LK)xZQs~o~O}ELIrhf?~mLKnD)A?
z3#hKEy{mIk0qx0=eoRwR;&D2Mx>vNK;ft(Ft-><)hR(x{ujGWrw+WkS*(x4S3wb|q
z|NiWD)9WQQ_%?~1f@e<|Ym{9LGX~dhY^oP_Kw99PMsb0i#*Ios552taxZjJ_H0FJY
zIb;DuUXFi?UKS{}19|(7w)zZN#5<0*ZtooF>2aFsNQuMDO>GUbUC66qX6p(MTJjQ5
zs_y1G5<_9n8C=4lOfFRMFyk7-JK13B=Pl&|<%j&?Y461>HtdC&v@4HGDK1J>v>z<6
zu*?v?<<b&2G~L^qFW+;&|FZXK#}tR&F}!Jc2)}>WOHufV-m7=srMzkz8iw<#h?(Aq
z^6U)K&+dArjZzif{ecf{?ssem#<pKts9;Ft3e;^WB!`c{=`%B4j(rLo#f)Vgl`%Nc
zs^Y%9-RL+%)UdmdIX3pG{wAaC=jvbdK~9`uw~C1856yc4bGsJTA!_cEo68y;FY)S0
zXTN`C<V)pTo`6Gw;f@7LGr*$9XYS2!*cosBYEY|Y>t;6#2G2L)6fJxe2b1NfMrMRt
z<`qrc@@M2DTiqXOpV;4Z`r$lorZv*b%I<Mu3!|v86zpQb@-~0}L)~!BV^mS8=C7AR
z7b4GD=%+klj*FpwIE~E4XlS2zp56*JUfzGKS7)y`m_Z>amxc=#3v4n0IEi#o;L1>~
zelcYucM!un_LQg}gqaaykOLyrS0~wCThbv%nV*+zwdNQfjhHiFA1r!YZnd@0)@j1|
z^2m|?b`ZZ9RR@iQ5Z-Cm>$Bo#pXaj?I6qiSzqx&-KB7z;dAB`m(^8>DgeadFlG%U8
zVlLhJ-oU$p<%!9jea>52{bPC_O?dD2OxVpYn2pbrIrl;C?xYQCMc_hzQD>wq)Bu$f
zrkCr$J4|o+fzDm4LZOh#LnEVjI^!+k?gKMFmH{k|RY2R^8Q$G}+Wn-6_{%3{v+s`V
znR+&TX2?0+8m#;L?pAlIIC-hgLOXwjIpIx&cT)T8rm!wBjqDqZQrA58xPCY`+>p4l
zd~0w=#-{5#)pjn^i#T?kc?5xNsAHCwB(TJf*X);jo2E2AW=wxly`ter#O3|ZRAKoV
z$EO|h+|kq%6~Z8g{YjQh9x{8eWDY8$!huu{TU}4wR#3Zb>*@JKZ|Jzf3yOadFSq*-
z?3WbcYYLj9vpW*_8|&c_Dp}o=Egj94dGfEu4)BgwAzKT>5PYTFD*bGY`vzlbI1xyg
zwv2dPsiceByAu+5v4)vdKr^-@T>*K?sO~J>VagN51I~xG>E%c&4xV^9D{P_HIU?Jw
zmKt%BD`rnBT~p1L3kmG{MjwB8%+4I*^hiuip12~le>VJt!c)e!s|38U)9%c!9+f&-
z@<opK2P_|ER~ofG+Vxgn?3q+!hXkU7pq3TYF*N0W*?!smVOTVMk_01Mm+*$VQvHKq
z4t;kL_a(b~J?8`P?J6i!cp!DwNc#IXyh>S-^CzgfFp(d!&XTh&G)aG4C^LAPc(z~7
z^cwTho~D5VWiG&Wx+bZIQigun;^l#{RVUJH1B6Y{QhW!l?m3ENQ%@Nyb*pY|cvTqQ
zdRKiW{k0-h<r`(|0kbKqj%zoZHH|t$qt8JTGUM?^!q4iZTWH2y?%v5btlju<V}i^^
zJXP%#sADj?BjNS4n|yzqz8}tu6$2+Myq^m{70KTSqcL^fj5v$yG@4>kxm}Dnaz3Q9
zveg?S4cD8T_q@lReI-?1c06&+sZW7vC~7-uv#70+m8fwOyhO|m!aomm>u-&(-f_BR
zL)ksk>lfc#nQ%Sy3X#UD*0PIz_j50(<Yd(2MGE5tod=d5P^y2bIdn2+Tz5RN^~w>K
z)2{V5oil-2@Y)CJNDt3z&Wv8a5m4Na%6e5@==d4=sx-^ADp46^;#^&>O^l5)_j$Tg
z{R$KHY}On-na$mM9fPZ}N1hPG2MgVt^AZpAOik0ymlcIO3)LX1<~<)Nn>R7CE{^RL
zyq6Ejy|6SR!VZ5=I94xe@09&Gg2i-XdpNZe<&Kh_@s72As;_RaHkD%!bmrKr2#zo#
zqtH_M{#&qJ@%6_X3pVYrJ8-=3{T7v}{SL?JZaT)FYssj1eolPSD~R5Gz~tr8+{D-S
z{q;^(?>V<Gp&q;G;q-fkeHBuj_4Css@1crv*KSfjVE}(mz0-DY3Bt)vJi^NEIDa_&
zevhQG$_quCN>71=cc-dEI%Kl0Z?$ll*+Z9Hqra@GTpJdAxrAp+#F(e!i@1vc<Mku*
ziC!my9+*{T`&4?K(IJ42wsnQ;6l%o>ybjbN+n-o`d(1DibG&J6;9hYoQdg#3@HY35
z#N(aEE*O8mj;}MTvrxRQe$RB5fYijndlc`uhsG^GSayE!IQv5PfIfYY^zqJes46+{
z;hmD1GaGG2Lgwj|dOm2?^LZ_@+x2ETn9lEKsbPoS-^meb-Ec)AzoFC+AOG~0&77%9
zHKfam{Hi>maRbijoD%f7P`1PdG{??@b8ox&x!r$bb1bsXRNr@W9I=A)NvCZ3a4u<X
zv!Z)l{|*lJb3v9karNpQsXaNY^j6vNN*S^JT0^OIuGN|Z*X(1(SA<$;_wDfvw-ycQ
z+Cw)-WhKOcjJ(Ghd4Hw^tSz50EIoU~)^5%ccXIoNoH%A$@oM6HR&p8{#G~4EA%^eK
z@iBkS(b+*Q6K>P3QPDTUI`>YdGAO?NFtyb!<v5FSu^nJy$Y1{;c<*7_whLtP38ja%
zFQ{o=nd$}XZ1iG%@G!Oy^k~vuJ%WDQXd%w5LDBkPeV>ta?r;iS#X(rvo*m^e62-IE
zpWJGA<9Je4Fl8iN+FzOZ!j8kE2QE$W48VW&U%#I0>X<8x)?8Rp_u1$CtZZm_!ngj)
zkyjj+GX;Tr6K%A!Er#bb;FkW^Qu8M5Xj!DD4rK_mL&PiCMr1Z$wc}>rtUMC6%gG{X
ze)h&*8Ak6wdfEu|%SXa8si&Vhm;k|456mB3xt1B8SI4M&uSSksY2jH~=|p$(@y&m&
zNXx-HC37X6_ZbS#iVVECnL*B7#kXC^ol3WPwn^=7Ls8t|E;{3&+YlI^>3-O(qt|X#
zu#GFP!~LS+m@8BaK5=JE%;?=G#;2+~qf$MnoCt(oZqA^JXDvk=?H$co@igk^cO><(
z&h6W?`20$k()NV}RWgavxA?weRg`~4kPACHymZr_%s_kwc+m3V@aV1ByVGDbhmB0U
zSlGV#!|$#qD9u<<Sl-_n_GVaXK3l8T();biOst}7t65=3t>@V={ps>)bcRK6zK4ft
zF{JCII>F|QP-Bg0ZG*n;8#5zkzwOhT8VYiH5OZuX>^X7AsINIQgOens!jFGjBzV8t
z)N?A9Uu~+PbFY0Tb)Qi8waMM=>6Hc#t2xXUoBd~>2HCL5_t!hhwtSdRzI2!%+|gSm
zDK=)1V41ni0+Ft~xp4Gg=Ecf-w8Tr>h9$Q8EO#C0d$poAG#g#a2rBPtS>7_5`VB<b
zjJm+J_&?~MtW=SucUpFzgyDZtP2Jtgna7t<&6B5vIrs42In=$R>|2Lz96QxAHm|NC
z38A&Lo9rjAl*|oaYw69I+D)by9`JCcCP3tPo%KQ7x$Yg^RC~(C$T6W*0mVH&o{co2
zXh;OrPG$TU$7SZll+lx83p^F9c~Hj#j*ly9aeFF8C99TG1f&HmMRb2c49CyizP&e6
znZBS7EN<TYf&QeCtW(&*@Z4h?1<*O-a~{G8gIohQH6xBWZDKGa-(L*xcsaw}_9}kN
ztmWP*uWQZ9y0gB4(TPPaX8h72KIZU7Dy=z#8n9ucp}$>f6h_Xc!=g%MwpH=)T}t2G
zd}4uUDI=-YOHv85SDJqgmI$XuU|q57&zj)6erz}Ip{_E}aF)n0o;&L4??cAJ)f?G#
zP2g@nX908mq~C#~aT{*1R~^iJE_X5S+_oP6w8R#j*gjp^{=5^nLY6rSG}2yfHOFRL
z(mHs@yX`@%7p%&IBO`hcU~e<ywkg7mJ+$7oD(*rEWz<5U>y>}dPR+O0c9*i+PPPnC
zxJFF#k~MhPHy(L(-OM^;ic5pnUP4WQZnP_3hnKChGf2)Us?Y|nE6na(5}G=^jmi4M
zZo?<TZ|Lr%SF@Z*r<xw66A9fl%i{I$<%>64-WolDy^vv{PWLDoGv$1NplwIK+<rW#
zt?JZ+0an)Q(0YI2&I#iInjCOo=L2I#FOYl$y?g%{e-CWR?9qh+IZCI{L8q>S7kY%3
z7UUN+M|WaAY=P?z@i$v2$Iww8jJm&^)1YZ(XK*_%Y6<SFy{N4m6mfs%!P_TI<K3zJ
z26|h!&c21p4{T@gjNJAnYkJPHzv{78UWG+!Y|luYeM5iw5uw&2WSe4X2l$-s3uL{0
z5>xQVfqmqN*b&AjEb?Tr8w2gG+wHzi56_=qEUa!R)hY5TJB*L%dwphS$KzyAt-$Qr
z_@uU140N;?*g-9=&o?XOw0D_Ea`qoxCfkvWf2fkvCvC=X5FL?pt$y4X5@+CTtzI$*
zb${*pX3>8nD=o0;#)t9(2jP5Wcz+qy?PJGcZH}M2TI!z_jIxg`*?K=pI3|iBb3Utq
z*lkeIcOMu#P+>7U8>$^V0p$sveQ)laaE+_0;xx77Cd(7kkDIz%e5zwwoMcpFkj4pv
zWTP^w`(55<79%U7vU3b5_lN~m$>6UDJhXSEI9z{ax%Ex&rgN-&TT3`bc^xbD9;7|H
z*eT3$63(f<`Sb(O{dl*x-gG7*v^E8NR|IysAQnN<(5g(C(&FQruf!qE2pfiCOp0wi
ziXOVn=r`UejL=b$-w>&(phr_X(4TM~?VI3R;jGVGg60=~H5k2E&F}D1UnU<kP@BbW
znZbW+5YnpmT($M8tE$x-*)~QSvFiuD09A8BgURn0^n1`=Ho}Ks!t%a%ni>fl^B+b!
z*~^x^=rh4t?QbsWRS8Q<oqfXGZ+wnVQuJ8MiDSV%`_Adz)~ATxh1DUyNl|5dCuien
zLa5Z}8wqrl@(@+?ym$)BeGObJy3e=j;m&`oTn~d3j|`xD0CUW{K6<ci+8vz{IOAD<
zvOG`<%d68Wxu=@mG@cg_G)b(vzPpp@+P3M`(S%Y&NgOwIO<}>y)AMk{`4+QUcgbTT
zGw-S!^JGiwM%fO%E#`1`t~s&yPN?jWsR>SgcBz6Qk19iM{{)tV+TGHL&*=rMYn6W{
zCb-_4Nj_B_cvO9-Rb%n;9SGPeV&g>pNq?Sk*yv;{N+IxcdoWP*2<Ey}VX55S83S$$
zW2Ho%urmoaU-hT;bY#nEl|(;plNfs@Uim>8nc!I%CjBnS=){tu@&4F@S>-5ie;iPn
z`&EgW{P-D5UnUf+b+*(&w~zX{|0{oYinEO`d!FrZv)=ba)wegqE2mCSY?hjDevn+k
zZsWqmd)wO;3ESc_@|Epy^g4cHbhkt-WYy0ZzHQ#A(NrLnH=CUW0Ni=GSiPx05?T1t
zOU)ftZ~8hNS5nJ)9nUuuGhfW|=)rPyR+n3zHZ#o@tCU%^&J5v=7a;eDUX*{Z%|qLh
zM-IFZzmwWQQ?A6`m`4?;GdJFq9r5m&9=xY5A#=fr%J9uA-K}OGu>h{;BKAh3wJlY^
z*h&YZh!_j!U7CHp>Y^Pq&00IW_l)*Ebf<3Kakr}ffw5d}B+FLbCwo0&bgkaL(#zK$
z<IC-iFRg!%%ej#nD?W!!;M9L_A3H?8ZxW`(nwWp9uYD`?w(VS@!i1$E3wHy{a;F5@
zi{-;6JqMk7r>MQhGwhyaUOdBhqCtMYMtMH?9k^qVLM|%emScL#+2jJiZ8S{``8LW&
zAt&@3&wMyh0=7<(ek7gUDLeWy;k-bWOyW#fIs0RFdHaW0!{F#y7;1lfBd>;lIghM=
z@6lVnbX14wXEv|`&*Z1uQxxpDZOS{6r`H>k8MD2piY_)X_;6U95@?ymjGrfIA8)!o
zV|M59JUcIklsk<O|4jS|Vcs`BR(swYJW+fUWmj2Mo|173zID%Xi1b<M;-!4udfPId
z@YY9%$HujHE_in7EEIniy>^Vw3@)?j_jn#{eAr<`LgNyRznS<Yn8y(leU+Qm2;ZUe
zXDnWrAJ;SI-<z+d<56#rak#Cy>t>~!{a8Dow?iPiGLF}t^|VsAr|?F-+;gIUcN_-(
z-U^xFv^zane3!^wuhJc~aM<cJ@{V4%FuHXhHL=)yzFM=}CX0W{-(V?TV^8`uJq_Aq
z>4i}sT(s`&hQrVs?3<&GvKiUZWgBGn=)M<h5ls?#aDc9NXm|B3#XA1$=&)-thq>Nu
zPO4(^>CR)HE=%fb_un{LWSzxxI?4Hhobt?(S5X<N5}cYl{SkLs`r&!Qk3)7(4@^b9
z)Rj8+2BZ4;{ZW6Ry)ETwtX0oAI6`C~@L{zx!Qs1EEY;tqn&~P6iL<y%XPz95ZhzQ-
zcOB2-PZb}d=IaupEXvpbK6g8JPg2@B`CY<$ET7Y*Cfzwk7CpYZ>5A6#2N|>nE@U=y
zTAra!`Ov(nu+oK4cZ*{elgsW)Jgl2bBfICJ18(YjkRE@%&O`GK!6w(9N{V?V&Ugx>
zD#bRvxRKHMv=knzbU3d5#Z2O^*qQgVx(ON;N;hZs>`>Xz2JCsvtN74i`vm(X0ce|z
zhLiUDH0Q$SQi_3zm#@us<&AlkJm2Ehhun%CrMz<KX+O7+g8FqjCn1`*`yR3!od8$W
z+79(7>+FA^+4RUkE#ppo^uw@fzf|?>n8oXTaIX==g_P3zEt|kK6H6z-tdRS!c_T(=
z^1#~M4CmFOI&Cw&;0UCuo}Gu(sY<gA%}*8{P`#2Ic>mfw!|6#tlP_1@&BzPMHT#ni
z4#~{6_J?uvM_-wdvmc@m2h>yb)tKCPh2BS&gSUS~bDwvsJ>9p##c|w+n>Ftg`ugj+
zZN5uoBGV<652tNySa~0(_PuyJJ$(1MSR2=oET<mHGlDL?WQF)i1#qi2?tY7<=^}p;
z=ao8mU1(a5s`O@t@ymT(DVZ{ZLt7<UIz`SZ3!cw+t{W}O7P;Nh?00Y<CgOSU<pbU0
zAC7-WmjpdP%rCsUHG2Nl_|iRE(_*bw&46?9gTQL%{n(8VzO3Wwn{o|!yu$LINyTp6
z0Kae8B4K{s)kW&VQn>1k8Hl_>mP&9<i_^QaWBcx!jBng)^r0MV0Dmz*;WEtOd#34j
z>E<AGh?V=YwX=0}u94Riu;E?**f?B*T?Bu_vR&=)V?4FX(Mvh!`)|G)8#!1_DSLQl
z5$p;716_YEZ^Cx9tc>`R`7xWQYL5%6XT&OQIMI5%JvjRPQyGd<7xi|}PW%n%lGV1d
zlho|Dv<&dHN4VPfr9)%Xh90QRw@yBv?Rdj~ayNIw8?w6Xp!p(yH~w){Wb0sJuHt`%
z`=$4Jn_mY-H(j80xAjuYvk8)(<RpuBL{mE5J=ip<yQ9uLNm9MrON7z?`b;FMEs?dT
zFh8c?e1>v!gn$L#)n{Wc9Kk|dks(boiT&)ZsU6GuD5M9MA)An*!~VvZqSxdWRyw!<
zTJazl?2-V*2^=q-U*^c{{@6YGVz+;T)2dHLGM?ebkL1=y-1a_dXWVN)nJ>Sb)OMLw
zbZlm^0I#KPb#IX~g-Ua)a@U<?@P3+f+Wz33G}|n1gx4O7Ef2~Bw~*hQ7FSe)6r7}a
z=zEW`ZK{oB@B5tWce9&bBlkTrtzluaFFSq)w9|tkgeDc)oa)2>&dkDBV55IHuC;pi
z3qB<gxhq&_38NVyWqyu6wOASdQrWontJdU$zT>ekTCRn27hj3d8@uv42=CpUnoJgD
z)n4$SHQr{9eS4&Q{j$*eY-uDzuIIi96`;=IVMRUO4#0b!`vdK;lLcK6r7?j^4bt*)
ziaQ<%(=I1J(<gV|ym8vfG}V8?H1|<qT77(KW=KXdc~slDsc2pzbBSG2tVh@;zK(uz
z3|GL@Nyf^db3t_6hhAnrWIR`D*qOBpA7H9?nW9Z|ThBbdL@Nu<7XCE9i0gcnOVsrb
z&CHpE4OgBYRxd=96q+VYooO)H5>p}=DN_A-fVtx4<)#+V1p;}ui=TgHR$j&j`AU;{
zFO<D<9HYFBsO-Q|VMt-z7?Z%|;88@-m_veMZ<UAbhBM8s&#cYgKE2^BZe^Np+U4V9
zbY8T#N*Yr>pDS|yo-^UatG*!RkbuOA>yhR1=?00R5EJ97x^9|4p5EM&X2;%wibA)z
z#Y^p4UZ>(5Hs_Y8vUPt~?0H&5p*L)Io{o*3;p{yFqb!Vzu_W@@#xVA|o6&r#Gz;hB
zq$lQHB#YfL+qB<XG<mirXZ{AJ@1)UytoL|9SY(pCL2Q%>P1VUB{bDr-^DLK*n+1<8
zWilRylUHs{kpk)3^BiV62+-05pk_FlY?L3pr>@S)xO|S?*tLJ4$1{*K;Hl#6(qz#&
zDl^VAt%vBUZlsR7?`qyLs*;S<3VS>l%*)BgS>q=Mnx5M*&&B{t_NFkz-O3cnyUD5M
zW?#lR(-c|MXQ#l1Jgi)ZEZxnStYK|epmy?(UyZHN<Lx%4-43l~vqv|-_7bBjh}NWk
zd!@bW;7RxivV(u_!taige~>A#-JG+i5vZK*Y8i5;(9G{{)J*IPRej!F=5iVw;);*Y
zGh9F+Y7OiMF&o}UdK#jXdS}L1!0=1cAtnsPDdHAsMtqJnel^$)qGNiSou9qd3coq$
zineAT=vpZWv1D$4-)Fk%X<yvro`7~}Z^C?$1EitDp|pSAI#MUQq}S}kz<!P4$H|kj
zvFyy7@UdBG9N~UDg)g1mnG(%6k-sf<w+&^{f#$OE>Bh*xJ}+9fTrVTq#a^ufwbV3<
ztcz6D5xT55*#@U?2RR(QZ$&?t8e+B2kRB<<7#hxd?(v){Ew-^x&`Foj)3*Pybg^mA
zO*~D8E;4_l+EE);-cfGB{pL*)VKZam;J)FP;-PdpQ?bKQCl%oPBB;Vesr3(!g&G+>
z@2IM?o$Sm=;>_(3ReEt19wmCUbD6&}e#AU|y3)O6j7hP2X0Odl|M|;`b(Z#)mYR?h
z@+%`bk`{zJIhPiLbrQBLcubG`UOehdf92d`JF$Pped70!Zw+ww&p)_Tr?3!#8sCGg
z*wXOARy9ySF+{mHMh&s|4SipQ=r$V4B1@6y`|mn1-Zt4h!P5=Y3Q9Qn=;ozH>O9#?
zGASm#(OHZZ;jSZ(T^=LE?|itD1k9a$et1_SPewy`&tl;<`98WEii?27ygMcLk&#)P
z#>s!J5V4a-4W{WE4h(HLk;+VV(5_XK4^wAxhf#R1O-q^NwpWjI;>N`8aAxm{DFXz&
zWbD6QFl>uVvmYMPZ$C(XLQ7#Mzop((>l>QIp5$(=vlCg=9de#2#+<+-?s0X;-uEiT
z`dHUnxaWkxn}YYu7Wk?1CW)A`yz)-&R^oq2?s}8V<?p<(*&s=+dpDCHr&H^r4?2Kn
z2j~QD^J`pcT7ZF{&amz8hw!K6FG+A|KVmy3_moFL6x~Dd=7Ef9Woz5nc4TE@yTM`!
z_B@+Aq&TFtLrGKr{Md~>#|vy+rk5|zTgluHi{J-zm>sc9f9>l&P{dejA3O)VZP0&y
zrHR@#9Oit#`Ofj%y&G;FH#*;8h8cZo(RbrQ|Fu(O8j7B~4UW#qdz2#$i}|TGG+p6r
z?NLARR6=#Q(W-ID=t2m_dm^l+)c(a&HK;0w*Uh9CnmyJPj8E?^rN|4$=o1c`bJNvo
z@w3bf3@Y5WW-CMv?oF>h6|wC?)$xC3e)GZZdp>nnqobCcPvChK3tFt3gQk7zT~!XK
z1b6tg1rG~mnu4Qc1e)XxDu-*A_GzE%IpB?Zx@9)79S^K|E1UNypntLWeT<Co$s7P#
z_AEnq@N?7cj8=U~I{U=K=8A^TvP$obV{y36)2F0=*N7#wxq51A#qPjk<L7^htD@Bf
zIjZ1$wM~u&=f4d{nAv0l8%y)&?qRI6qnPzZf=!0S#J$AZn7H)HE=}Yl;H5A^><edX
zt-N|BwQcy0hc+ke(q+#st%#4CprH_%?{J*NVy6w#Y+eYnYv;Y2-BNpKT5?<Cqb)Rt
zt1_PtRFl1?(259R=onE^Havfu#|mo=k3Uo!6hvU(a@=y;zDTjLJ`MZog2!*@UU0gf
z3^GL*h01GdjxsU}O$^+VtGg6=K1Pz|(L3CBOYykJ`@>->AHoV$*fv@CP)?64ZF9;j
zy_PYPKBbrtfT^!MlL2{*eI~QFYO>5oGfQ59^I2)u{uzEY>@`BeOM-t#Lwcjk-Y7!^
zwmse=vF-7KN1%eytHG?JRy7NgR&k9<Rtp#FPpONfS&W-hH!yeG480eJl$dZ@`k&I<
z!g7P5uIs|c!6kY!7)>(G`z>J|XjWDZS(!VvUi-8tJ=q>rac1hgyF}$Ptk=k6zIiSl
z+Pr-?yE1F~VXyJ|im89a;*%azyRvMK6<M{&^4HQEW}lYK<-gmc_h=8`I4!4_*sa<7
zHwG-KD+-`AL$VPDw_J|{C4IF8K6sCvFC6elZMbQ|y)E5SWZz|21#@~pwQqU+GONh4
znrMyD?!AifXE7y{6oG7l8J>NMZ*NtfffnC%P}IcSauMN5aRq<h=t&gqP@QKt{ZQX^
z>#7YyzC78Pqy;?NW22%-n)^Xu+*upr%m71>%5900bu@L<PB#Zs;qGd|S<l?9u0O=J
zz25coP7PoGxvH5dndC;bu(VhKR{yL((}HyM`v=@-N6uf&Fke)^Z9KeZB26yY*|XS2
zA}9W7%HFm_TylSD9&=T^+QQjpT+z$7bUhy$74_QjXCu`En;b42s~4$6o<DGCd!xeM
z)a|?Y$_Osr`_8J`3>=o|jcL<n{vi9n_Jg&odYPMKRYu=iR*q=2MB^fzS@>%y?eU2)
z&cIEbd!3FN0-sQa(u+jo4l<FezoFY{{h;7}2{3{|T?v2GuyljXwV#Em5ccvKDC1=P
z)_det_JLQc^h^XY!_#C@T>_i+E()~YIC%oJQ>~7jb(*!5R%_4cOUqeDN;a__*!IF}
zXPZA`Wtj6A($MjNd)~S3e*U(t7^ycKAudOU?c)2J!#yfto5hvK)6$P|X<Mi54o_Kd
zeJMnjb{l`X9EJCoI3FcaE1cdlWw=jlSD4ko1s(}MSE-5j9J|a|-kWzOiu51Nt0}2Z
zA0XsBkuE4a#LT^IPRd1H-2xb}|5Zg(%iF{)s+UKuHAZYddfGH)wrj50vUm=B%Y;6e
z>{M=1RO`i4({dg`-bXXZj(HRgVwqzft8wRbUtfRD8~`G!^4^{!oS76BdZ)n0%X<6J
z)Wr}4WtxCNoFu^L%~HRa<gHvU`Nwr*!PRpOK8K>i_r`(tNvF)eI<0$UVsfvOuyqn9
zd7SFP_3YTxE?jLWotr_vL(UV*$ISUuenm7JCK^?2K1^CAbS3(-Qn@Z1L}GDr9TwbE
zJL!MT8OEIVRxsdG{gjSKaLyv$C8Y05TAW~AIzFkkCx`jzgWcsPBFx#gh8`2Tq`eXS
zTGs>~H^8EE(*p~i8F|YZd4??bq;m|y(YoELrR*6072_M2A<tL)$1)GKZp$tAch^OR
z<#G<_$)OidnI`Y^tyGn4IAXAWBBWvq@56tGj8(}!lUe)ESH;K)&a<VDCnQ^o>}zz{
zjiMAUxmC+p=-sUUG{<lJ*s0`Wavo)w!~D33T@ssYVyFssMOhNY#sja-(b098S~~As
z23lw_vTr=BJkB*MF+tX*!_v0lz9_p~C7}3&`*sa8HM47%o(GjRI#!)GDn2+??r48^
zn&JMHTWyR5Td(I**EX|r>-6s$Q}@)?=jGjzoU{li(9*UF&^0)ED!?c7g6~{Yehuvf
zX>lVx*1DoMYCEP*WsGY3D*EuJsa?);-#%aQ#Ah*37ESkRIG^s(nBFNzh-Pm@!^H>Y
z)brN%k8?A08NDE72Vjl7eGeC;cHMuxM{`wwsIpwoCYbN!HIH*$mM;>OXUjTn*=IyC
zmlx^ek5A1~acogSNLcIN2%H(C@7dkCq*B}MHA265d6vPMGv9yvv!gF`%GfKnX1Umk
zR^)mz^mUh%spr#SeB$;8@E-Hs{&YzC>cv#m4e%#-4p==_F<uU@kPl@mndE;$;*N$=
zCFN~zshmdN=krjI$9HYJ(A5>TRrQS~TT)Z(jr6D2-JVQV%UzxFEp26U$_WWC64QQf
z$UmyP@Ag&wx4fY0!0T>y4Eg)zTDRohykE`{z0s{RN`Lbi(;nfHf+tR({n}H?-bibP
zy+;`bb6-XYT4LF9wZ>m18G3(e3nfBfg>2VZdXC1=*ji23+_}+(v>ZO+a8Xh1#`sm8
zlU73_LY;E`-Ke;W$;Q$}$5Tw${VIVcOgk&%nM-_{#~&>%HdCgiWf<q{<-UxQ)|xcm
z8hbLeFaq#imhO;`@bd$_Vz?s%b>RbAH}+OjNhIgGnl}{0#Fs_mRn33rDqGvjx%ORj
z41TeYJ}0&;{=U@`ZIyK-GotKqrubv}bb$37O-j&=OsOg(wLTkvN5d9Sa=XLvEk(yc
zLFr2#PapC9KL8Ct^1lziyfj!wYNxty-3OrQ3_Fxv>al{gPY6-r>AT^z+JqD%>lD3*
zU3=FRH$Td2<V<8O$I5>^a$R49amA7+)%B|qCUfPwKY8ey8MFByni6xSjhj1sBsO8k
zV@x$E7#FBr{Pu+}AMHS<D%3olvKW6n9%FghV^+3n*e)jl(C;;?p8zUgxlIr3+ub0$
z&7y&Jt8G+VVcK-|5jDJkGKjpC!<X3g2u=RYl`(6SVdHwLl?{J4c4&jNy4qwpcs)Ok
zU1ffp-(+T<COjc<Q~jaF)zMjv^vhemY<K3{!VT}?(OQgmQY+{wlX*G&rlQ@;6>B_o
zZiQ%6N@J1?T)jyA7{g=!B}8x595g#)8c@6H<5FR~5MejkE1=P(O?uI6U+%qCTjCJS
zp|Lrn=suq=^z(m{PEj6j!dZA%4iN{j`^`he_@GWdKgy^tpN~TIxkJ7=8U`{tHW6#Q
znEp(_V5QOWee@F(#xex)_CevkWyAT?Y1Z>Z;hQgvKg|8#H<}kdBw3!eSISiC+)*k!
z5AB_*YPP&6`x*NM!<6H_v`EbMGrw*NlJ9RYjg8*eiU)rrO;mSGG4nJuak<Z`GBqDz
zAs7NGXk-&{UoNLJ1|Q0vxY2j0?BwmKRqQJ@*e$24N_4f2L?#h%^Vqs?J8el9^6xuc
z9S7HP^9$Fy6kTLu(|>O9x0(V^r#^P9OLQ0wj=0Zqvix1s?rh|jK!98uK1-q4j=~{X
zmUkPg@Hv01`)wJT&gjnC@8tslbJwe~7*$v{IA6K$6JT^m@T-WFPkB<t3I9%3I3~jX
z!tnj?FP;8OnCR||n!&HTRHXb^6NlwqKW_(Kdp!*!&dj96e;P2y-=YUh!NEj!dF?7H
zfe&1e&8O3;PsPYb+?*@2(;50xX%tc)hZ>LhXHtI%s4`6U>t-Y+R(|FxjP){;v6C7m
zUjgW7P2&U$-4!?&m`uJa@yUE$>byUMdC78{_q=Zs4KgF$m+3`SH9?QHSS3i_10cf$
z$?U-)q;!(MG~5+*Je$zdB_cucIi{R&SpAV;)vQ14*GJ6dwU#1CKm=<?80+t75#b+P
zzG;7oeagvebac&PBk!=j5wDU=62}y@R_r%~Y8+2os$Gtqh3fqdQS>swz-Nye%yKen
z(J{<p!R5bXG)mXIhY$vMb8xGozWX*GSNl=nk4g^tPd@m#_Ka3(>PCJ7L!yvp?dwcd
zcVVmc>cr*5K>P5+3V)PdZRy3u%-KoQ65oH#>(9QU9Hylg?On{N3-_h7B!d0y9w|1Q
zSxs(!dOCM8xuM*`5BuPl^=8j4q4ccpE3Ftj2;kP|-DK1ezN)mSO{eaUx0bu(sVkk^
z6XQT9sF86}sG94~W8gx-CVDa=`a@~`%E65@>uGm7Wm%$%-^ZM%k4AU!dP*5y&<%gY
z?V9y1JUkshIgT!6St$04jvWY^2>Tp%b&CB|U*Z}H+v=L+PI)lC8fgCR6{hadV77GS
z6R*4+o40~rseK5SaFnwJY`?pGU4B~`U_1#Fr^`xHuicLcuSk4aQg?YrhhgCkVMouH
z8{s#tl)(goZkHrh>9a5-uDn3xEVqAnPlYoO11L!UDc@c;Z|df(yNT)+3j>4I`LCJt
zLjxLd5r2E8F`{iexUHW|7+UB{L+!c<guEUTeJJ&;?QdP$mOwY?Zf8bg2^#T^d_;=Z
zDk;s1W8LnX2d3~yu0U9rr3}7&Z-$lp<AYW=l>&M<k@5BW{4XhchljZOuqc1S9%No)
zvJqAZSaMOW`GBj9-_S8QHL~X2wcA%3Fitbw(mw0^u#YA+>Y#LuJ~XM7<&K@ud(ih%
zSW#f1E2r&k{vfxP`zR&!7rqxtKx{(P@tZ(g+*kjO1$2)l<?Ldl_fDqhyJM))O#z%q
zptF$*3^%$`xLi`w55m=CnjwFwWgv8VUtMvFHv}kYde;RUtBVeLtuo7lftBfGQ&91s
zBK<)`JH+hnT~D-j7U+Hdke#A8w-hVet+pTU_SuVUzlMYDZ9edYBggtERu2`zusr0}
zx@qJq@A^1{_V?MZQ`P&DE=dsHNuIdQ>k9Fjs|VS4jnc(8qzC4kfj@uhu`MBjgd#3;
zGLDM#06we;WcAmsR3zt0?$T-d<W+T{-lsR7wDz?hs>*YmJzbuv6PA(DKISbp4pus=
zDbD+r1&U$2FReMC#EF~J;Fi&9zQLS*(`KN#Uro$n@jd*Uks@2dT7#{IJZ(STu-(>+
zm+w>lf_R!b<@x4z+8lq>CZ;I0@xzqXws%5D^2XskXKw|@z3R!i>K%-|r@hdm%x#oH
zlTu|>2FlQ>xXQf3aLM>boz+a}r}2y;r$Do=mqSN69hDWk<*wP=7c4zTblh~kWh5x3
zI~N{|C)B+rbkuq?TR;r;^i(%+4L-xYyQS_nid!s+83stw-`{_w2mkd;p_0DiFC(Qd
zusB}^OPFrqjlHy{As^hSCCicg^Ug`<*>y!nS;sztr6ke%VF8SSJFbE<Ia*d9r$YHH
z55F!Sb%@g2o)cXRr@6^&8L16-nNQ&_tiyO;YGXzvYacU@%)A(-QBS_idPv0HQ71Po
zv!PpG<KF{z$f<wvV|}o{E_O=5J1%{!<Tp~WQEaQedSBS~#hVJO@m`8ZC@Hb$8_OTk
z&HPIZwNIeGd_h!8ti-MuoJ;}!xnbLAoJDv)o1GCRpU=S1Ffa<ij$NZsYa@_7j!E)r
zt@uT_7MO`{(Guy`+U(*bncrh0oPLYEBDJU|1KYJzKE{8jY&IJ6d?qUFC1E07?>3-3
z&otB@-@aA(%&^krYjg~sa8%)-bpUA3#Xw^r-zF+z=Ln-^W~KF)U%nnwe5-_mnv)ZS
z;(n5Q3|rxPVG8q_D~(~L{KvG&Hoxhd<{>OeR*uO-3?UU!J+la8@!XOc<<_SZSdFdS
zyzh{m*N1;a#cTzkc8|k8x7B2*@|h_-(DK5h^S}wiOUEL5(=IV)>`3;v*5EzhVgALJ
z_*(JN7@O9J)=Y#xp9!PWjL8XVK!o6~6Ok%^=J;8bEIf20MhWpMhi;{+>dKC$Rkt@@
zGP(Nnz>?=zN1HK&k~4*{Sefnyze&5A(%GEG9AAI%6(HY%vk{uoXs|@7;jI!-Y1V;o
z+bfGcYmCk7Xz`KrKSBm-O$GoTzIw8gdYp9gkw3WhPU}Z+8D6Q)ts2&ADp>-7Y=3fr
z?z##gxL-LwE_=mDcCuUJYQovjX!lEkAk4FCCP)?u8f{}hQmp&r;p=%iW4Oy0*g`4v
z2>5>n&38l03~P!|#Co22vS4TM)=l!IM+}AXK@yL<txkg?x9%fc$%};*6Bu_>Z>;f^
z*m!gATWjZ^*ap9=s%tzp>6rT>-e13;?z-wEZZ(d7oj3t48}3zq#HTR-YwfBmw@pIC
zrn_g3#$riAaczviH&<b->(BptMyyL3xOsoy&;7xB5=?!swr5h%PTjubJ~xSMz_}uG
z?z;?|tK@(a!^igyNSH3HbN8{WR>KGR*v7I<Rg0>N<k+<?(%?^lpA@LYdreeQuRcbB
zmLKl;MMj$Gs1K!#;?ez<^SeEHq#!Qb@Qe7hvAwu92}N$54E5JfL`!i4_o9A^fct+4
z|2R|s=eO6v>#xe%R>DjRl$B%Mjv1JjN&VRepU6e2-29mHbiFwvd48fnN)kJrj+K1=
zouYZyjF^XXk3MuykVMK=L$Tg=w4Zzgy3NDLedXDZay8mdQ^Pdh+4HJYd-Qru^u66@
zt(xbZ#Yv~C&iNqEMZ3_!!cgpo^7nt8O?OPc)YW!3=Y!z2JifUgFaQ7oYE7t)3w(Gw
zJ}UeK+15Q0bj`Q%2v@6?!C~87dc2GOdFco8f`?+#C1Bp>%=Nq33y_KZ+z`{RtUVu-
zllSiHIWDAfX|Mq3_1li?s}2xOYedVkP;32N`*}{_q-VYwDEo)cd+MJz;W&SOkHRB-
zuix%(l>(TnY>8GGV2H_~Q*LQ@Q2L1j1b<F0t-Wl!6uDizr-!5k{5VjH0%xeM+R{2y
zO%^N*?_BM6lImt9lr>gb)bY#plxFKjJrh#$3|uNhN}eYA0U;mmRrJqKP7C+Xj!%v+
z&{*%x<glent$y4SyFC3d@ri%sGQmsg>wa(rG5)EKHwh_T`J#pKYyU9b-G$#sHl!d|
zf{oDl$`FAb&F0>>uSBAI^Zp-d#$!A##nq`)`0HT4B|Imx?7CW+4w`jk-E8m2ye|Z!
zc~PsTV<^YlvyQ<4@tv4h9=yk|6SFgN^OLhOK56=$+vxNz{>X0GA~Sz*sLd{BprExS
z)WRYRwu*?v%c<gwH!uux>7vXH+Q`N<nJ4jj6Br~VH+-WuYUauNs;le3r!G_MtaM8D
z?koajX=mBL#l^E*O2^G(11j#{xdPucCm?S0ZNHAtd@^I8YbVw-b5pceCzadEGZ^1?
z6x>o%TP<cbf_eOYlJ$R4ykIXH#81z>ox}E%Jn4F~RaWDz@;+~`!Xv46P2r4U+6;h1
z&W-IwY0P|XR|jN1O=$)CqnOv11LGj*GNXj{FSLk3^psWA1o#A>%tB`P%OAkMUH>Xt
z&s7+?<vg2aeSV0)+I@5!c@Xi2Fs_GiK;5l!R_xVmg3$xx0N{V_+^F}wf%t1e(5dQ_
z2>)xmQ<w6cCgGf(UXhrbM#s3j1_xQp!8d#C+kTSb*Av{fZUQm3+CSSj=}P^W9Of`*
zv$8TDSkvy97-ky3NIUp`_1n8|w`$=WL~CQM*3#ibW#WoqZ>nYI=7zG$GuNV&gFesy
z-mN$uN=KAQn1+8_^ZD136+MBTUb3+%mT38tKx2pL`&m)FF1qp6o!1MVQ;Sh!nvjH?
zC04P6z~yC3{j02xqxM@eW$t7*PfIbpyBNG4eT35!^1DYfRL%TfjC^B^Fm2Q2*tTuk
zwr$(CZQHuXwt0_j+qONk&%4?Ep3UalWV`?Lm9DOwR9AnUQ<cuuaBj>!4m(Ji)#|Qw
zR;L3Ap)eZ&;DPaMMR5Pbi4g!OG$v^aCoGiSPsg2i@yDY?%6Uju`TCk4y?WCWC~am+
zRz{xI%z-s`FQq<FSpG=-<5l#B9WGL4`xB#(peo}Jk%j=zV^ebb#WpcfTKG&SBu{N8
zWa5-0wHbehG6lusrP!asoAC<K#HNAH-|=Ne0TXBB=tkv-$w!PIu`!aFAFk8eI!;;H
zlV$9aGxszqffSTfGF2vuz6VsD##gg}E`!jbGUYbApDyzK)<1!PnURq~o?>Q_R*r&3
zPE^9S{lKQ-y5d0mta;qCtij2?ov*RJ0OV2zLB)SS|HT>%tPFFL6ywu|9Q(ZX1IfPe
z3lGL@=FPJ0>qndS@3;Ubkq7_)K|sF0g;4*RGcz=>Ff_-oGcr@GOj1dh?{VxJ`5)%X
z2v}Z+hrXEWERdAhcGyEGW+&(8)n%t7XXMnVC#28pvC`_W=Ohl&xX-_rqei`CYLnAx
z#Z)))5I2#3pjJD@@34=Qpsk*K=V1F@stf5n$d{ta7f?+)KSxb2V4A30uDPA@<#O>G
z>PyW9k{X_BF{J>S4fAo7g8mmW2^xyAaTz(9@u}IxNh-6u#^&u`A%!=mtLnBo<B48!
z=?95RY#|l0)6)~PGyv3d($lh%l2WE2aPC>n#=vBM=9asMNZ!_*^mi#3c3=Q{Cj7<9
zf1@-eNgX#UKOrScH$O=;W4_0Dm*vwrEKnfp{F<kF(H5J_YK{*8K!O9WQTOjd6jZX6
zOSE(qQ<BTF^wTnu%agK(^}1e1H?{FCCzoj#9z$-o5oUdQgkS(Ee~~(C0smLh6%P@G
zRZ*gUzpU+`<fky6TjX$VV)k!YG;jcv`alc*AMfseGQONFt$_#I>-?20dL5^?yLByf
ze5d1?r2L%-z>hA1eaXM(5KgWx_pbjFMV>a-ytrY>nV;cei@9k&NG3(DFC4&F8qDGU
zJE9(c-00jqz49xBB&9Euc41Dx%8d)d5K_l~-#a=oNd-i`#@{V1D=J{_`)8cfZ1#?1
zXuExk;%QG!dyxjG0T_1h7YRIafdBJvC%{sm|4y|WPCK7Y4~-u&@mKL{5{^Aa4)0WJ
zp#WfR;KwT<|3&q`h>Kdv=x(-JP213%%fIsUk(XB5u19;~<2Zp(L{-pF&<amNHaGWw
z_%b*9g<IQOmj;dQboUe?g9*H{l!BGF1*ka8en5_PzQ+_jexm(&rpwt=EiOG4UPV^Z
zO3~5GPtV6pPm53DTsy2ooT&Di_1v$Aa@MP2>|VVu<_&OElQ;bmI@Zf0$9wi6%Ty?5
z%yD$#8GV@A<YkIMnlWz}pOf3GX?Li9(3<F~0VXL|bsF5g@jIh{-&=cd8`O)MY?Q%j
zrHp~cUlHpM@-0aQv-|OU?haff4s&e}`+rg(<;x_Dl9x?7Y3)>sWZ3)nJm+iSul5ch
z4pH|1RwV=C;woLkJ2S<Eih2RTAgl9tyU|btQp-%vieVU5(o|(puMe~%J{1ptKg-Xf
zddxEZ5Ui{~)M9zR{tP$(ObV&#|28wqjrQQL&E7~&+CHz%MrUr$uRm@Y00tFc6sy7h
zFY{RUNBAgr7zl&6)3@iCiI%4$jJA>yf6iBRdsI&Y6C(f+s7TOEn9)Yjiy>ekb6!_D
z$}z>M71Jo49z-JTl#E=hB#j(@-6;nft&!B#Wo9DO=v`Eqbm@)HJF{t20M<6D_6m^y
zlB<6aO3dlFeH!hQ%{&|~XSd1dkEl95ipLX*{397wii{Gm8K2#z@HeYIacmn8&*YP*
z+sP-L)!+bx3j~T~p#RFBl*|krZ3Mj(E!{Z1)PMZ8yHFoL9s`5+H)4W+`#nu@?36b#
zzqb>LINBMR=}B7Y8R;=;lmy<#60iI5Z6J%J=X|O5r5@_)joJdhkCBQM!~C1l&X95N
z|8Z+7(cYVMJ-6#wn{&w~r83)oS9ucpA&Cr3pfL6y%GG!~7<FA`zK&O~WD(zDS>E|^
z?(ft^IG@@YdRj6HgWMl~*ubl%I<8Y=ujYF)nINi>`t(Ks)f|RF?FTx@%+DQCy~RUp
z;Bi@z-E&-)jWDq@QgoB@^TZq6g~%LQ*FXLGW-Y52@AewR`JPb%jYuk7D3%J74JCgx
zg5lLV7D3l{@NM@mDnsmJ1kiL)bo%d$%wwk2$MiW}{eJh?4p}XK<@Me0_34XVIHEG>
zD`<%)A?r-+JAMT11Kz*xpV_<OaCurZ*>zT68KtDOOf6tZ=Ze2U|8$yfc-z}JUDoS)
zaOO-R8}$FL$~LitFN(nbrPDEbiRs1vAbRrtKADMh+WqiB->%f9{IQ*$LDvjJU=^)6
zuRtYWMgNq)TJ>yyl6QRFSu;)6`>to^S}+4tRZi1V$;r$|&(BOxgICIW6{&rmx|(GJ
z+9caL<x$0{dFn{Q2hdz(swsf@f6HBhLB9Fiuk+)=D2*}*iqwS>P1}(RhS1W<QQ7Gw
z**}e)`6cR63A2YR^eU|R5yL+nAfHueq3_8`WHcIa70v8_7>y)o)mG7KtYgJ+izol7
z*zQLvLi$(o`7kno?;Cw(@xSytHZDKADmx)5GqXZHCT)I~g;t9_GiI39b>_1KCHgs4
zgN#ngCZeR7m#CQprPd*Si+!jRZT;vs6WjYtO-T1nz8FmoF!n=e(*W<0!z(ZOpDnH3
zpOlGM_-{ae{Cj{Dj<Sp<gO1b_mzIbQSg~gFF^^b~Ik$h@gWzFM&L#x@)G%6L<n(FD
z?X|G)&~%!R3gDkIk%8&7$Ee<mE9cGuYT0qeAhz9zBOL-8WE&$5^tv5(PpY7=Ffan$
z9Pg$$_g2ajR6F0luviGa4gsGMocEPk_Ap{9(4uF5Dnw<3kkGHh<w~y4sNL$chVR9f
zJ&sp0bpxi)DcYC?FnHvAj=kkL4mblqs{<?bB6#Gu#!f?pFu;V&zEDQFPwU=E4hdH+
zM9f?H{~cwgfY#jV37Ils(noL7{m!slFwL25R3ElOx=?FU5``P8%2o#ci`-NL-Y3Mc
z=>bH4ZpEfb&ZUvupgI&Oks|rI7)5z~-}M@~56JWzzaNB|#(vo9&+tZukqQrYG%DKx
z06VLzWE$`f8C@DXW$pn1fC2+nm<6ExXRX!GVF(HUJ`3d3n+)xdGp%!Jha>Lx(<iT}
z0jdc_7G5mkZA&B}B$*<k-Q12dF+m0qk#~N7Ad~6qtj=CBU)L?BM=zgrt8=j+th5FK
zfk1$rxw!1gd&<4rp|b1C{^{p-7tX)+&js@jtJ!Q$O3A(PYRk8Q-NsILldG}^`%39u
zDFeFiXSOb!ncXLMTeI1z@bOKKXjkjIlFG~E<61N=Zl^;kYqTTA#t2LeXWRFL-o~zf
z_$9b<XEwH-FC!gQ8%tF$F1Nj#9~F<*@0wbwwb~CJv!--i9kn(KhP7?<!plzk$kzhg
z%?NFD|9<LXrvoZi``-{RYbom;^poyuTs<eX4-eHAXl0fXOfT!{i&8d><CVpuD6<Z`
zd0MgVRL_oonv$8U{#q|gLvuUyP<)+#`3!lFn#P_dF8#JwvvoG%+j<SF4lLY}vR+SY
zY;XIV*P@a>tLpie8rvycE-zJ-p0;UcoZxr6m+dRi*S8g4wp~0NjB1d&qCrvl$4&gG
zRLz%GjOzZ|o|;W=>l+<IzPwO4IQ-jyIG{@!w2|$-2y|ENxF_jVZ5aNyyUwwH{*qy>
zD_pt1F?ilL-y2&Suvw3WsH!dNnjMC`bTRmBslI+TD}!^mx3~KGzi)<quD-FqZyfwR
zU&lvZPaj@lw-cosy(4l4@m~#|#<w=hZfAXeV2A9MM3t+nZ<goKC$)LD`gOLa!N{8v
z$r~@tT~`+$^xH_k`#Px|E^vE);_xi<8;sMmdx__nhIJWHdZvt;DlO-m`~qdKDvd0q
zo#US~TC{7Zzw9%<PsoevuL>QJ^s2juBUHPbCM6el*bwFzqQP>T5OVaf<b#BiVa!7e
z<-6OS3#q0$&n8J%ln!J>rn3su()c@i1G>p2X*Q1~sQ3DwhU}t9Wy@KAL|3o9=N0N^
zHROB_J7?fV{BCsecX;g4i%aomW0FDdRx)zgd5d3AO@@SF^295&RIVS~^hM$6=NF<H
z-KOsuBcuG;bV)A3_}`Mt+bbG$zB=4+$4mGIwgO28$$H)rkK%rS?w&}Fc4!%7xk}t{
zJT|eDOw;7U#fb5qZlKwJ0-7~RS+7c}o6qbgUwY?}g(d??FM*;RNd=1B-QT`T?X8)s
zYcAgrFqky5=Ucw6CdM<3XR822EQAS+>;h0TuqRbj6W%~#xOq6<$gYB~uHH3ZYX?87
z3os}_YiO<wLw5j}T(3d0Ftu-2L5ShNhy6d?u2fIxX{k#&i9;oS%?GGC`?k1>ktG0Z
z`OM<{EEb|;D+W_4jSLw{u~NJr&xfV+u5i=-K=WtePU%6V`D6<w{!PDc9nhrNm}PPl
zZS*SC>V9Si8Q6#>Sjg_Ze!C*bEg~Ox)TTBT45_chqModVd@%5`VW6s`$^u=Y$*eHd
z-fuaEP^sHS=j8W)La&$DlmduQrTu87NOu8XT?#Z|Qr;`Yl@J^Dxx2lc&tlAvi%uOs
zcj=*IN1za#V}N|xAKz=mg`FLdur=B`)jtXWPg=&+HR+%ylK)jyo%Q^-hw9U7*4Ssp
zw99{e(RxL@{eUB8klG759R_Px>m<4s55QAnM}%ryl1%-7<80@<fl?g5LnmH-+GuN!
zn+6_ABD^Q*QG&}cD8SoTcw1D54Q_D~Yc@@g&_@>r_&X*DJVxMoGFnv?U7nB8v*;J4
zx{sAJ^uGRnjz@5P0v6sZT?IZfv6c~Xygm-2*K)mQD8u!r6yDqo8wW^pLk}Z(iA&&O
zEHOvLCBjC33@n36Giq~xJY4&J(5d@zNi8C(3NO&Dvtjo)<9Gs2FE+s9>GB;17;FqK
zz}#<;)6fT^h*nL*nHA_5kMV3t6`q*D%vU>HGDkbYS%Fp*Hg&rWx_=9emj$(O`GB?j
zmjI;bt>z7<<Y<SkUU?<PSM-n(<j}{0iA%E(jU3&7JW$39_`sx(d!QpClDHLQdM#@*
zoL^T9e0nWOs!LRAlmGpz*^7``)YMJy*?2K4*@|&N>_<)|@a!6qA=0$U_Gc)`Vg$@+
zn;m8;<BL{|l-bFcZiU!z9W-T%jhd0RBs69}rML<`^Je#z7Qx8SkRPXhwb=l4a>sS>
zXRTj<EgcO;Sf6`JPdlcHAab3muw&N>#RZ9I>kWGi1Z<>8WF1uirCP{2z5as|0A`7}
z42`=ky4R|wgTf0y^^+%C5wKj_SUL&k;R`7ng_Dn)Y-8GJ6?HK(qQ*J4%25mV1>+Nb
zGAdV$zm_m~Z#cl;{;O24IrWJXJa<Ri=7nc}-il1$=E>~@n8L)kc7--3XhEpD#J=2n
zJ-n_1g<N%c2Re2HnuBLQBHk%~?^EDq8q4A|vl*;oiM4S(GlMy6mmvBLHDe+iMJ^)#
z#7{O2n_D`?EOeL~XrxjIzU1<KOItXgPh1RS3|ykP0uL9**VMFZ=)O7z&a6jAn<=w@
zlc%+d<MO-yVRXeC2*cqDuvs<J_JmlQj%JTy0JuM$@C$1rjRWM69os5mFydY+<J;E?
z6;up_rYxDmzq%!@C(TeyAfz0fJ-E95_B`y>!CGZsW!6(TwOFp4TN}I({T}58x{kIi
zMefxn{|D6e`;Few_lk)zj8q$K&5)yiLL{KSJv*&CW|}|QS_Cg~kNw+aI%QJL=ZfKQ
z;uq<9aTzfUEGzVv`wy-c8|&E8k?XEkA2^oG5yqkkUqlW49%!9XS$Y|{{me2yZCE?<
zWoxVUR4xFN^#{=tl~mv&B?2owNW2~xPbWmUM$+A1X%PP{ZLT7^NGi&!L_E@eih1At
zB>}fu6nC6jD3O+SLblGu%~QOj$&IFZ_7FN2=667OFol6!=`kl^>?gmF-S$k3L-Lh3
zI1F`kky8Fmo5j#tp?qp^)SbYn0g@5?aW_P8#jcMWI1$<EfNM$RB))H+0nlxf7dj0!
z22wJW%<Sis$Yf=IUo)Vo=%5IH<EirOKn6$4`x>cH9~3$~vR+weDrr5r8-(u`-1TcU
zq8UCAXgP7Q+!wWXMT>H*Vc5)aew(OAem2rIFZ1zyBPREnFMO7GVUiCgj$|j2`<Qx`
zz|9)GoYQB|dl$x4r+&vkEdSXl%)gE8!-ypvf}Lia<0N(R4DYn3><OWNf(p^Pb#_nI
zB~J6#Dr*7K5uW2P`9#(xXUBVc7y|bmMV|OY>-^t53HnfVF6I4I1ss7J^Jx>9f$9t%
zX=$05^o)lA-C|tje^HS<Ej;rr7~rdHlt%?C#-qpf(%-#wU||q7lgfb^HUlHNGjL^(
zKG(WSzlm_p_GFD)0XnpQCvONXIC7t6F7@SK^>A={x41uEK7&uS>y$$hV$y}e^|fWL
z99~Y>Ul-Bg-DQ7HqtEl{mB)lwC#FvMnaE#(SZ{Z8bk3Mu3F+I+23W*!;a^~uUnLH4
ze18Xbxju`0Uc~uvWJ<lCbhGFZbF^Q%d5W~5Jzv@!{pedqKE?)r3WRkJ^T!c(fA4s@
zJ=&j%V3Q5UC)m)Q;)UcV#(iS6<=D;$;@3`C#DZ8rjtLcl7r_+I*suJXB5tUpH--z^
zx<=q8F=;n4s_<t|PQIRw(mUQS!{gIWZ1{dXxY)`2x*79l{P>vd`E|0Q<Lh?vd)Uy^
z@%^}beM7_3%RV-LKl`%uXHU%kxsjFJre2nQcC+Nu?cwhEaKe&h6DZE>Lo`kamnq8p
zbXG6TE9p&AexzIj-3}b^!BY&uj!1>3N+v8pI8n7^g%p{*TNDSxXIOGu6Y|Gs%k&}d
zcI|zcAl(5D#xoNKf?_MAk0Mivz?Teg59Oi16u}}2SRO5ZNGxNG_#TAZ&7Tm02y7!W
zw~1cV5gIG@8iyr8RH&+emLM{>ye6S!dChrlFY#3+g)?iyI4GU#V_aX}2;(cJ=ymV!
zX}Ub-=43hdmk`l4l40%o>sN*C3ar3=^0UJ)+iV%d6rh9<W75&=npxGT3<<utFKvVm
zZMbC7ofC0?hgq>%v-h$c0jBzJVY{uXUrcug3(p&C+ameE=4f0n5{IX2>vAvwDqVf!
zY%QMkM*3iIaK1jT)bc4#(@5f;hw7JRl+>y7X%Lo#gt&~Yn4p)op{BbUGkILl3zC|w
z;ilJ-2I-=4Y&uij@mBEMIgh~O9qxRq(k{{yC(K!YQN8ek)H=TNix?bdf%Num=~}*d
z2&d*;8rK`LUJiBRLFkdG6rX|{1s^)fy#ZeD-wR8_M3-3KlG}+Rv#tAZtA}m_BPze_
zw}$3q7MT8m<k-}gI6V^hNM|@B5A2trb#K>WnL$ILWZYASF+~auZegxz3qIQMSsxlv
zZpyKLGyFwXyB9ldjNcF;>YS9ogOP4sAA$y3-<TfCs<Psrg&kn4t9#%d`8;w+S7Ntd
zi)4<)wv_>2owdL%067TI`zJ#1cZzF&`C3_&h)CdeYN)BjXQ8}#d{Bdc;k1Oc0ev+P
z^KGx!7aR`Yk5|ZA&CL)O5<;Q7&(S}m;t7X;oeSjmqGm0I1oKL1S~+zKvyevQa*<#%
zhS5hRk(($Aiic1a8<PQ<{Mp@_qR9(x36gC0lTk+6jXjvd5MqKtU|C=U6)KDC#d{7h
z9c67sSgN{_H2R!>K9pahiDzWUUYh&)1-q^u(v(2XV4{%m9R8jL3XJ6J%EYQU49=T>
zSgs3us3@Ks^$Bx<>T~E1W#<<Ma>Zq(A!Gp!qrL<Uav=5gV`I0A+jPMfR-jCALF{0&
znO@Nba+_nr2DGYWuFXU-*7^|r8k}u9>1>31GLM0Lq~&y<(61LZ*cRu3=G0PLDBp|G
z>JaAuZWL)!$OTWbK)G4j55*gk1QFtYmD`$Y6BN3DOJlQBOoauTZNc&q-4u^a(;;$|
z22it9AqluVIGMynhK?hc%R2a4`Zm|Zv_t2FI!y$5M8{2^l4Tcdq;i22Z>>umiYS<S
zlHWw{Zm430&oU$q*;MM4>cKo{ttG_Pu8LD;x}9BQ5v>P<Qqf%f8W0B!PJ?}aF=~~M
z4jJDZ;6q@H+}#;byEpxDHo0<0c`FFcmpx{HhUQuKPWpJ;5J|>JzojJ`yKlxJ#A_gK
zh<CrR!I0Wo!{aNIvZG+~a4L<Wh0y`Bp2t6aT!k3geWCJ{4k#nn^CjmRpt>R2%^G%2
z)QEcUK>9M#AT6hPx4g;zmXRrc2#Esv04}1ZRs3@pQpw<iG7LJ{>s-atgs)`<v`ZeG
zz#EN<HJ%MQTMPsq7VrF8_-?c#W^Th8D-v2<0zzPM>-CUcvyog6m?A=HP<E^N<T{a7
z7%4MDtaU0W%1Wiu#}KL0YetNwRr67p%m8)lW2^xqu-JB;K!kx+qNpx^+zn#@m(5-;
zG#6F<xZ`CNDrW_aURgYcC!<e;)eh&?i+#dKtdI?A8`hMSz1Ww%4>BQ(w@sejX`5DW
z+SboOPjha#9v7l3u&4Bz@KQW3oeomfN+@Cjh-$F~Ne;_BR3b)eTK&am+ssa^o<qU>
zGCC@T#00NrUbatN*ObhE;rm^D2|z<QJbI`em!Y@0=l%jWub6GF6S8C-Ci`ISBX$&u
zXe+FaBRS54B#!}UT9QF+qB5Dfm#8p6?hRo@J9@+O3D4oq1*yNa7FZZi0>WSn<4c{Y
z)<V0<!4XHu&VP)u9$ejcszl{wQOqU6-!`zXWruXIzdItb|Jzr8pCNafn2~_G_uB_)
ze_jv1X=quCQalki9@#xkor4Ev4s04HkOXI3)eAL8LQ`-P7CU4}8br)PyaHlKsGC>x
zWN?wAHkkyxD8T!dq&RUnSf{_H`Y}xaSV(Dx$e_gK#k4EN!EycrFwOfX{WS%VnnF}g
z05mi~)mSp&UaO*ik3`%UReh|0e$%QNduHBYFlbs{256N>bz<*uDDL%jTwXRZF3yQc
zdKH!A_D^XTkSjoAq9mOy=RA^OABa}=9X8rnzn~&*>4MW3y)mj`wPf~EQBPR5@8QSM
zAmjXrIDh92YeRLod@D<4>;=2cur?4%<(EhGcBQTQ(E@OPKAQ#=qOL&*0dtITb?@55
zN5yd;qpGG%Fe)KRnW2x`k9W|jq!3{ej*({){xcf6gp+{O?2Zq7!`Z08F=ss)psyjJ
zKSXAure{tATKV=AFfpy5!xn{HJf%U`;a(?0Pq=YDp3c+39&I@i+gO54`152TG+ETf
zjP;Xbt?{ydU4#z9d)zQwh3*sqf+vl6pQ0R=G_LnJ5knabIUsE<B((U@PB02<MSN0@
zrW;;9E-v0zft9PcYOZOq&?r>*XH><!&#ib>3Q>qEUSx#hqWchg_xgM01>O|BU5A|@
z;necflpt&rS*n1E1@LM9qCg<cFp2@*luC?_;+j!^>06fQ_UlW1%bWT|?*W0~xfIca
zA{Df4FD>Utrkugh>G|IU&-TzOa2A5|q1jCJX*|uzp~KTQ;#}gbznpYhFSqZyWk8-A
z9#mgSk}izlcn?jC`q`(x$_p3^!=9RG^Tr2a`R<`VIQF6Fd|T`le{aDmO*pmOv6(j}
z{i<7kAdqN{j^yhpj%aF2t31*V!<>K@o@}#z!mMQ98b%7*4HPJv%QV0g73q>JSS&V1
z3w#So-9=d|O=FE{vGSZ#p4%H&u|%S;Jh{DlswqUk+FMl*+-^0d;^Hl3W9&pol%*_6
z&_Xt)J`$=E;4^o4p7+b#zHc1Zruof74^5VT+WB%S5S@uK7%V53>l#Ir`j*cO&C|)h
z>+xDl^2pFGFgXr{n-Ux1-|brzMQ1gS0tEbmVlg8jFssK{TP#rJRMD{W2zXV0<tMh>
z{Aj$O(!23D%cN)5vT{JIiB4i(cub@RD*dV6j6fWFdsQEhiPq*LW%vp~3R%McrV)I9
z(A(C{A6;`X2$T2;5-O@~RHwn@DyR7vHVF{m{}wJ6GNbKzyI+<z3@){_zs(y;T4C>3
zq!^V}`;f(J+L2~)mOAB>q)m4wKvkmH{rPfn=tbIw;8Svl5B?BKEF%U(wMG~EEiVEU
z<qdk)2y7S9Lie4w2YPXaq4S`~)E+Q@cVt{S(Cr)8iQE1uNtOClspDJ}2RQAN4piTe
zgE|W@iViK%Vw0s}Mm|WjN)AQ~K{Cv_SlLW2fOb!sj(bfgZP>kJA(d`EUNuoCvS>nj
z<4@!-f|6@;%Nvdi+gs~er4rScOx~ooj<_2Kve;w!&*$k|p=%*d!qkBGsqEi>+^=@=
zd~r+~wW@$?^x-I=bhSvbR9wq6=9`(JZozQfT{|%uLp>P5TR5`ebk7}b3=4D)g!Pae
z!CUMJ*M;D@VL>`>(w>b6rH42S)+P=?tJYCVW;lgpf=&fn76(9|1|4Nl38F6N`Z781
z5(fHy({4BfYMep6dJ)Fx!cX3R@%7BNL-PYj6N*IP3WhYM$D9hB4yp@LO$i6PbvHMx
z&WT23vs?V8pu{>BiJjuu2Cr6J9l+i^u|k&heoZN{-Hgcg!r?{|xhaxc!6PWneHAgV
zpe{`m-v(<hqkHde9fy&$_SuQb{L*2%%1ovqa-_@F`D6k`Pw*$)%PEC_sebHZpb2b_
zRIJ3p$a=*zXwZvIC6%r4Y{vZ=a9dToc!@<)JXGhN#joxW=)phMG#PA$?u~B%9!Nu$
zT?q8)6Mm}}nLQi^DSgr*z)DjDzvT(xo(eLbp;skiQtc;*Kusz<Gjz3t78CCwTC#FB
zWPaQ=$`NreqEdKiW#pHCF9Z#V;#eKkVCJ(uG10tNT57t-Knj(+LgHk+h^kqcxoxdL
zmVT#H^~rv##n}PB7c?1S&puS|SBMMKh}5cRIJP<iO{ky>x2CH;2Zhw8!8tBe!Y0sf
z$YV0)!{f=);*lE*TER0fl-%sheL=mq@t=XpB=vg+;a>jTvPy%0!iN;I+fYLJZ?a7?
z1)~+Fkn9mKNBi6@^P8kdUGMbO70?EKv??9^U@q+iOSl+<mFpB14h^67vn?JKCy-T>
zSU{nXyT42gQ2Q>XzywJgka6Q-Pk}ZLFH%V`;~*wX*i~rHNf9_=dv5q^nZO*glnHkY
z(DM_=XUG{-i*7A{hkupJs@-U7c9Ur>#2|Sd6IJ%_gsy|duqk&a`^wIWEVkB6UiU+c
zBzIK=8jn9rr6z3uo^Sx<wkV$_SmhiQ#<QX51fKoi$yB1%-9KVe;S2$Bt|URcCBflh
zG^hHMu0R{=NSVHw_dW4-6Y<ykK<7;Kw<Jg)Lfk6^(_Kb?-jpDhG{R@N&uF<}fIpnR
zGIO3tYiHs28R5tV1{Lh3B+_2E*<`r!6(VJ73jrc`6i_U0Bkks|9sd{x<w=*2Y3pR=
zYiH4fGGz<8VZF5P^S2nbyNIy4^G8R6k!)KrhH9nli#R<%kud7#Y*RDPI9pka#9#(d
z3$G{ij-QQxhiT`it}yqIll&voIMncKklCh;^R*j(sw^~rk#Fd)g*8Lb0MDo?ciNwn
z?d79~*4vgYG5-9OxfOtBCd2S*bwwrz(^x{yVc;=selK$Fth;&|<~2fAp+`mc!Lv?Y
zgAR$*P@<K@{08fhr%NPNod=g}>WfLSBmy+1fsLepihD{*;Z0uh<@)s>{PXQyZi0+K
z_f7_eA4x<jmsx3IjHRYWMAnGyR6daZzH+E6D-`ol6}u)<iWU#KjaP=LdF4cjtQ(ux
zSP$M4PF9GQAbM_!iajaHa)LF{>B2@?gCYHnQdKvT;_(ywZhEi;#$YDT#?#?u`opl~
zQ??U-J<)c4t8|DLM2MprRdQ`WNx1Dw$3jDW0%r=3;*qw&7CST+O#8}BxL}g3ncmdV
zfIO4APw~B<2Am6&J&}55#L~mPqxE<-+%H{YkZ0W<VsA-lMPnSvnBKrhiuUa5U?`PV
zg&(z7^-^<8l!USW#|int6w1j^h$=3c(xbqC0h^%g`{{`o-AT%gB*RW?6tqv&-Mpe6
zmg9|S<6#eFxH``7bpYQQAc)Sw#Qn5=y5j6oxx+5XYy@|;3b4TE(@2;&v>fxbF^M}4
zQfwmB)un(Tje4dS!#^t#WwvUwlHo8r8>hq48<y@K03;6cH1A}`2Yfv-t78hPPyJMX
zwv57NP<s5;mnI@Y&#Q7&1`tN^Fd{5Ion-{Cc7qPu$4mi#90mM<sVp!wxK4{iFJrhO
zh`v3Xx5Kqsr33#hwjYjNSo`-xlnU<2>#thSc*z<Z<@F`&hg^EAv0!B9Zp<<g+2UkM
zzv3d}!)6H<Jg73~3kt#*-tjJW#O_;vhR3%IXX$aUBYhivb4WjYS+IE~tE~gW@ISIW
zl24#fCRyLw<N&x|Xeh~U$MTCt?{61c*_cXpIF=)2&ukR>-iFAL*HJKlvT)OOZzn^>
zNa!`TfEJ8vAE}wUr44uKJmrD;-U&LohcXq^go#4Y`RA_KPoD6zHa0)Y<8c&!nV@;-
z_ouu7flY#z65GE<#z+pGFT<E0-U^XTRSI=(=_Ziu6YKq0{wm~HPw{$s1CU~$)PBJ)
zl3g%4tq#=(H3)yw)m>{;GBook=mc?$3Rvk-T+*eiB#%;s#BqHKsLUJt*dQ9&!PSKd
zQ8lkPBy<8e&tv3v<b%?+vgtH`=mq4nA9G<X)(n~z6B=@&mo1?W|IE8P3yjx;@zb!7
z;^KR(#GCXshZH&%KFE8}EEiBx4)6j|!!Ko;Pj_(4FZM$?+%7u@Ofq}=dBW0&|KZC5
z#wEIqMv*5U#VmP3a~zy(v~<+^0ZeuJBvjfblTF3PtR=lRb0~i;6uPQ^|7BLl%L=h~
zsvaNOdkmW7%+TfFXseKx-E3ctx>p)#z{lh5z)cC8@Ujb%6F(9F3R}qqF0;aclQpww
z8${Cauvt$o<{nand_zj^xv^zgbrRFR4oHzCufz~1ii|<Vwxjek-Dp`kHW5!;Exb@S
z`i)}WLmSnd2J!U0BwF2n7`1X+l^Im|I30n^kCzS<EG^_xX}!FFx<1dFS}9;y=#xS(
z-vo@}6cp!L>p}txJZod!USqcWKwk}`mv^Lz_4XZRnWVmU2pT-)ijWtU{Qa!^WV3Cg
z4ZC|C&TlP@TzC~x3)W`lc*n&qazX9Bcrd(ij$0!5@zN4=+)e9$0Gmq-vxV}ey#!2)
z@*2B9BLa23F?xz^60ptz<sly_fAeQ#xnir%Qbvp<Sr(TDTY~e&Hr+r;n)(InfD)e`
zu>o#*4{IzgpLW0I?Ip6N;P7Bf^-E{|MF}sviO!zO5RcsVj))Ak)x)}#tkz|k2~G+9
z+6M1SYL#l>kL`Vb$<uxYIwKQOAC59=w%gL8<9U{N`!5kzDzpM!2qg%CwRZ(TQg2=H
z^9K~Hs#*w}EV!9$OIa{2<E`)?eDRs*It&RIZ}L^m(%o8%V>D@vDDn*0zoh7omtzGN
zvA;c8B7$UkAu@C}hvc97zxVm>`$3{=@}sJ{q5@ky-k6hrAMZhXGl~0axqc)>et(P;
zWISUoZxk+)pjEAZFw#YI)>@0@p-kz|bVcw3e4H$RRf79sQHP1%{5A-5B=*Nl%426;
zvz?1PH)2cMAWMF9Oj_|_#NE_auBGZpP9V9sr|pKhb0np7I!TqyKpexB8TXxaa3ISz
zPEHry=YH{jjO)Pu1o0#r&PWL~k4dQ4sqbr#6AtJisE1u=u0k(4@@wL>+CM20NgqG*
zj%#tCc8V2q8qx6NXK=ZOzG9h~(o1{6??^?}&Rxqxw*BQR@JGZuKOh8z3pG)qOB6a=
z9(vqfz2yq~@R<Gu89;1Pf#M-MvXtT~J+dLBhz!Ml8S&m_Az00195lIAn*?NUS)(lh
z*eHt-YiCaQ9aFuaF@;U_Upz|v;wz3^%*5e)$hKh)oiV-tBF7(XE(?cDb8vQ1GG^w~
z`q5ib`w4IjM79znK38TUL1QW+P-9eN&IIk-w$77-zb5lIuckp37qZ~yhJRX*m$DRk
zD>jpV5HYdcuZJ>+dk$PLYoky(!-9>k1Ex;7X7Q&VWX>jvJsb{+_&dGyC$Ie@lU*Uj
zoil&ddWIv0%2)=yMyaU5;v)$qQp|5|Ux0H9t0m8R$4EuY+&ja}yBZKlx6UsnY&GL+
zwwp7%;OeB{DDhp|@n}hzCWAiR=9ui=uw2!DaDHU#Wq`+X+mP{u`qL+YeFAiB)CT!<
zbM6a3Z`-|4a%c1;)=xwZKD>+zh=+QGm#So$idfg-?A#)1A33|pb)VmVNFQ>a!Y1I<
z=jMibOP<k6Nu|!2^Gm-6M~z<N4dFs0fF(=n0}+`<>w!g8JT^LlC+U~I3A#36R{)WJ
zzf8U*N?bkUwgM)b?u49KU;r{g(SO0n&vZVX=DZIO-tvgXwjatPXR|HR@smuflqobe
zr!*0OEDNN&tr^54Cs%9J3A+Q)_f(yrgeu!aq#@^O1GPj4yQW&V>mXXn$qgSRH6#;I
zvf2#H7(h7d{UgT?z&Vk0b6n5H6D(bS?Dc%HjFAtatn@|AyheQh`@ZVM-jprVXUl?o
zrG_qrA|<`k#mLx=$lu^@-sHiI^raM)ThrUiPu-XT_7xiy9J0=Yl&z6Y>YpyKxuT(d
zi;7Uxaw<1LPY2fQ8ThirqIx{ld3EtI8otatJzZRtaqt8j{2Ma{wPZ&1x_)negG#kx
zL!ONPHq^RF%3F_&og>9XXhmo>bmyWlm2`7f`3%6Me|gXJr{%e>EOb1*L~GJ%eYOL0
zsw2|bu9MTI(_@J)X3p<7K(?bHwgKtp*Dkq}a@P=%`%&q*rsTTKR+E~VS$82H>iS`!
zeWh~zs9Oc;Q!_M`cV-VO)ms&R;&o0XHsBI#eo-6K;cz32?8msq7xL?<lnbTN#Y0fM
z#Q7zAJ_`gn@Yl;=4yQpmb5T!Mo_&XXbn@7)jr$^9=f#?4|I)(AaOe{rU(^nDeW4i>
zM{z)}#*K+Ob#A7m-W{~g?^6TOXDwvkvVJA6Y(zD{O_8#TOx||@_?9ex8N$XSeJG}s
zwgk0QvBjaZrGqHpH&EA;E|)z$cFNNJtw!PRf!ycxA@<xr30Sd;BMD!#UIyyUz2F*?
z4N+^Ujdfj$Yq-8<Ia}U<+J#mq?>Lqjh?htg$3tvsy@4`7Eh$c<)Qu=u*FI`@X(pJN
z0V}~xd2n1Az*ZBPP4eM?A9cG~Hqk}HveC^(6zf45K=)gccN213z(OU$(3fTc{7ADb
zCusVBTs+>QU|UnclLHL%XDsuLI}Rz-Nb^QPo&X&nY067iYR23zicO&^EEf^DGzJ)Y
zB0LXn9EB72i6BYX%yfp?<n<{rt@e<^c;0{>DVBcr8feXgU^;7m&=XKf5G1)Dwe3ju
zN}CT9P0)uB`)I0@P+=pOB%-f&2De<lsbwZ=+16^31Iu)DbMm^KTu?u*HZ#zH;f253
zs$75+lgDc8GCT34yAE?sBYdEq%eA!YHf2|69v&`UI8>KD@=b!kPSKCfQMEKqzm=zf
zqbmelpJO6?Fk`QO-lWANF-#LnrV=*Ewk?*qV;ih_T8ftGluPuolK4c*K-6%mQ=YA8
z^iVCTC6*lA)liIL<pYqlWw&xW-M=0$KS$q|E}ys9g~7cH3V#lWMST8p)|W_$5_TWn
z<5Ynsu?+rn5#Qrc%4)<)&O(A-O%@8YDzqwPfN_Q-tAq7_(m&@?Wd@2Bxt4$QD-|D8
zbPU-QJ<1AjzHB8EtI>WZpgHtBM=Vhf#w6GCM`3pPfesXSB_yHyDM@l6NS>w63@=Mn
z5Dk*>MOm+oO%O2Ei);#$!eZ~4j&w1kv(ieHV%MiCYg|VIBWX(t&X-~VVnIt<;A3|H
z1ySb3!0Q@+XVE}Fi2N3Dh>xB$@*xRo@ecQIW?k;v715|GW<VGf>vMZ54C7K<0T*a5
zvfWtNPIaopxVw!!?8)!H?gdYmVXphRo{#(Dw@%aoWX%}I;bp(2pzk*&+NbLc%!=xO
z`z@`!6W=Go9&dpfJChQ%0zo8o3fI#X1;m>wo$qyj_a|m-jv?l2<?g0_9>pNKixo7Z
zF_sD;3rb;bsK=tZVE*j2TQK<Fl}@d`LU<?eCXf60xmFZUv-6T||H=?NdL|&3Pngsb
zA%fkRTZwkCjydyXtuKf7cb1VLA~Xj<0~QA|r%uQ;pqE2}d=nt#R4buHN+WG|%Me+I
z?Ofk~=jY?@DoN28j^UFrp+Xj?Q5GeOGo}*@4mnaQ6P=h4&cjW-sDy1D3orwjk0y^d
z5i{oNx2lp}|6qi}@n8IwBbFp29ELoo@D01P^#@rCyO=1hz=QHruplQY<ZE^yU?@bX
z@@9X9y5CniKA}J@Wfe}N+P-CBp^(hWJ<ooB{v}kc)=KmuW`ZJ*Ry7%#)3)Wc{E<4z
zmix5%lWcn3hx&;_D--sMzQ0~xpG1UzC9FbUIz|Ozxre`Tt+6%oZIJP}ay6VxJ`OU^
zHptJ}){!gwBPGZAiF0gKDRGpn9yQ_M#t$YaC*xP(iO*Cy672c-ekAVY7^u}!A}iH@
z<Oa#BA@|354dd40`%$ma;Wr5R)Ll;I*H`*?hC&RHMP&gNb!dg1HX<60H^zvXg>M7+
z!61V-$Tk{5&q!h-ESHSWZ|3CzNQC@d%#b;Q`fgT7hhhPywyx&jqS%sPf)x3YG8Qfu
zF_At>=_7ZSm;k(cPv|8X&Er-(BH+A##(JKcMk{bGYS{V?x}_1Id?GZ}T`D92{}Jin
z;AnbtGP~)uuLp*sxW&%gpWK3pu%Te?ISs=__5dkGKH!N)I+7Xtyu2zx{1a9L6IG7~
z>orUTZZj)tbN~ZD{J%8~kJLGZTroG{6F-5Rh7;Sgt5Rrv0tY24hh9XQ<*}H5;(#m;
zNDnc+J=RCe4ZickmIpTX4=E`kQ)Vt7&98EDBGj!ae$PSWoPb@K>?nf!1hH@_ZCe>)
zp9-Ch#=iGp`NefZ*pwGZQZjnTkaI9bgX2dN5L~=*AQ2i@@yt1$s@Y8;L<5McJdFDJ
z^N4aqAsToE;tFH`aRcF=+rf5!4;WIVd6^W4`u43^M#UORTvcuTR9d$8>h4NCWH_Kj
zIAw+MBN|d5m3xwQoa3-AQc~g(c4R+23S65=HHs#T>-T}`82-J=NrzSYL~N{|l!}bw
zWymIbsFnh~-TT$>BYqygel=>#(J|9iv(J6oDgW2oAxz+NAm@mm0jH3EI_!M7q_T^<
zx<VO;ZjD>+-jT!k970jScH+Y^cqY^WbxOZS3BhNw2txQ$E6Bx1#pwgNv{9??S#*LU
zjS;0j1$aqUbW7R`q!-!EX9jxLN6lro4PXPHbaMHoWeOkkpFO1c;z35!dSMBZYT=jf
zOIZ~$;;Yz{d4#iR32q*L3cp~jH<{L$0mLiW!Zp`<3Yd<{K5-VFE_1Lbx^%Cq_0uU!
z9103B<_lUbBbXWJB9i@1ng;xh5&#0F7)EcQp(1KhxyIH%uyCIUVQ(>_N~KFz1is+q
zj@#4Vx&#QxZc}nQ55aj8l{%ILe=no*Tp~_cwlKZLqq4!&9s;L-7N?d43;&sn{rYo6
zh<(yQ28|!`(@JSArYz%E0777#h<4|ycGt_)ZSJ0%yWlwlN;D0YLHjaXQ?pAEZka#9
z5ISlJIemQ}epZ{>@cnx7-1uiK#A29V?NN2107M#$+Iwefm=0vCRS0o3lAXxYQf=U!
z-j}4>44(`nMllP2qCzLBPoR{HrB&%2e}=`4rP|sRK(J)Ay1_ItyZGB>CTZk?0d4j@
zFM;e1!`AZRBf7hbUen<ReLrr3%9QVRbP3Awz@uMfM5|S5yLL3I5Z@I-z>^0(dW9vW
zt67~zO?kfObjgNW7D^0D0(y*8^d+YY=Yo|AKYiDVqA^Z?d-j5+xk<l9`9jmus3Fz~
z+vb$)E2dmaUM_xZl9Wg<uss6mo%CyfoAgj(j2WIbg<-Il{8K-i=$5+8L$14^xz=V=
z1fVtux$51xb9yhmgkx(yvEOH$+B?YduA!XW8whGj>aSBep;yIAU$D`?Cjv!8l*BDA
z_!Qql*=d)5O0Xj!l4xlk5E+59@81njFSwLJ_0bVTpsDzMmEyqQHNL&1f-dy(8;|DM
zEzjRF0|2=4vzMh#v%E;%__c-xuI5nA<ZR8XYBcl7K*z{$4qvMKx;+9<K{p=v|0Y2G
znF+!jYzxkCbAZUM`x4-m==6BwBuL}MD~;{LzAK}D=oZG=2k%rN2gZcy63G3t7apaD
zu<U{sSg?qiC(QEZ4J#E4-Ile#z<hbrxCy4Kl}N(z`aFpJmwC);Q=^&fpn48h$5JJ0
z9onzdPJz~#l#>P`k%GKa+OXHVZM?y9Fh~HaN4sj4Z`{F&Y5R$ipj(-04D*z$EI#UX
zj1k3uv)3^xB)|~x%6|4H#VRaD{PiLzD?16(f-f9dcR2kl&JI}vdiGRbqN#sW9FzTu
z73<hxm-~Sf2wZlx(~8>Vc}tW8x`0K_)ItE~15Tj4Msq{@W93=-H7UCSf^&p<8!@m#
zZf4g3>xEdD93L|j)=`0rH1R?{#)$7vBVfmWhN#4ITNPo-`3RLX(jh}5a4?fptpome
z@~$q=@sOHpJMjXZUpN2&tHefL*FGITK)pAwFip$(rnS(5>X#2Q-uHL)%%j~xcRq#a
zN{bSEr^h>pu!^U$aBQ!Sk|5h3h}HerNy5~->;}9(q1meR{<xpdT&VPdaqJhoM|CNG
zPxspetG5QuC;>NBIj&JF&s*dN;v%vNI(Gyl_<l8+MRu7?!DFRMNp{xhuxNp=6g(7#
z>N;az2}je+aMMBJP6z>hZD%2?d_LKZVlU0Rhi67=4U+T^oc9QX32#IRWO4V(6fT+3
zYe-kN@3t~yN9pMXRZL!pHubjdDeBID(xQG$1tImTPro`vpav*Y%IS39b2q-T`#Ip0
z24D<WiE|#>fz<##KqKfo^1~fPSu(n=-(zHOz_~!Qji3CeI*4%p)PNzFODP;y2HIzU
zF`ejEyiG8}oGorH1XjdAV<CA=yISMY6)RDg0oLh^BMK{n^338oR<G?12T0R@0dwZU
zJ_pW@GU3vyc-6_#_$(U}xzP;8I5}{9=<s{3Gi|!$YRlw8xPZs=Nkwwd4-!W|{gK%&
zx-Z&*d0?5oPK*B6U`JxMwdO$BMj0>jR_W+|XuR&d{-%16U(uvny<jo_jxkD;aHaI>
zyWS2pV+9JT<UsQ@PI;JxyGc`jdVE#UT{V&9U!dvxT%+zVK#v?i8;Imr1!-Uq6aa{S
zPLeh)5rF^W00%$-5I40ov$Qc))qnx8yMa?CR)tf321Ef!RoAuO>O}N?ts_9$qXMgZ
zu64iW0P}(#?U1|?UBGgSDwKzoDr}xoDil)Mx$-@;zr{@YSR?VZ!1=p>;=qA3lf%i}
z+{qqlFh0{Uv!kbz!N<Q+*Z^&NadcIUf=b=NIp$WS?doGWw5o^IaQyNW(=h7a`~Lc}
zf4F_>{_FA8oM8LsL+R_<6w9sn^eSC{t@%)<*|fggWYhoCALav=zYW}6)2LmA?HyYB
z=aG7oCN0_&37xAbmm-IMqHT+l_;Cj^`>kZ@cHG4-)Dvx_RB!^C+;#KbDBrakZ8Hd+
zdwqq!HjKaQxe5KMrre_r$NQubJxz6^zDs=NOo;+}i!yj*#oFc%g0G%<zJ7gK?`3n~
z;MH|@dj;o4-33xs%e=4qq+JiePF>zOx2>*i2r0LF&nB!npEPxULD~XkwhP^KJ8vs!
z4J|r*v}t#Voz{2*KkQ5mTGcQYr|yJRjWsEgEPnd#_2iZy7x~odfmWw({-?tC%QR+`
zfj09x5gcrpMdQR-m?N0z5$0;5TsH9%evDsQi9t6-NXwMviyqhT<EHi(SdP=+pHq6Z
z&rn*w`Ifz(<%1}HJ22|_zc0hYN#b4W%7d|!d_F&4fAe@i*1&htq%!tg>ywuqob$n?
z(^E=B%|UvJVeJ2$v5x(Wwz<PA?-hk!d-$q8#L?9FV40C6A)~?rJKgC!(Vw~tO{HFB
z!>r1dPSGSIn~SCHA=+Q!F_@JUFbPD+!xUCu=xMnJ4RjcP7X^hc4c<Do?dL{#r$t{5
zQ)P}b&vTu}9X@0Q;6lThDqEy%Y#Ar%c~p#w23HmXC6VTDp^|KiooldE4UcqrW5+i1
z{Y)&}s?gI^_B!mmaKkgtVR8fzE>Dev4L<}OZx!wU^y%*2u~o%@G?f!0?|Lj!B2Lb6
z`Me8!t-Qp4G9Wf6l`~ls4@@5C58%K4wO*g0YdD4Njm05u>MR@+d>q^^ix)H>#ZCof
zdc>0!t>pqwvP4)tlkA7pVFCZ4>Bd5Y<<~HCG$i4sQ0>G!Y#tFrmKAU{fPl3(r_*|;
zO-K;nK_JH)ew(V?q4zy7)4vdwW_{a#+-pN5&A9-7A&mSX?bW(Vj`0vaHZO1Y-HiqG
zXj#h%TD8^%p$!1pO@Pa#tfqt7)&Y6D=@nZB?d2_{D%Y>`pj=_Ei%!)*t*TnJ{?gx7
zQLjrozA$s{+0T~MJQ*f4<%4KVXj~c|chwju%J_S-+$lFsWWw_l&5SSWw%LZXXRBE;
z-}|M1TSU@03-;U~cc_K6+8E5{ZP?hO5@oT#&J?l@m{om`D;iVaY=kYkle$}T4eHz-
zX8;hbHRk~)Iqh6{v~t-asGAu7_Z^+3BS4|b1X*((i32u&)rfkzUM;HU>%#B~GD#%_
z_f8xW&>ioa^Hs}I_;|6a)*WS3s7<ZPBanzP2EO<u@f4zf_yAugN?&TV@s8XtsTG64
zMqzfJ?=>P2W1j4~TINPE0~ImACbxVV<P5WzfJZb102V;$zq8-87C;36K0v|0v-R}O
z3$@4ra5Mz~JV3+0xBa>XY!A1jL;=t+1pppE;lH;u`vzwZ1ppyH-oLk-3I_oOe~`mf
ztztH~_ZW;dFvX|LJGdcPNC%O7`*ZpLFycn?p769!t4Z?%HG^z!E<Nl&FP0V(#*Z4j
z3k-f#o?7YjhHvI#ci;oJI!Dg{>V~39J#bOX@Xi1RNSHmZV||6lB%kkWCrwoOty0BP
zn5<C-3|#ERfY@$!+;iLWz8Jd{e`6e43YNHF$J9T=#|Lxy`{62@RfS9|7P%x{Zie9Q
zHug5KH}<&qunpIzGVHf5qlvQSf=6@#rBDGq`5#k>6Bv|%8w-=90q>7LL0y{AiJ+mY
zsDuj-HaS?KvAJ%<Exp&a*WS{3e~Dc~N}>T$Du5~(r&|z3!&r{j?tQy&e}L8V>QhZB
z)RrgV!4eep9Xsu7@A~6D&eJKah3cm=9W7|43FWck!rR<zIQGMtd`+U2(vLQvs-=}!
zDP;U1$QB3K?sh)b-w_X(WVxK`nKfkIY^0E^tm`5ETXIJIg}-bADuY5}s)nq>-c%~&
zi0q)2DhU_4;FEuDP4Ql*f6P)SLQc;FB0>=e1TekH7qJ76#kH@vwj^^L;N_JkOtH@*
zz*R|5P&gJ{hhb&5{EiMGbBbKFPfI0a^%XBLCaAM+q?@onb|%)jZ%7ycm}iYx^O{w6
zq*-6fQ4l_7^8;;#z_!;S_vY8c-c_4ZXE3!sbdVX=MSnDJBWbxBe}=#Sbc+HHd{-z~
z$0O3Km?<YuE+k@2$p*sW*B0jM$Qy5P!j%%5I;iR)kd=%&Bx2+~#>abGN#}F07T*D(
zR^^$FX_rMoLr*nibkYT;p)45zH?RN@$9FvOe{Jzw&uQ|Et>Vl4$r~_+Q5^9gP!T{k
z)IlRv)Pg}fj5W)%e>%Fj4DDGxHAGVg;#88#VI$O|YnB(eu^S)H5pM(0MJ)45%^Xu0
zho}|=8-nB57B(m5J@68>vPzOmS5omsQB>^yl^&(P;nKf{WanZ*_QDr<DiW|s7}DXC
zNR|Od{?@tdN$r08;goYtQ&6=u@WjzgAan|VsMYv@zTAPaf3UKS_>rus%YF!?Z5y<X
zj<T++C54E%7CfHzur|OjsIHn=no5}F^4e=Ng!HnIOM1yW@OIkQ_QLd&W%0(9GOxrk
z7}`YDWQkhF?$*7u?RDRdEx#iberr*Q#1ACwtt5(U>H`6!0d2@CP5@v)pTGCM_rf`)
zDMXeD{POW?It`UfwtxIu#^=*(axY=>AnLNI9<eFl@>)3>V;oVbmMB9I0-hKJ{Wrb0
zCks=_<tU<poH<vFxs{2MNc3-ecHo<X+j0gQPgC$RwNz^i;wDx{D3WVs8|iCa-lKiR
z$sF>rl=W~*WTc(eS>i$!mB2=m`H42;xF6~;6BU#+bn(WEnSaYuNYauN*qFB@v9nkY
zdwUOTBUqHvR>-hbdC~pltqDy;`n#V^!N15GZ@wZWs;DrutOA8?0FmgqxjWy|Y_<fB
z$K{16B{RJ(EYmD8&PIlCNkk^rbKc{76WH8Z*9VcAT4#c~K>?*j(CF%QFCf_4i`v)i
zYmM>BiJls@CV!oEBFc<W0V{R%+?!u}l1|n(817w_R?}CmR&-ReNCM8Tj~)3_we<}k
z>~`FH-1dtu%qr@|4J1^d=O9TVG16==pjdCXu_Ix&;^N@AqiUKxbcL!UlS0lGK@>4Y
z2mls47P%HAcQ*po!qLr8V_Gb(iRaV=I!U<({3X~Cuz%$HAAOos#_35;T)GsDqQs;Y
zb|ex>7bnxUk}Z4u?b>|4v(8ngooY+61!#|@K~u9e#*%Ib2e}vDU<pemwKWwLF*&UQ
zC@JKS5|OA4W@a`EU`Mzh`xB1PQC7ViwGP4Z;T_AE$x;ZgKtZ_m++S;|gK>KvR<pqK
z(@m5_J%33}E6kfI<CD6ZTwHtdSR5%ql+;!Hbkvn;QYq0s>sfzW3lh49#09t^Y&SZM
zunn_Jhc4<!1R3RFdOCV?^X2|vf*B+XRZ);Kmr$+2xUn~1#BkQ8jugooBFd;~vUog^
z3RA%vq8N+%w5kE1l_7mS?05$f6_F({j+&uW6@QsrBxgttpp8}_l0hsreB2F*0a@xP
zelj^JURfQf&@h@cV(e}WfxmJH2G`!_0m*WVpFE?EinbZ0fP^wrQ`9si-aXYoT{loY
zB}KvB;9?f9p*jM-icGeCj>)H`62~$|)CyHt5C9+oSeq!{p7+G*lB+LQN0`qu!lACk
z{C|cZ6038)&meX`Q;Jtrc%PPKHPO*!pA((twFX&Zqsu{UL~3nBui+Q&E<HPufUea2
znEWlwGg|z@oM!a&rqW48OHmn){{YIO>2(3D2KEE#Y*D;Y=)VZOC4rNo8>*A2WJQjO
zNGj2zN`h3%C-p0;lqGwMF}3|zoTusEw0{?g^(JiBF<y$wUNmUw)~H7>ly1Rvr>GSJ
zQS(af)*Rawyo<p8pXs3EsI#ehmU`J%zI3Q>Eo885Do_F(79&;K=ITEVtOuEL{-f%=
ziiBj@uS(>-K}8IX)bQ5Q`Ij2OGDl*PfZ7~yV`4ZUTE)};JaryS@k!FXF_&d9Q-68W
zRMXScIk^#)K_Qg4-LIf9wx$C}br!d+GX9t7yqh_jGU=Y0$*F!f)8;N;3~r(MVWb@^
zxdPpXVRMIB&PS0wMO6&BE>RSQ4HC4FEUPO<t`6Ew+tfh5z_IthdDkQRXlK-N)3tv}
z($!B4^D|3JQdi6>Lk0=z2nid#*?&&t4XulWA8Smc=Cri=w@Fk>=POG&hPh{TEXo2{
zD&2QBYl{+g*kaR}_*LP?Z5mQ$JsU#<O3|R2fHfPF9h58iB>@@<OOSSRb8>M^&A!Q9
zD?Lf8={}>ID|TTlan(YVFvwA?wj`^l^&QI+EIF{iS~8EcuZC3_beXJ~mVaZH&nv%_
zC8g6Fg;A)3<{d;52HH)J%GgzU{{W-9u8DGPp6Xbt>m!xsWw+yvN8XNot-rSizA`NH
z?Ee7OSu}Ca4ndQ3Ju#{W;pr<pk1aJZDpo~Kz;CGjn1GKf>VAOFa_r+c=^D(U8Y-Ct
za=c3L%+r#LlFX&FCWn&WmVexk2za8?Q1z7cbq4N@BQ#P;cb(Xgs^a9Eom-LVQg3oG
zlICe8CU;paD0!hqQ!xw*hi$+n+>6`tINdy8qk8Jxj<_Uj*@j-BgSD08HjG_`%9SJw
z@^qhkVjWENwS8Y8j*b&iG@lHM3?RoL3J?@hEqig^_B$K}hbp0;;D0MCrFEsJifAKv
zRG&7YNfo_F4%$JDfnl&V+qN!QRM}MhFP5!qM>O%&+IZoFq{=kLQHv96H{Jji_9)jo
zi&7RGx{{i*lnlb7HK&<ijplKtG*kZok4WO%osF-4aU;_?r8Kp&#Z#<=m01LP`VkG3
z3vcQlO}{{%n>(e*>VGG%>57>tDyZd}YDoi^@=mKO4O@$GbFgmv3|HpLdSaV1%U3m|
znbsP(C9Q>K($YbxLj!(G7f+>+*Alh#*{xiajpWM8L&7AE0i#8OMjLBz!<+23Hn12a
zE=v?jDQcEBmZM!Jm54D?>J)m9HgbpB!)ynza!NXSDtdgIFMo;}T52^z9Xu=yXa=IO
zs9OPh4aVN9;W#L2D=OfE@kHiR8@HOtvlayE2hujN2Xo07h3RU{(K=Ttl3P>EX(D=v
znS+lw#si}cLB89Aw#T@{-9JxNS5ixxl#s5KLm^OOSqa!kTnqO*1JBOn5vQb@s(GbZ
zki<)CDmB@NH-7+)$gy3?-))a(8Ko^&RVKL{G?NBZEgihYZcWD<c0Vn*H<+cWtID(4
z1yG3=p@X=2LKaW~)OWS5ZSP~a0>a`dzPI>=Pw_%xm(<E5i9#yJZ=WN%+W!E)0lbv)
zJS|mFvQfM-uadO(l*p{gw`L9L+fh7vtTw}}&XNz}5`R%pQ;?0P;pLJ1$ryq)G30Y&
zuq|s|{9zL=UXuj>0K|%}x_1k%z-nWA-s0lq?iBa13iBj1xs*^!@JQhTOHr%QtJ1!b
z2m=1rz5T6#aaEDmmZpa`S)iU#qMkW`V6mU^5`93}*aNoK8<B#pWg1WA(@6v`eclE>
zy~+FCTYoKs51>#iZfvbAl2zxCL7RDzQ<by2L%Q0)rmoh~yxWdO;55(bsfJ?pSzb)(
zO-=++eb_4@Ky0sIH#c%_PRa$W1_DYt+HAI{K=sQ~BeF~)a@rk9^{LV>4v}Jay^h;i
zz|N_JjXNdcx0n}MR*aBM_uo~)xi_)5J>lV!I)A!Xgix`Psa=*y6rI>80Z10w#1r1%
zToNkdPkMP|jv1p*EmAVg>MToNxjTW${<sMW)zy{LW)VEh^9?$Jf{wtjV%(71>A2q3
z+k7`jY2}6|2|P+v=}1sGQK;A(i-Ynz@q}icsF<d!rE~z50I^kKL9=c*V|yJU{{V--
z6MtEESyp8ft3^p&3^4*-Q{ka@8jbx#$9wPkTLFFkKGVXKa={!iq>nKHNRwWxSdq`p
z`(w0O<uHPxYS<yJsCYp^8M9NcBWrPc+kazjaRRkMHfdpJSy9A#);(JE@1%lv1dazh
z`wS;6GE~%>YF35_<UbJe4PSP)xs2)lQh$3|%VBaYYy|1grKwtUbczv68wk}@snv2c
zu;A~#$j2!&>S|czs*oC*x3kARGdJ9mV94picHfTUbBKnjN_iScWjezE%p(P4vXB+E
z9xt_pxhL3Ryyb+hl*h}bjr{3lh3x30gQHL)_G7W_#qWTYsHjTsik4ZAn8?H!)PDlT
z_q~^!cO3r!k}}j#2T7a>7tC2?kODN_z`IAmu+?C`{PBqRo#`a_uPnN;D*FTIJdh9W
zK_5M>gHy<#3mI}0JNafe8d;EWX7@i&+iU~LpgC;uyo$9{lM+`^7i$kgTJ{H>k2WWJ
zj7M6QX*JVB<;c>Q9#tmH8p?F7z<(bhduhG);&P33vsFQQMR=YqNUJJ_AfG~;X$N3;
zKKJd2a9SO7^HTcsuAOSaNaOYcZ9##y+lycnA?y6OsUVZi3;zI+C{lq**mfgeby#jK
z^TP(G^9!v@8P=TvV<10yzO6y5EH(n$o7nM*It5su6$vYZO(IYjmA3nL9Df3M`F%+|
zQPBX<ypc4nSp$U|kz=G0!3Wsiw)h0}<u6Z0^))qdlz%VG^GPT#tg^<wt`)9+e>@hF
zT(WT=@c#gdSYT~lW3YEZpb~Axfz`Kbf=$Npl{AymLR3|0a5R{-mSs{xvDCXQ!3(+L
zZM&lytt}-)RA`QfG=OG0u7B1AfKGNJb7AgHt$>-K%uZh^H8Il^W%E_)n8;L@Q|2wN
z>m$-dy-F>|4zlV<^Xi(4xidOyC~4_Pe?Lj{ms20RQGJOg*xQcSf-*3nHToK$25809
zym5hjC^dc--~rz4ZcYVc^)$X>#_dcefFXrl-a@Jvj`rn<+kx%Q0e{P))E5$0O0mL>
z4ZiJ|Tpc7SvtMDotZl#t4O1r+{#=meGE6HL?*hh41}5q(Y`4EV><<SG=-xC)YTidP
zJY~dMkK(rq%A<P_EPeaoRj7_#bn(&Zm5PbU4P_dKuTr1HcLV`pbG7Y&nxoBTR;H-S
zrnOYkXjdf|uBI2#9)H90_c&=R@=YSuM^!9sEUcnwb=fRUuW|q+Sduod?S}Eu)n*YU
zQAHv=DRhoFmi8dA4{`;^r)$`ojxW1fiiLQ8hg8N+tp-6WuBD6uNR4a&1U>ycX#{+M
zWy}#Y(3&9xA+Ewn8JEfi;n<)y)xCn)jt<<dm*o)G)|x10rGKO{6=a&HQnY;R0^a_j
z2FrUXCc_V7tjc2P_~Me5d1QeK4=Ni6eI=BJBk@=f+#4TkKtVKuo6dm6A_aTJ6lJ7=
ze+~7XH$2|=Huk_tWz^ACQ~X4gtRz_shCQ`L#`=S712Dbz8coMMVYY1zJQKl|LsK$I
zCDuf!b$vRUn|}`ET##&c+k7G@hs8+LRZ+|;qDd*`Af}a9A=IYiG1I=2Zg?94LbAyX
z3{`ay#3plQ46AY&uBQ+4wZ*`&B-n3&SW1-ABs4S2B<T85Q9Aw&KzA3mkUc*zeeoe0
zy%e!hNgxI{a}v{$$x?1u*mf7+Yu|fezFe%5Q_|(_3x8IMcav01iX%q7u0s)i#@5qv
zK3GDUN~AKlWsqFJ<}~vJWE*O@+zqvDvA7!pfQqi`^^Y^mAwyYNQ%NxUsw!rwF~-HT
zG{Q9^HnQrlV{u|JBb9Y+Kk*NYUY+U}r;b4*f7nQc>@_O0uu!K+)LViS9`?lcUy#=t
zXOSpSz<<z<M$8L<0VOn>DDS<t7WoIcZec|Xb+nZ^L!^enEhku0eNGDZ=GX19?R*7m
zFsti%;F5YSia!aMX{m;6$1;j1FJcdw5^8;cHtsHc$j2%CK<W;MN<mT5QRbC7olv8v
zT(+fjnF=!i!N4i(SlIv{J7Ugeqo}88>K<7qNPonU0HqW@-~nrPKIGV&A82~OkyJ%l
zEg#<oF7{RdL9kUS1;E_y2{$BzumB;tx2ilqsLrYc{uEUSBZUFcno|gY2H`=}E=}$D
zN%`U>XYp4zvQzZ-OOb0Nipt8NJdwq?U?YIC>gK?(xv;)846XA@yuL@Qsi(Z3i_QQ`
z=6{tJuwimWlYgEBd4(NiF%-0sPU@=CvWV=>sEeo<*bCm)w%5J_g;~)3Q<*ND{{XME
zKjE%K-v>z)Z~$F&lFmk}-{r9<<%vwACF$Our#XI2Lmf1A5$Kkn#u{ls7T!kOn}Nx&
z3P>CbO~FZ9RU}en^z}N*fIOxIvnw6K*ne0Bv9RX**qi_xK;*wYq@z}#u92mvnNOI|
zEQrMf>NK$dY)Ao4?O|(z4X_c_nUmDo8krMVMp<lQ(<-gNvw`Yun(TXlZH6NzlP+?!
zt4}nrynsdt&`SDr-+S%}B#o{Bw`@L&nwqj=k-K^Dl!*XTD)u)80Re@D@3$QL;eXFi
z(8)m@ukf_g${CZ%&?5v67jiB(_u~9t0M{~_qMn#ZMJbvmkTbA>bfesW0RRGaBmvK>
zZLlVpb2Rwas1lY1aFfiMh<O+m(g@T^A$Z@^ci3Z@Mqf>y)VIdVJcdyt6BrW^7Wy=S
zxLvGwumbno3;`t*G(Kv}EE?F$5Py&_;WqOYHMebnJ<b7Eomx4?MAXYtWO3yoI)=V(
zThnd;?|T95gJl_%xn+6G3~^M5%S99|_)}7l3vbx=<By&tbDorJmKtouV3wANVrePb
zpzCs2i)sM#Ykc;x#4cUuGz}$6vWU=r6am~3bbT&&yAyM&=G=Dx;wPw@V}D9yhFN5n
zLsO-<EZpkce>ohEE!zz<sx0c7L;O-DVLI!bS$x26&434P1^)mrcfl5@QG_w+O0%FV
zf<}U*Z?|)E?lwPBENa%iN6fQXQle<D4v845gRvw5Yp-qZwf?vdKlqoCWz7(WR48dA
zk|5GXRby5Q<rAlIV{ym7&wn0z@fH~1>T23*I)gN5byB<v8a66@aw}T)BrybjeA^z6
z@!=+yGKgf=!jdycYUhnXk)=?@7!r0C2T&)t81tuCEF-Dmnh9E>B&S&9Wm1<GZ86(p
zeLE9we=GzxM_DvW@b6nK=6y_pLIaQuovv(14E8quW4`;#Wx@(P%70aMQ5Tq`yKD%!
zH#Y6L*l&jEl;lt4R=sqwM-*+Ox8o0{>ZL7a1pLQ5fxacA%yMkUA$+=zExl=wOlsM#
z#t^o!907mn^T19;n`czHm%v2t6$NLM(JXAiCmR7EQ@zOA$JqUGDVk<+Jp<%(Wpb^4
zRe6l6+F#C$Ye8}iwtwV%3kzW^j#_HxjAvD{Mq-3IJV~lW(AttQebsKrru~Whrt7I-
ztj>(tjbFpkRG21`q##CMS4%dfUC9?0Cw+<e2Ib-He0l!>Oi@?B8h=4aTo4^pX|WrN
zwTM5~`{LIpQ&&;sbr~E%#UVyv7R9tTq__Biur@uvF^s3etbfLzCyy}cQ(XmgH0r@5
zj9d{Hq_6|=T~%7(@29x9#r~pNT9Z#JOFFlfS%i^-$4CalfTH%>ZHFhYB|FVm991<*
zB<T`Hh-Ci&$Rg9fr27>c@SuFaEn72+s-Vpp!x4;xVQ;+E_(`zaQ3I20$={4Qt2EHf
zG*QZ`?g~GaUw=bfek*<}5qtLS^22zcdVMspO(`{D1cy?AF7_ZOCyNpfCcyhxIWf!Q
zrj|;2(FCf%tW5guNEcER0ewJ{O}!usZGt(eN#mZmMC_1>ys>EXsWx3isq~xebH$0a
z*m;=c-bpbu5v(rl6mn}K+5jLF2ILD|F}OEA#>m&p6@L_^q7gGBq)x0a60M4-t5FA6
zsPouqYw!{BH6+s(wN*7c-tx^)sIRE#DeN|2NjvT>w$E#0y<1a?iK&O2^Au?8ps_xX
z3uz5)M*jede%2T+V^L1?QO8d+Bl#B#9E!TxK`KiA8*bYHb78g4>r_Ei4S&Q_O|1;i
zD@5SAh<}Z)Y;_BO)*GM1M>r0Onz?4CfW~5sJc1UISBS<Kv9l0twg7r>+!9Ujr8R9_
z5yPf9S*fRrchgE6@ODNwu(`I^x!<+Ilu*`IE@hqOjZ;uU0}_y}YKdEi(Xcvg>9FH_
za9K3<0-|q()fR#yK`f_SPo#x9hQoE(aegg;cz=Q%*EFbqh@NGsZ-{qQW^(p7BVlV0
zH@E?|u^3Y@k}RsA(d6kZMLJ*ae6gw8<S}h6YmzVYBek&6%`DY4P(xKbvPPeJ%PPnU
z*{`IF9f9EPNNZb+M$D8mO!6&!aNgHfNX3qk#{U4>z5_{)79m^81Ow&9paoz+{%!4k
zJAYsNl!k((*eWVW>Z)EL$Wq9@q6XlX7X!FA?Rlnxw<AhXoQWWbB1!y`LludQ)bC(6
zJTb%_PIXgM8d$u>f+-J_s)3b?J%oc$AIeVT6VAf`+O?_>y(H9{BX=;lw)^j3dlA2D
zZH3uPXq7KC1W0rwSM_QH02x=1Hq*xU`G4BMmFBd0)i{qij+yFWlTmcLG=vKR=Yw)>
z={tVq?qQT>`Rz3&9W-?{=ADc*)UPtIiFDfN3l0UWNU$Id1_GrO4P2AWrluI%$!dl&
z$`a9rBXDfK`+V?GJv{61N?D#4CJ@LAYTZm?*SOpfe{THYZ1J+18KQ$Ncbc-D*ndx=
zG3B66IQnu8w>#mL6U&-Z&syyrJhQV+qfVv(-&0%*T%O<nFSee+AyAM~Q^JWXkX;0l
zLL4=e-*vwi9CCQ>aN4*{4Bm9K^#DWaNaQz>MX$a7TkYF>TG)<`cd4D1Ln(w0L5MdS
zk~y&a{&(8geNgpnJkkesmj0{XN`IbgLEhUR(BKt<vJ{dSo++A?s+pooY0-0XyLw5m
zu_W)w?Sj;-vQ!u%pUS6bmY@caW@y7VC9Uagmuq(ed_^TRiT?nWMMO5xHz$+Zz5Vy#
zUe?2!n(Y;F!BHDC+tGOeV=HM2bzA{!FXi;WIZ9DMNM>Y+rnNGJmc`JUn19@GbsKjK
z2frbuQ&iE=Gm{iB*P#LwUoeAqLKju->)`SMHtmG9)JsZQYKnJ{NL$BqX$uY5Y+3mR
z*W_Vpp+{8ZqKwNAl#!U`xYK3Q4#L3tSb#{sr)}&3-bX`QoYT#bt!pJ69G?*Fw1^L-
zRDcG?_q~bSjj&gst0JQ=hJS`Cw^%PDgdyW7!HEaDfEL_<Z|OG|Dn(tEE7m0O8hK(z
zWL3U}1Nk+7Ho)!0tTqD;R!q{KI*ie0j7r*CNp)>1Tlqm7?0xKY9gjE!GWv;f=oHUE
z9aJw$VTM<WU{0H`Al~D99(mi?nIvdD^roa{rD-(JX9ZW)y9*7QfPXJ;aPKCr%_GaJ
z1XYzH9ZNfamh(Uw6?FsY0ZZLU3%|Y@!<j4@eM}Knl$q2CY2i>2(^bi1dw@G?w_$78
zU=qzU85VO%mpholysR{k&df=-;^X{46R<n-FzS}BkuoWA`dT!F>Z&g!lIgl=4Q@fU
z=G$y|vBP?XYMi-i-hY8fK=8t1L=p`^6kRMeAH%T(jmZ|??P4-VOYugHDpH~gGC3wh
zlw4~A(yT3elXdlP#sS<fDygcXnGzV)g@L!PN8-7<mIB1_ZHIA#C4a=P!&FN#qSTC$
zQ~m~FaT+G8?7E)BgSGJ?)Acz^<<i4ZJ3ubeK@^e+B5`o82!FSv+SVKY06-b_3Q$8>
zhbE0wGM_XwB^7SKfQHsOtb6i5FcVe1DHczaE_0hD*`R5nj*wksk$|$gvDjM5PVKqp
z5fxPw@#RK}ogThOX}r@SLPe09-r$qBqUUc_?R(z?I?dW>Bc-0Cx0eiTwD;1HGifDM
zo=Ce3mazbnZhr$Tx^Z1Jl=0=yBRr~;qAl22!6cjAZEJ9NC))uX=DWnxK+*{0Ljh?8
zwF00sT;F}cAX?WPbBUUIMhT{>MDtLXgi*VBp}@Io{vTC_{{Z_9j`%HBM6@(%nnzHT
zQq;)dV2V9Y8B_R(?{n|Xt+pa^neypl_$ca(lC-uUf`6^$as!KMQpe?e!P{ei(K4>7
zj#?=xp+Lqs)m-_SfFpBjZ}RST#6>*y6?xR=hI)!;KBkQp(jCF|5v1<OPrZfB`7x9;
z<?kG#HK?baR8a)PmP&A;n%Y1cxkXjI{{RuVI~*1+sGFl+UZ9|(e8LKFlTQ^h24fSc
zC9R|lN`Fd{rq{Cz-*6LE)U~1psY-n-Lm)B%DKar&NU<jFE~J+1r;%(Hhcs-uj(SYe
zH#5qNu*T@*8Cu1hKN|z6?rpzgaSr*`X`7^(gH*>HHH@>&%7aTO04daadjZD8T<wJ@
zGQm|{4poy!BRs6JGR{%M$PS_uozxMqBpu6q@P7!+6-708NS3AALuovlU5@8cvDsU5
zU^@-DHgiCgbIA_1cw&W1I4W7uL9(a-cQzY<MalZFMVCZs{{V(l#I<c2G}ARmSk(Yx
zEgM{20*j6BaxLEQ)O8cSYb7d3%4B5nr;arvLY1-P5w*?7Ci|Xn7D#I|HPv5}3E-9x
zOn*%r%5)be(!dQjum<N&+yllTtFvs%l9}Ypp0*`qDutHe&Caq1AY8Gr_xa+*mLSP1
z-KLT^ggHXdfORt1Fx%B&TWi~6+i)WKj;?@8%S=SV^Rr69?Hd7g3v*%kx$S=Uz%!a?
zDV`dvrnw}YRg1*wv1J6c_OT=zd*6&NQ-7Eo);1G1wkqF&G_Im_*fsYe{@;j!Vh$mo
zc%rCQp`3ZXQiq7ibz6c?o0~Ax+m1e1nx@U8ho@??$S57_Dof4hr=^ry?ybMf*k0t{
zZTAM*ZDg5Du!>Pm&Q?fPH;J`t{2^>V5w@-P-*MX!+NU;=SIDT2V2iB_lG+aj)_+lQ
z4*>g)*nVr}po*O-e59FNOzRO<A)4S`Ky?PO0>JI^JC@VdMC_pnXMx@19(CE>y)M=Z
zunb$B?X~a@YHPFX%~l$xtf)^b%qb+=hn5r7t8t)e-oyiAalpiYpz`Ata_E!@eQ2|4
zRA8(S+>J)Wf^W#+8=ORD6zq|$bbnr5N0>lqeMD+o>5cV(3AN3=h~M2vYou0qnquux
z0Moe)%WE?&$OPQqfGxF#*awo<JZ%(IbupTTZ--K^x(+m`x2I8NJRW`SaQ8pV@>2p*
zW%2ouXpTp!4!W9WL-%0Y@QbiE?Z)=RT-7zywCOD(xrnd{HOWxQgoGpzFMqY{Z*Knh
zLP^o%iYbhe2Z*|$QVB)4bsz!*Sz6m#*ZEj6j~yx{q0J$JIjJO;ww4$amI0#CBQ~;^
zI#sWv8~*?+9j%Omt?I^m@V~?8<VtM3h%0j_V!UO57b3tDrAQikfVkIw(3DjVm{JFd
zIGf<nBK%BHMqc_`^E>t@wtvGIhGm*&SzbX^l+aYUqZ6}L(!}ll6*{g#P`0r5C3xcj
zUd^J4I(VxEYfQ#9l4X+O?g(2c@3Fr80C4M~vs$_bt;px|BB!n0hK)C6{5B+B!&d`i
zwfGoMF{r4`D&DS=osA`1D<F=YVnwjA(x9Cx4+C?tJ-2Ztb5zy8!GH4Hww?%8qMBz8
z?9w`bzF;hMU5kQ3o$h;JEg2<6Ju22yH6*f0Q4x(TWq8$ndbR|Sq>@xybzS=#eRob}
z^W>*6lCaUWGx;$R!~WyV2ZZDgTCKG;inCna!kw3AHF*tmnVF{&JyVDonZeK<NMZm2
z!-775e9D=YXBA$kYJa70vN;i*MT|n`&8O#e<df<8U^H%#rvCt?Gny>gnyO0gjLXwB
zo{+MML|_=1RqtytjgS+%ls8jt_WZjz%9@6sFw2%Yw2+62MK{rX_#MFIZVm5qwmjoU
zn*RU;%<`&f>Zsb2G^<v)eNucxv7*SgTYw9q5--R$HwPB4{(r7@g$8Ca47V@IRw{YK
zdaIce&6{7ThM3(5V{@=kZOyNM*Xrn~S_q<e4O-PT%tb7*LWF-XVYaJ-xb43CG^?bm
zqM|`IVVK6zODS54df5@8X*2+qc3{Y%NK~@{slUaF7gc7SJwcSgT)8Ge1IWr|fn!r6
z%CZQ62n!1qb$?LKP}=0_;6CZzzVY6>C#uNe>DqcanrUh#r_7>;mYlR;F);=%zGP*F
zm2IvFBH#c*e>2MbRhmj_tgA4m%_!z^GtlO+tP(XS79t0Z&mpkAgk+llEqh~!?GNEM
zM&wzAE=x#rOvY$TQ&wgasRo$>>5*ySVpy%tt6zI^cYg~*>bh>e>VBVIpz&L)VauYX
zUooJegH=kYkd=qcb|75X`a$~Jd|<SU`L=OY6UCWHM+%=ZpUi3I5syM6uv=d2qgKRz
z5KYCy)7~HTwHH&P=haeW^`8$?$OSsQEQDzSMvv8_YaN(uc*TaYcc`eIIU{)<rNYRi
z)tXs3+<%)`dX2ja+qLnX_)E~$y#W?&Ra;3-4K8~mGe=7dhE=DCYa%z*xR4hEW447i
zIMu4FooZ;!3dFTEfn`|HRzPJ!0d{x2i6FTLVaOfu3~6G`vei!}MJqcrN+X$Nvsl@z
zyKq1T<GrtIVdRv}mr3#PDQJjaG2{$&I@<PY3x9$}`x|fw0{DMR(`!CoU0pVFOGhA$
zJeZ{_s!pKA0bT;Kjk{?U;9?e%29PyfN*SIV_3LCP*158e#0|>Z<=Ab2QUhC1C$7s@
zqB9vyZk~ws7Geu(HXGYxwTEtSo8@Aq$*-NNWQ-~XT1Fv8kaYTX*pv0=z9ecYt177~
zqkqf=Wkhw1v!(w42$5qtLs;n+(swtoUGU<Df`+P`PbC6WmM7Nw2j~eT4)!)X@qoEk
zSBRrZOwc_{52YS7oPrbp0xPw^-rLxLxV9KldBG|ur;$=A;VTS7TPx|d*6gGU`jqfK
zxGeP5N>-Y_nY_wHNOZFUrN-lNxEuEN8-MIDSYU)zQ<)1fgvOWnhESr$>HxXZameD>
z25=~%rI~7}-bRg=%MsX(WAP1It#6nH_WbcRO1ZU7I?W`tX;QK@G6<jUni8BXh4u_@
z>h3H(m8xoZDkE|}?HzQ6XK3As*jm8)Ndn#NaeO+;sf1BV=nBmsB#9c)NQ?j*5r1_9
zat}5<dxc;vZ4DDfo@9)!G^nxsAnCZc)OZ#<aee^9i7N|LCUHASRZ}Fg1`;E?4X({_
zFMde1x3K_ig=*_0r;Nl()wZW*aCQJ2SxNWY+>c^A-8geu<DqItsZ@H2b@>30u!2D(
z>`U2iaeLm@z#<V%Th-<22nE;g6Mt`2#@fY*(nYRsaecPIVW()GNo18J5wTFJ3Eun`
zwYUI_Tz<GK^%Y)eEYfBZG%!Uv`9dhHs=DrUt@Z<7b8~U@!E)@rSd_g@M9P|rqY|X3
zCcqs=z+8>a!sClzCI(+tBB!g4ji5=xuDA69!_-w;#1d`Cxg7V5%wdt3Pk#ob@<Twb
z6^jF@m?z7)An$H(+4Q-<tD%a5qG@U=pf^vy^&>a{-qumSe&XAaxMtNaHKb_7aK)vO
zBwb6Y*ES<y4;H=eV|)Z;^k%ws6zc^9lQAGj#QIp?!ouu#7r5`p+yP5POqCMEbcd$0
zXp%BZX<|bwFb8`BYaNdJ+JDB9phNIgweq4OQ~|81WL2=RHoI5@YXDBePredN&E@i*
zqDqLPjg`bhSTU*HN;a-oo7mf7$-qm!Sdx-D2FrkjV<0rvR%G>BO^WVK&Cd2!17I&1
z9Sub-1dNLu@p<SZ4p=3%gJMVm*IS#LownNvM@bH2Jm_9%kqVEzRe#p*#1XZ<Bq_KT
zvBTJAl7@y%x=NWHrk>Rb+oq7Bz?M(}Cf2vHA3Oo+x`u+5I@(8tCXmVh0B_|B1Q$Bg
zsTSVDhTDuo$|>_2xGAEkr%H;7ksN{2@zd>bcHEK3u(%)ty{;u^gww#~sgjClB34#J
zQ)_NDl1T*IbFd&@*nfti%c!a;D<?V4I!_bYIE8>x&*}=q+WU?6Jd2N30G0m$!=a)y
z@0nDJ>PnKZ$rvIv8jbEk+Q8@*xZTMB>@g*oQcIWRnM|-&K`f0MNMa*oZQIlsi(Fh<
z{-6K=6SQqar9^pcTg=T#BZ_Sbe7#Iqw3fA+%m+JI8)K7G)qhsf$t<y|O8^%@=)WqG
ze(kO4RK3`XF9&<+8e8y-U>-|7USz_id0SOXg_<<7Bm}Vy>)BKidF^3)af4Dt{7AOB
z);gqpA}J1-&;)N+_(272&*Cew-plm1Ur9rm#s2_M(bh*!t?;kqN<9+=Q+{kf^9QJ#
z3zM~to6^Tx(|^f4wUq!=>VcimNv>95bEuPX>K?B5JOg40wH-1;3^Pe2{#!E2hF@4Z
zy+KXft%$y$0XvQN!D*qWp{IIzs{qj&#IqJEva8r^%m6LE<6?IpSer>y)lv#t3Ri-f
z32BU;Y|z}ai*gi}09{SNI__>a9>~_Td5tqeHC2+gTz^uT06@aq>Nl_qFTb^bJB$Zl
ztc85io=&4=@}rD`Wgso=HUN9~7XVyhHnx(MM`_-)#XNf2nxYY@Lyj+d61(oCT#MWd
zu!V2LWm$7JTbI<=!vdpI(-|M)bzI#MnTMHw{g|*F+hq>=UpJ(xK5Z6ujU6bSI;fz1
za|Mv;mVYNgJtWwVRhw^MEC}?zWhQk6Om7m>RnZ9Hs1r!qSy@h*qPv@;S&H3m4Z{tt
zWa@@%K%uGOj(9=`3n=pBRO3js^{Ey-3tS7_+Y$MmU6xj65Bx5!EWW0m2U={v!Y6ny
zU`r^}9evRITEVPw4Bb(k^yHA!XW1--X!7c_YJX5#o<!M;JYf0JuY2fVq;p^h2}&NK
zrpclmF<X`jdRb>z%d4uT^L|ZBc-(|LojzbvRX_()*w`v&*OEykG|{3_v5+jXS(@wW
zvYmlrU`Pj%>~R4_{3EGygq}#H%xG&=xRN(h11MrIs188}<B)Z5!)0RXPl2t2F3qJE
zF@O9#754JsU_<$F)E2Tv--HJzOOav~w;?@CmPJuQmGrf7#Kie66yd`r*VM;Jwa2TT
z2YfZ@`bq0*SW~2d>EBSSNxgA^EG`(_>_HoLBWxO<Gt81%ssqf4R!Nsh3A1v7;OQ3N
z_qnj<?{aLmwz{?yRgt`j3l<totr;Y6Kz}yA+uOIK9f(a&omEuAnOtj2ID`+sLuGwH
zhEa2<+im+EcHb2I$12P`^pbetjcTLU7%=Dn18bopNjinSCd2YJzkdVEl6fd=DyD**
zS(HLGyzD^-a%{W$NwDPY!Nd+~^ouB_rpsH=CzBMCNY}OZwT-!Lh`F}-2pDp@(SMiI
z!$g|uXV(~2WkVv}N4?i{W2}%Yru!TgnwF&`lvGO{ByOY3){UkhMZgNA4b<P#3FE!6
z@}7>awr_^ZX{yecThku*3u}Pf5--2j=KBm5Dv9eV1wC6+K^#o}Wx-i3ZrZja5<no6
zxwU{61j@rxBy&|)#Hx-~{!5jKQ-7OU!?@npzLHJG36}XZ{#U_E=0h~)n6xVG1&9Gu
z+tjva9N3-6xVhzYRh89}W%(^VY{k<|j~eS{++BU`bHKPP0kOWs$g1VbXpC^{ShNI|
zKo!c~K;QyFBXGMZ1l$5_NWe{I{U~VFWviKFAPFQ1DJ5B#(_#*ieI#x<`hVg!qbr_O
zmT6~I0v47r7+D0mRA~SM>A6#Ti;H`kW0iTWMso{8aSRlQi7AnV<hlGnk}i99xV3@g
zA2O$;$nvUMyz;a7WHU(`$Q93+=-^x|U=})58<kUFNjAVW@?Xd*rsyJR8YXux@@fQ?
z3v~wDoyN=Wdm1w&brVi1;D2)RnoVr_oL=f%Sp?}UZg}UMHZ|E@O*|D)*{x?u5^~|u
zlmIQT+#T)@e)wlYD^xr}T2)xZLIVtwX%es^=*)dPUAeWcP6Dyxo>7;<{35xh4<$5`
zQR{(<tbSrE2^*V(<(-b?o9s3|ZtBW>lP{}=o#l=|l?p0~URSd<!GCe9b8WV<DmTZe
zyhX}~Q1pgnPXw|pcCU(xZ!Co}Bda$&`aHltD_>)eB=tw|IV&k@W}ZPi>qm(e!cw{r
z5_^9(_t>ql5Lvw)L~~D9kxx=6wK$GxRH2nW2nq(<T>3{JEL?m(tE0=Sq0IAqsicum
z`UF-oNK^}rXPesg{C~bl^sQA)$(c_}6h99DTSQvus`p!s&#V#}z#IP9v{&@xv=X%N
zEK$^ora&c90>&A^EKb+p+vY9a!Cva)TNZxQRrx%Wl2N&qXj;Qb+RDIM#1q_|$hg}T
z+Ip9kdEs{vv5}R$#{45)^$YhmHW<>gtd!Gc66aNv%`$+J$$v4BD3RFevs@e9fGkJh
z0N>jgmR(y@mgdtSf+JN=E!$aFRyMgi5(vK*C*J{llO=h4E^N(USYDE*NtU6m(tX{Y
zHB<)5KNjEvcIRth&rnrn6nXAn)GY-}RFyD5@jFI*rJmlX$RLpARO}Z1<8z1%lc*$)
zfuO3dsHhJxo`0g8s8o*Rivka-<J#8aaxr4-j-RE<aykjV996?lEOiu-$15rFY86%c
z-)VvGwZ|M_H@^t_fxJPYucvCsb3EFH)$K2o(KN6!vlTi>*1##VF6_5%acHKkT6(#w
z7Akq;G0FFGNTn@f?eha-JAvOGQt->J;ppt@EaHbe%zt18F&i3J_0uxRu?$+y6bRgI
zE<j!Y$E30h{tlwmSt`;QSv6DyP_*u)Zm$?i1y>|6(`y}o-q;EkjoFC*0HrDNoW_A#
z3d+`bDl7+<RT|1HPLgi+BG%GASf>4(&Q(LydDdB9QyWy6<4SqsiA&6jb6`z{kD-mo
z*++Y0Xn*Ryl%wm;o0BH^_-UWbR;0ejvi|@p6(`gQEV@tfo8I;#%Gs<POVGK()=*@%
z)Kg`$wJluHqY>sGRB9ZLP&ygH>~!#WI1g85mDIHIOFm;%V?IJlE6X^D0713J*4S}t
z`mM&~)crw|GD}SsZ$xE)i0>rviIm(B;4!`Ly?=$YjzPr>qWa>S#0m*1YO=->W!hj_
zXK?0K2^kFT-mfvATN`k8+TdeU$?^=Eo*KDwtge|Os$0}SI=w_lgjf*~D2u-QTb+U2
z1$KXnst;97D#-v=Xk|xP6}1ZxO0fpvj{g8M4Y2xZvf$Lxne4%2nn#mS)E4(PCs1E<
zMt{d~wlnO@FX(KuIO;6Fs`7d&9M3J1C84Lv9Mi=uGoX<P^0fNNB)+mNPo!){DjhxI
zwsFy&N|g1LF-Mm`%TXM468uAiEwUG4S=ceKzTtKz!uSo$u%~sY%_a<j2S}r4bZ59=
z2`hVSSCBX(YwqHMFA_&2&z49;vNNcVwtuBlz}#X!kFT;H5h$e1GHNK+rK4t*VN1n!
zux23M#>7|+$pg5;<1MO{R%!_Vf<blr*;M;(0|DFT^*9X+GmWV;NoS%f0wWtVa$RC1
zdX$!9sMxU6d*9{`8KIqO;%T*`p!D7_8%2R8<aV&QKKH_v)X-DTrW%(j$g;6mT7Sys
z-jGQxro`|2VF^}hd&}e(2hvwp;RoCiW6#K+ruYf9sDecJz#Gc)fkLN$!)w~xZciI;
z^TT;_15K)F*a~$Xh6L%;_^p3_aNe6SF;6Utq-#P0Yu9%v>9FH_5N)-O0|cXvI)fZ_
zRV;udkrgzCZOGq)Z*y~F^1u@`&wrY)F{jM3*wrSDzU}F@&@Mj!B<b6eW5xIE!Ktb0
z)j>$4dW9jSm0f~0jgW79+kMHk`NH8*g+#3AwhWQ$KltnmXt^hCjezbjDJzW(i#*ZM
zh-Qqp{le=U0$5u5Ty1N1xbJ{z@Uqf8lgp)q4zYP+#fY}%#2vn4j@H5?Pk$rnq+=*+
zqeJF;wHpFRxgz9^*b9;JxtFJ;r|O!?4OVLf3^CKl?Ifx#DwZ-?6l|j8ZDGj*<Cv7D
zNn~gwD+|XPhLSSK@@z;Z+qKW8))w0M2a`{g#g|I4Bu>n?4;W@x2)WsSJD*6s^!K^B
zvSlg~c7{nBIOUl@mEk2MPJc*HE)OGnlk@j#xji&(6<isF<kb=+fg>gK6Qo+?DLZ@Z
zZuY(;X!ALuf-0&R1a)fR7EOPFj+HjJ)3MS?C;5*XTo~q*r6ideGZc$ukb}8fld}Bn
ze|uk(gy`m?sF~@|>j@^aAwsJvpGc11h!7sy&is3mhxPRWd8%q^-hX*5G?0g#w_6ip
z+O|6f7XScpTdA1P2Zke0>a)3L-PMhSi6d}qMTPtJz5!^d9)efQ{{V=jGQOC6lD~RZ
z<w))UupkgO-;6jF)r$&LwBHjoLb5pubrnJef-PWlk?w78mL8gFTGk?xqPCirVgQz8
z0zlybHwYNJ18}@v<9~acN6MaT$>+UANQAhM+enc))Bq%#f<G0_@C~D=q?UJ<D16l#
z8%D5+CGB<5w)fje2a~;p@K|apP}Eja$sDgFZ3@G$bxj}n6JRz5z;1aX`Hmge)KRux
zvC>khW|z*2d5)(EYXk!Rq8sr6{wwc-<{c@Ru0unc{1h!oEq_OpOdoj0Lc0PE>&3;t
z#l8WRZD|uy%}+>D8bv{P)q)@hs?J%CtLtJxCdT7#RF*o5sVd(uNhWxul_QbMUE)%1
zRFl2**?8a7ZESL)TB>SFdU2WjJZvKjs?6idxI2-1vgzRQaq}WaAzIm5vaXh%nT0Ay
z@}d(cSk*ltk$+cZW<QGLYTnyigtT+!8LTx<B$g>@!MaJeo3OpLZ+mQS#jbg<jI6Ap
zq^*r$%SMS{6xAXp)f<vV!A92AZq~SKafVUW)Xh`zOrn+0G?Rg?&@$aYU;#UkZ>6x6
zOfNT-uOFLJG|DEC2uE<FkQ7?N`wNgy7UskYHC05yVt?+B<rVV^rjP}VhS#;tfB=wp
z9O8E?&a#~DdYq;_yiZ(+u}GB9<~1=?0J8yLEwQr@J@0%$SFT-2TT#+^!KZ|nmS?A&
zC=90INe6ZpCw+*y`eDXfRZW*rQqg2JFh+=^6^%4U1>J>|jsadz8=LSB0zIoUDP)pL
zjJ8Tgg?~KQ{ON)yXu;BU)Cjtc3X`z!d<_2pqxyd}iX6`-%%{sUr*ArXiaKSERs1WW
zN0V4&<k|N9hA;8tG<9?cC#*_zcKh?7N(du&=H!8L0OxJ-0<D+Q=ap1%pG)xNC7KpU
z$x4ftQm?ry>1G?2AU8W(0a=GJ&oX}|hdrQ{xPL<aVb18F1O(D==hIk~7w(0}q}&UE
zh3O~$IVC=G)4B1@GP<-LbZV+trmz@BSY9{NzJ{`qdjZMNE5!v=ZCRI4M<St$6jUL0
zAcO_2bEx+Kp7z@K&hvhS$@2`-voy))&2oC9P}0{{WzrM?NCmY58El~G+!hKB_a@-d
zw|}9!YcS1g^G=28Gm_?%$9jq@f<&6jEkrWh02e^$zc4pn#-Xop8sOpO1f4;VWVJM1
zf0X5!&O$*-y4;kk^962Ncr3?BQUmDJJp}MJhN<wQGOehA&xl$5MMAWcad}h23we4<
zvlHY3p^dF_NV9Ew;+L5CQPdPOQ>T&qd4GN)Bzm}WN|degL_rY_y-`_gmpX5zKH7i*
zuuyzvK~G6j(OK$ec^8r=qM^)c14{~A*izS7m=3{Y5-2wU#N1DG&s$`9r1kzra91~>
zV72*Nt)`-d?hKlvVY;&s_ltP<{9D-J*;h>Xv*C_Jl-(^^n<7e9M@p(V>gfc8+<$p5
zc6f(39$%`!o&XIb@iQ^aa~Lc7gCoduY{n@jg|i&5F<MDiR9>inm<yI|T9k_rEI$^Y
z9-gMDhH8Y8cy6R_k1<f>3+``XEC{f%`Qfc~8&X2#EKg6Eut@Z)G?4&o+Eg2<Vh^|{
zdv?Zs&|hdClX`d?sAEjL%9<8hh<}Tt9Vd4R2~c#AeZePSEw$4o>O7w|iaNZfHlfWD
zSG;KjLTg)hbR$@~WB7*PA8TMZ^TkN5p(<l*6jw*HB8E3x3v!`vZZO!$)U(oNKMcqv
zms-ps3eqC%%$oKjUgy806V4n}(M2Ro!l1N}p%Cc1MX0xOc(Q?Quf`%wFn>=?t|1yk
z@Ar{;2UWrW>eF#@L9iCRn}9|Ei#yTIzXe+0PIWkDxnf7)Cfj=!76*TjE@?8F;YzfX
zH4ut=pTCKN*vDc$Bo5$#YjAFUtg7oOG{;p>5}z|!N6;6{B-j;DMxIF1YYU%jDrqKx
z#5snxj!4x|U(&#7Du2Xw+<yQCy(hQ?6iG_RYGa-WOj0b*9B3}CHVSQRE%F`CuHzZ+
z+Djtpp1H_qGYqDprD}3H^~%1H1a$Ij&%MD4RZY(WRr_N>t*gtY&E=fZ%!OqpIATE2
z=~6%%$-Vftz+S@J3)o|mDw8jId10%DP#;7UPy-y{fewdJ78*cSHh<g>FdlmFSFbYs
zshh(3?8-XosG%)GStf@^^j=u+zMUt`DsQ^Awku<|ROGT{`Kiis$sx^Bs6|up6yoV;
z7qZ{jxEquD+a4tG1ERCf2>d~g8d|1};gM?SrGZ_OOED@CwXbHn+kQ7Y<JcVy)R~`J
zbnO>W(?>}~T|FB$Tz@qBv_R3Q4jFH-3_}CG!M%;f0(1C1L#(vau$6jw9qJaLw8>@E
z8*;|sK(GV+C!TTQuN`_)r>49{sG-ZU%B9OH<1^Ai6tMEk7I3y41}(llkx7|nb#gvR
z43kpKRpdc^4pbIHxb|{CJ~zf!_Oe)OdLJ#UG74J7pW$U?aDRqsXw?G7mH<78Hy`>9
z$=?CyoR*?2vouO-O8MrY0a{9!!Cx)TpHNd`bd#pW_agnqHfq_Yn<t{1Gp3q4r-n(^
zIpVR9Yb+!@?WAhI1pD^K`fDpPM=b>knu)$4*;`mZQBhw}xw4Q0a8EWD@1<4dwN>;n
zN1NA*s+w9Khkuf0BzYlC%zB#I$u~Q00kAd%keJT8(lyBwAc~Ac#I;PbBpRI`$|gA2
zovaF<)sFlOQldI~fmc;aP{+@xT{n+N^$~kpjki0EkL8Ubx}F*u73%8K2>_TTK53I%
zOs3WYWYxNj{r4MTT)(9UP-Qgr8GQ{kJtTCGSyfUUMt`{B067J}n^FZFo46xY?SQJ}
zl$l&`R8&UqF`77m5{oE2x!b+L9ro-=+h52!q7Jm_b@2A1BBM2{kkn*pR_YwKs)-#Y
z3N7nW!sOrIu*Pdi2gFcj*}i22zZTO-@%h^H%)x<T><*BB2)G@|AnY-8=&ZK7Z1*Cf
zrlyuUs(<8JqKU&@5vP%SwX+t}ayM^LGIzjIIxoZCwCgyh>0Fzp>f12OqcJ=I5;B8Z
z0<0JU(rU2MJ!aOi?WlNdoliqra@SX<iD&panshp&8y_v!O-v1$irV~I)&N_cC-|9`
z%hIxRga(!lqY{|&JjEhtnxZyf%Nwd)Rka^m0DlP}-|-wpbbW1l3A3K4q^C*=p^+n~
zilriPDnvwS8#RGzswf0%Hn6~N6`6ZLl@74|!Yauu<PR(j{Wkz{dy}x(<IG+(WOQ9W
z)l%o#a_ghbgq~9@zI+kZ6Q$f-`Yamja#Ry=ZLr+)?-Q~N+oGlE<jl;}xs4G@d8yP%
zQGXi*)EaeD_eB{sojU9+ag%iSNOkX9bgp@kO-)6b$f|QXKZl+f9Ms0PD3IJqDReqW
z3#(p(a82+O-wrCK=&re@$U2TeRRpz9MKaMy)e>rvtR#(Ne_hHgtT$VCDsj5?$J#e0
zrRi#{ucz9Pkt!0BrlM=dJ>EGtiZX9tQGb-Zh*Ee`K|WyW>WFfj!fdWLt4aLA={0<2
z8mKguP?AKXgkq!;H8B<d9@x2fcj2#A_={T_p{Iu~yHr-y$?Nh|$sONQdYVQgSZq$>
z`y2$vO!U88c$Jn-UDL_r!%)&km{!#rEKJS4*pIz)eUw~-Vo2v22TAxF(YYI>`G2i5
zbsciY_mkGHxT11E1!4zE>EH1dBE<2HGcCy|I%g&sHc3$gbaYTNm?eEuF7~q$04&<O
zUc%Nl7;lwR#ZjsnI(4jR3Zkg<0gQvOwe7&Sa6u=XfJ-bQIH{<z8YMK*L=~N5vc`0c
zOmu)y1><gZ7uyTaWf{d(B+yKa41Z)GETkX854QaCh7^_6xkFZb8_p$2j5BL?0f76<
z1yXDc$gubM?>*BQRR(31tyGz+Q?of9ScJW-2Yc>){&&Dn*XA%)RMV7^Skc0w^9N-k
z{3Dw!j>m5Q04v5<phEQYwCukU%P?Zb!X{nS#gDy#<c{0l5*e)(JOj$7lz&u18~{$z
zf*Lp2dI39ueXKie&g}w-B=TpT6iosdBohPX18~EW&9A6l*8_XtBBL)$H=3cDNG+?&
z1nD4Oj&=ucdvS+Q=j8-zPY0SK3dU8C`dDrOBF5(Sz5Rv85PG_rc9s}pRCbeC5|TiS
z55sN%=Ye~RTw}E`nQ5Ca6MyT8!ify6cep@F03BKf1mA1?unwy8fbHbSn%hmsmm=Mn
zfhq{Nu(I6m!Q%|wCXSM7wn-%NWHB|+nBf*<q=Iaw?X|n=_s4Y6$DIJLKv2KI@beid
z+0)KwY1MS^q@}H<{F{P#Hze*QC@7yasI1CqE2>v5Ga8dD#I~hfgAv@_ueN^zB5CSr
z#8V`Okk3wIF~;m&)Sr>C8oxpT>@dqNRBCwZS(r&07hq&<BWn*(7Y4_<<on!Wee&wa
z=9ed<ri!vwiPuZ5oOc8hW5$b%*c+Q3F&Ru^k1RB@%&yGDg>6d2DJ0nG0N;Q)xg2j`
zZ1u6)Nd5>Kmq5(UE;MS?7`T5o)-7uQR@NT(c@-4(KxA}Gtp>j>VToChnA``u0Bw8Q
zRgJA}ut;guCI0}rcZ%_}%jQHb`~n)sVW(F6@y7U$uFgIpDIlkfB3Rh^Q*uqa-`_~S
z`|vro0mQR;P@J+`F>tEFNyAw~YGWGPNWS;ddwi{jURGICl0^hGg=c?&g!8m8Pc4b`
zu>jlv0KWTL*exYMq{(V&7J(_>F=)K$8%~u))OB(0PWu}H&G8>36HQKPsOI=|iGeeL
zp*86(Z!m4H%51#sZE^*HNh(**Sz@UWzMYaONQwtw5H_K5RPt;uW8Z5Bp0-L#7?z3{
zsU`CyX}~Gw8{L;ugQ$OQI0DD8zUFBpiYi(vXD~F%(i!D8EX<$;^xQ7O;YEoAUgL5|
zDCqA@YGPn+(p_6y8w+d+0@om3#jcIG00IbC)>GBgJ!V@`Qyna_HN{i<qXOD>Fxam<
z4{MM%!)Rfg)3jg2Vul)tBNwWd(Wog3#BzN%vmN;QVQI|LR2YBdWsoevWu7()q#K2~
zP+0f$s5c~wd%2k*qo}EB$_RcX;aCi60c5ckTd-rQ>IL`UjyM68<u{R|pvfv^sUeIh
z7zBX8ge`l5JwdJS-wRDcQAL!7T5GhPnF7i~G*{SYUr4z(Bp-i#JIeZo9E&fhks^QM
zwEzlfOA?+?ZWMppOAWr4Bw?;+nN?;hB(F0>W@JdxFjy!z^d8qBUi=TQ0UVU{o@xXN
z5>h176KVx^CroU1A0{9Y+a5r9n^k2p#w)1fT&04jT6m&zUi*huzxb5dNxi_|Y&DiR
z@}W;DiBH~^63P^|qhc<gZlJ20Eyn)<k!+>2Wn`s^MD%~L%I)iD^s3p51t(3`_R^|5
zTTv$fu6fYiEuFhmbwyrRUsGEgGshfsp=VI{K3|=2p@zkP?m^_7R|kdJX+IUywHfwV
zB`_kH>GLTG33Gi4#kJb!eh6K*8&A_a)fuWj97H#~O{Us4Zb|moSS^7h3-O6OyEdn$
z%w>0amQa6CMIuH_Zdnw7Mfb2$zs!Dk3J!X{DD-4e*V5*#)l?FzOC>!_0xF2rPxCM;
z8)hVtr0%vH+6yW1_b-;Ma{TkB^L)Mtre_6WF9}EB$fm~o0jL5<7x`mQ9bMCz^;ELu
zg{G0jsZf<kEL9It<%ZBAadrwyo(i2^yGQ2P21kEgM^6e*1d~q#2fZ{-@r#=Q=Bofy
zvk|NtTx<>kF1N)jf*6UY&)}^ODo-L)%x2IS+$bRKZGXjJeZ|HsIWLGkB$72=SJg06
zRho#WpD&$ARi`Ifxmhi&X|}~p`5@wf)qV|lbDQ+|q{^essIxt9Q6%p*;x!i1JjnFe
z3*LX=TVQzalUDd4;)>AHXL(^sb830aW+7CPR9no{lPe3NsQ?nX^%g8N?g0-`OVt$q
zWnEk4xvfmI86=KMN_AAZAeJG@u^KfUAZfYm2?9GH@hhfkKZ@$jW6PzYM|oA(%9B7{
z^~WGmH&QQSvVsY|oMe=qHTB0qSA0K+y(@p2p{b$rsZt=mT7=SBgJ|^@R?=870N9gz
z;<?rzXg>`4foW={=n9&g;+IrwJyl?2g-eEAAr6yqbz&@dH}(%wz=JbXVIvsjNactk
zs1l(gNx0+zzOMGeWXQUH(B;dM)A=h>^9Ve(F>5QNv5vt*0M^xO`mK*L<vu9&Y?*(h
zPFvC68<UEuVOk0bYL}LtA+?yhmu4o+yBiL{MXx6p+H0vY7NergV1cyAtvUEq*E5#X
zR$XIIEN)9~FQ<}80q9zf2>d&STAbpGr)g;?tW8nN11xHmcDtC!F{_e5^$QSxOvkD`
z9O^?~Q<(I%F@{<?pD`=x9y)6{+@pU&c01p7JDZY7u*By@eV}?{ttY91cBaZGX`voV
z%^(2G;Eggk0BYQj2FA>8HpTHX2q|ZZR+U&1B4E;$R5lmBs|{NXu6P!}Rw#T>&$?Tu
zJ`1e!3hKJpAIcNOODI`f+-cbiXM1mOY)0274Km*qygBL$nrQMqqn1evLeqcJd_I+f
zP0Kqpg=26nVd@vS_ahbxzP9P!p3JGJI+v)apo%~$n8{E*aQm+JV_?9G+>N*673%Es
z!OYq+^T$C`9Z6}Rs!BwTNn>YSAUOby)NDu_Sl}%*R8iJtapzFgN#!HR^wCDjr6$wt
zZL9DwC3d;6zQn{sS2b+4GS`0(z5qm4P7rKKH`r}^dz&4vwj<YFQQ`FrwDg@P26<0X
z46d?@_KqzGz>{fMs|`2&EE?cjxWo-tjM;@|R@rWIkS1wGG*dJ&)lO;%Vn~dgI*ed>
zY}z&~wm*i=fb?2I;)o>`agrL8B>7FPRaE}~FujiUZLW71naZUS*2jM)V<fceBy75*
zW^@b`*bS~MO8kZP1S!9nx?{ym-z-{Oqa%~yCYdb|%c-<^jc^IlgMU(+h8>3_Z)}gn
zuDORXh2fygo=D4Tl;teZ>QIcN23_r>k~aj92RI4uwI@x>)LtA@^&KQGm7PC3O*H5%
zWr9MnN(URMF|i)Ro-u!({g-;NDD$k>r0XOOnS@5VnW`k+<wXKS@4F!f{{X<>+Y>!Y
z;~!G>E=yMfrB-k&R%%+RSSUlu8fnn%#Zu}Shz8(V)*E9U@aAZSZv|Us>66b#X`-A)
zTSMy$V0qk?2a&sWBoOvRD!of6VKh|{!E+<t#uD0Yr1BK#1OR_58EhCFRJ>})==!U}
zn!L5-Y4a?ihHC0@BJNfc*3qL6QBxg(QUDCY++#0q68uiA4-Q3{=Gm<@MWtCO+1-(L
zyPdr%NCM~O?S`;;r`G7d#Ooo=a~dgPcw~|&o<&IASy&gh8Z~-JvkL=j8wM?v)UvZB
zJ4Rq@%M@&+fY*Ojy~*0cz43Qti!(7~d1TZza;-Wt$P~ot_p_rGHfHs;*q+0HxUA%n
ziq=XqPKXSWK_Zx}RJbKeX%-jY4mtM4&!k@|rjB_j=wzs^rk;17I4;cuaimVE1%ml(
z(l{~}Cx8Ip;3Tt*s-m}{GD=lrEV5=1%F(FY!eDaRnYn)&8I8X^_r>SK4-6}F$_D8;
zD5~U_HkMc>saGi!RE99@_S2<9BHL4ZfD3I{lIgs+r|RpvbsnaUVNnA#l$9FnRE1O*
zI|5XY81M47_}MxJyQjQ0tm-NXY|dCRg@k4jR@E_@JhCWZHdjISj9J4V04cJN3mgZW
z^Y0I{?udWt1*Xo@o{pM8T9ybTS)kV%i>a|?Qb|b}PUltoi|A3uoArlH)OGLi3Mw<x
z_0-h0>qeSpjL2O?pvJ>VDFr>kgK(*Ha6o0_hJ5`)*O6tEm35W!K|E`hz?8aLSn450
zjy6OEyuck5SgMi@kKq@Uev!;5d@HQ#I4TwDGpB!wrCb>jSbiP+iK3mIf`$z_AxJDl
zmH=2>GfxqQQ<n8r1zlAQJk%MQG&J=U3L^M+4x~J+1<+_Wus0ZW(%u<K;{`y<`b(_x
z?79%GQ57C@Q!E~A#`lgFRAmio5E%Tg-vrHatiL|#iIz`XnOd!AM2pVUt95Wk^9vpA
zzodVBu}Gq?&H6?<Zmy)7uBwUU`?({O=y|nlEKR^FE^aSyqSy;xRd_Yxhf7z(kmeP1
zac1V?N@;XQ9O_Qv%?!xp)UED7>})~B8~&*3j)|79BCW~gq^lqYj#)%vnazlmMfBXN
z*c<Kw_5&MMN#k;Lj#Rm4Qo6i}7$Rpn-U5GE5K)fdzZRzX+k=7DnI}s4&zDs6#2LJ_
zx&B-{`f28e%lL@a>WwO{pun>@*_fu>18fF$)U`7Bqtm%unUR{eImx3%ib{<uRM7<0
z<QGsGqbh?&rZ%~@k$qQx3Oa`>l7_o2%d%R0m{YYCB;OG@d8(>ZNK|@)uvP>WOIm;T
zxbARz8tliVvs$dTJ&>%lrK3?zq^z>W)(lt?w>BhkY->Im&C<PF)HK<KYVk#yQ&c5%
z-fRHDEBSJuX;i5J)VKlBSn307lE61vXkntO6#0-!dgx4g&#FtvO0g$Nu+pW81AC8S
zg{L&|po)%0s`A-X%{4fNK)Oby05N~L1h7(10ZoPNg;Ii@Rw|((BBe~SNX$bh=Ss0Q
zW*~!b4<igb;IduBYZJI@hXIi_mEOb)f-i0O1L_F9O*9mRXREBHGPxzyO9Nm=l>qKQ
zDnUQu=WHh&Z&re5>N;8$;h|}zzM;w3TXTL1A3QsG+L8vTlAV&A86b~I*<*iZU`Qv}
z8(PF3`;fRuQ|2<0G!*hxEQFHg3s~?+0^nO?zSiFXb%tkjWQ3?IWmNMR$tVEoH@$}7
zUgO%p3^<@PQc%e!pXG#(L|;~cAya({NH=4)!18Uyu-cYcnn~eiESD3=(GN&{H*0~~
z_W5&TY(u;=O(N2l*5ULZFQ<P?vwtu=3!7NjZ+rl%S>wCaDT*0X$bpKN16Gl!b~o+x
zHn%d=?-cchwQ1t1B^{AGps6Ln2EbTa<P)c3#@9GyRaH$+PsBX(R4TQiVsxN3HrVgA
z$F;4w7;0){%AjXzd0=D+#yR3;GeYVJa0svl#Dl@GBjyTLMNt5nqNsnVNflizL3Od&
zK<r63w<h=Rd|0dV%2uMJrl6WA!iA6U1`TB%VuoR}Tnly^TM(T!=gCtfvLtS;1FK3A
zj*-Ha*}X?)7O>rS8{?Uo3tJp%6+2T(3I~o*LJ)T)tZa5Sw#~Nw;H4W?PbEchs8^O)
zONj#zTooh$0J$T3n}L6DIo}6LDW^&}<A|fYWWvltF^ui0MXhmm>_3!@LnSQKa=hx!
zkw&agkV1zxuv1~Zg|6L)(+fwK$w(HmO%j5@LaMq&)a`Mx3fqpy`T%!XnHrfU$<l%-
zrGzq}X%4w`_Xk$YM)v)Uh9M{^X=bQ)W|YaWRd|cZA!E&y+>n1RZhuX%zGP|SnpyQ-
z^%+n{7`t4LKs(vKb~XTQ+Xu_KlQ8M}nhe)1t_f19GPEJVWERr5)(E*5x8ft!um=EY
zoYsO}RCQ4#&Vxj9LosA~Tn{j<p2Pxn1C9XUyj0MSBAF}dsv&}(&Xr&8)+q(ql|UnH
z$iC!<zatLl<c@#11eO?@rJr1gjirMI^*xBS&F{9`>;U1k6%oRfnN@yMPv)~|H6_@T
z=kYD~yI2A(WACuQK5WW4siCD5$OH;yaTqMx$QCY6pjPNFu;lTu#BEZ|{6b92;smBj
zf|4_#(#_^<+D(sQK(}qi_QFj|Q58IO4h)p_Z0KGw<wk$bwzjJcz`7A|E>FJr2g)%|
zGe)3K_#OsUc9P_4Byudaup@t|xISs%^Ju9gq^X^YlTL{V+?fdru?#>Y*a5KvdvLaC
z1hHnQDxO*-SXRNIZA_Ld80>Ct#_Vmz+~GXA7x<i1wa(R2LdE3rWDBh=VQBybPp0I7
zZ_Jypsgr*(T3J@N?+}(K%{ACuk4PrkfE#bY?|@S@i}-AD!An!;N)SaPj|<u|KN=9?
z{*moq4Z8pzK;XY(S1Y5cte_QB!6YQ>(J76Kwv9fk6Qaj}00!XmwjnBhDp6NZ(V0V8
z%!*2cl&#JA*nmLd{rEVX%DRUz&8w(sr2hcCG)jLFB2+$Hh)Stq!*(UTJ8kI$xxg~c
zsimth$xS=O3M!<<_KRAUELTtA9@f8o`{E%gAgR$Tv~iV;$b8UrlX1G=>%G7B;}bcJ
zWqh?p1cpeRWRd1EzNpnfuuEF&>K4_g@+>hKn!azB<<K<*bch6OjZG{PMp{u~r_9%D
z8xDU{V{>D1fPU{nwBW@nDuvydv}<i!$Hj;&2>@(9M-kcR%_-uRc2Oko#$}0-$_*iG
zLlxLue>IN6`{GiH48m$zDYIJ4vWh7q0-<81lj0<Drq-3)ND9R+H2SUumiUIwD02M9
z*=<3t&Qsyyr;bX>hG++t2~Z-BOk^uuU6p^YvA+8SDuTZy@dqh`HU9vkGZ~?ykTn%z
zQfR7OCsAlxCK4MqvQ4%z^R@6!q3RsRp`ZAjK61I-GPIxe`GP$?Oi|c^?<SN}VQpJ&
z^S1X&2=hZcQqtw}R2pc>WP!$*krbP){{RiFPx)JNtwkx#vifRzm?NeY!^+SlEKz?0
zX)3onYQL)Yxg$-t7!21k`#@!cD#?(V+KFvb<<VD)i-K+I&<l`H9Byt-_;Eqvk6lbs
zETh7l#$h5OQ<-6!80ArIN(-v1>LG=Vg^h-zxzi(po`#{Pk{Ws;GP+AAmP;8FT~^Cx
zONPCUpt`sq+YtRTRn=+KS*)E&k)?lbL^BtTj;@a}T6v+em-AjXjr`O%*!m8leP4$G
zm5!kMN9L8&QPT946mA~z3aF{$Scw;MnwxM)1AC3_u(hEi%=|;?StXYz>DV)9D5Zs7
zx-Op;kfmFM4I0Q}RRBDJ4UL({qR2(}8S3+;j<yOsy%v_DWoTp-W!@sh5bl2wN{6r{
zqLOcM#t@~h>#8bxhRW+Qy%isEEjuiXCQ@&!S!;o1EwY<iYw>{1Dg0LP@(i*GBJh?4
zmPck}OunH^yWD_?;0vzZfwsc@lQa88(&t$mxmQzk5PF9)M^u$Mql}qC`cAddfU32Q
zkVVGlVm3;8p0%c_zrf_x(t3ZC$SWNIx?bQq6kDx;C9Exfr6NrHP-K;@oOCsQWNPc!
zW{W3~$t^>a^3_JB1a6U_0F$KH5*q>M-l@$x2R5dqsPOiWHleJVc?8*PuM@(|yXqQt
z(Yh#G6*_NwgLAc4iNG@FQ`My`nrE77n^U1s2ne$qd5dNu!r+moN#lPTxBmc5^DeK<
zAgGh2x@)B=BV9&FqDgC1s*#`~FO~@hx8c6`u^7CsN#qdbcgymGsi=vUMNFj4I8X(S
zxh_z1upoSf*bFZw@T0GD%Gw(IqLK+IYLO$?oK6-JONJkG3cCYsBGHS7zZ;E1An-aY
zrghDF!#%FetDV*fpqYRCigBaTBUA(x32h_R00P#w*q8Bx!u9I<ES@a4HlHR+Up1tv
zdNVneVyO0CTB?EG_t@KWa&rvRqdY<B8gJsiA6uQYDv1?MGZ-Ee8+zcqm|ca)CigsF
z0kl+jW8vOySebPWQ<Y{3l|X8InxgWMTXsI8$JAHk4|8i-u2X-|JsU}#!<o_ac3U+~
zJ(E;dN_`h-?;?UDCe3U23`Y9_i}bc{ndMphy-Qh{%JoxI3Fc_412bwse5r~M75iy6
zAe}4*_S<?lsM|LbnSE17JxtLo@rAn`<1zwMYqE>Cq;F<67B&ZMOO)kaV9DrcA~C{a
zD^Em`Dr!2ks;GYlVr{7IH8-`yeq)}~<dr%r#$z1vESgO%EVT(Gz_?~bas{o;uYP&Q
zAnGiNvb!+KvYN;$^Ex&uR30~71^)mD0e~e>Ac1YiJei7UBb`|w^G!mq6C%K@iU4bt
zAQFEN?P4vlus8&=E~|oq3U}$h4fDv2MC>CoP{m8}1(<&`vk8iNODt?|F2>s(HW?p|
z9RZp3-5lLl7lqZ?K5taalf4YI^DRL@8q&iPvmx~<4p!t|<-2ZlRh3jSD^k*$ino1I
z0TRmTw*iPG4c6>AvBZ^CPE}KvMOQ5ry1@Qvn3nZxu<dINfd2qY2aY-uHj||DdR(_L
zsHmf>rd5CVsHIOPXq@`MD&qVF1y45|gNqGjPvU#^B=PuZn$u+&S5eNbVN(=P#X17V
zcLkJK5~zgg^8j0dFAL(=M|h7>)OB53)RZz+W$-<t%reA8i6bn&NpYsppHeAPVRA}>
z2JqP@h!r0R9C@`&)78aJ(6nt3(dB7kpbOf>*q49y9^har)%`d81F5=~r?YB`iK;0j
zFikd%1jtKzfMIe9l{VP#ZaH()l^Nb^80p-%D{6eam{ztCAwfLLeG(t#xxYUnxCGwk
z;V(?*ej#|NQ<2eA%kec*B+|7+Y|1GX5?)y@3uzJi3z4T|e0paw@Xw?=Z>P*TMHX3I
zSw(+5wDnbzIR-bFC`1aZ14x9D&ANsqeO3bjgi&S``6hiynox$A<T+S`MNo!MDrYUK
zO9BH~o9Fl~u-M-EcdashwaO>TI*wZUXoT?Q{!J8uAze%+*5;|?QTJYIzL?KH5TFZz
zX89!_MfAo`nsp{<$(uo0)M-OJ`Gl&JvCx0hF=>sJ%LA-M6Ei3mnRYh-&gZn59d1jM
zb;IP5AnIJPo}(|zCTdz}$b}vu7OzSwq)@R0Hlhg@Dh2}Qm-va%RUJo~_@kcWG}XDi
z6)~pD4IMhF`HEPg1y<JV10pNfx8qO`d*4ZAwEZdJ4r|t)JLeQxBz;3wHCA1f=5T+S
zWmR-gi1fD&BbHG9S=2z&k2?5cnPt5dm)BSIOF=Z*Uz=5z(H~RD=%-GHT}$RG$fPB#
zN`^Zr7^nEP<3CDx*PjgQKhNsAdZLbrDQReI)56lmH-qqv0h>+9EH>4(>;lKp8KhMG
zPf=Ah9G?X4WTmH+0%D|dsbg)uM?-%g3{PXu*3Udq=v>pIdV%UP+L*H*n9imTNANO<
z>EL$^RduU0WsyyV+g6pmy)BIoBBiO!WXqmY1d9P^<(i(E<A}v%@+4zzL8MnU)xFS*
zl6I)|9(9z^<()&)omJDTEoE$t{3^Yi4=Cz78HdymHr~uEa2Z?(`9*j|s4#!DMp<K=
zFq$TnTzPTs3bDUGh;B6iE(Vq0HhW85;#W>X)Vcf=NmEr(8%rcwRpgblvs5Y)2D0E5
zu>(YV6;LsjC&%X5)jXNaN+h$WW@(j@7;eCPLjW66TKd8-9>-xJm(52@Q&Cx1>ok<^
z^1xtC4#iN7YAtBci(hN%(!_tAHXzr;oW_%?I=es4GK$5Pne#fS)SD;?=96)7ETk|X
zayAyhJUp(ODC!z~$>xeme~M8p9J9JSkqfsgy@_2Ujqht>=|SMdZ&BuXbM$^|B`#f4
zNUu>_UlEQ*a+;h)3X;yE--v!I*b}(LqpAMOJy}ar=j78>R?yQ?tb>0uo`vCvs2YP1
z0kQP|06p$W<com4_-*1>M`l^nxo=SB4^K-_tXxFoXlC`dl+rO*VB{;Qaj_=k<7Df;
znaMg{b)=^1?7KJ2S|vV0LL-(fgD90tFc(%`ffukou1|+)GCHicFoosI<TC{;t=xHJ
z5Js<9)5ru^8-7^QygYyE4A(R94jP=))X|26;Sfys1nJP8q5!q6>LYG;7!AIbo?M*M
zRY+wMvV}%{Gaw(!XvhFCx!YB@9AVODa#B-M(N<=aRF6#`lo4*AfX8q&dy+pa8ka4Y
zKqP90iC_$XGcgR-0dKL|<LS@J@q#K!*OFQDS~v)8D;)CxKmdPmH4jbiZLk*EY&^%U
z5lt-W5*Bd4u966KwTAX9-rM^CTjnbSN|3~iB&cHQ6YmIhQ_aQqHYDGYcH<1$>oWY>
zN_CBqA!lHsUo=Llx6-e2Gyuc~xxU{#Bbqdos{yBuNMw!lfvhS!TI=56?SJfG5X~A|
zy0nu_N<7Dn6?cDSP*|{Y?{YR57C)9F(9fAlt6etA=^Tn~?n3Rs+Q0!}?ml>sr>m;V
zGbnR7FAY?3I6_q&0owLm?r(dY!ML}+2~Q0+bh1rXQ%6pbuxmQ`Y~{!;Z6JUY2PXCw
z2G`pF;s&OY23l%K%D@{=rYJROYe;MfHum80esH7)9LRr5Pb^DLhy$wy^)>hg_hG>K
z3wObyp`ux2r^{+mp`(M$^CLT>7O*${{#aoJ7^7y*hGMkU5g#pMvDQ6b;`Y79q1<=A
z;b5nFnrcshX(kT@p;?p@XaW#P^o!hcy@0<xndW8~^}`pCsv%OyVe>@p#Ozd?DYG8@
zfHpYgCSiXiA}sZl5==y`50=`|;#D?PZLCFt+jHCHgsYfER?<|;yOLBg>SR)YFDCZ@
zg1zo9fQq8aXRmpn@~4!=G1g#T{Ff%;`~DMd{g1XC&5-6%()@K~Gsu+j6)U9;fYYRq
z2)5^Yl6Swj#J)w;aAcW$%FrX#h=8U-proezo2`G>mMw1E+}vTFX`N48k5D8^K;afN
zSdXNWdl6s(ZOV)P04@SMDxOx%Bd8NqwIoaEj!8iB)kTGr6R;Qjg@**;)@ra#S}5X>
zK&@#(Vtp(?>^Hrw`JUD%5mgy%^iLH>n990}y1RKxto(u4eE0mWFcPjB3TK7rq^dK>
z!YO~vm7XzW)CXWq*r^vDpZ%~7b2iQvI3u2Ph7w*V<^i-N^nyX$k!I(cd+&zuR8>ri
z3ZY6_QNaO#VnP&R3w)R5&cmEJ6>^$cL~JMchE);N5K;EH{6MOZOIw5L`ry>`)bYrX
zSH~+mBS#v&#+SJVS-q|-N2j;X0BS7I<@A5+Q%glJpI&3q(#z(_APg?5zycJG=J&83
z=MOWS{*I$56!kPnDwaf;RHd1sTQ!gl3A=7BabPyYWh78kQPV=L9J0XWo}4q4kwwn3
zcv};uz=OHHuZPg(QD!W@RW5~g(=xn)Ng7Sb+Q9lr1YCRHxJSNYnNQ|?L}?W;K4pL6
zj4q<VK^lRuxjw}0?ZWS!Q_{^8a3IxBO!2hQwu9&d+}v1_E$&V2U~DlXnbJ*1kc_&L
zC=EE6PpLxrvamaAV`6mw04O_N{{T^5R+cH@T6$oU#Ue}Q0u?;Xt$jP2TVvReye;q#
z+*Vc7OI=KmrjXHy9q!0ADP|y>TX25?JA-kx#o9qtHeVPuF-)ctf5Wd^C}6wXg$fOU
z<DX6N+M<qWCWZkVt`U`0VwAkCJ6`eyVgR|d_y^O)?j)%x1w}<1Rc}p86v&!nk&T^f
zRGs-7k-r*87u-8ZO<h^$28ajJtn+D5peX)va&K#TfxWNv4Xf*N%6Sx1d3Ap&3%WA~
zW!#&B%x)cQ-11304rC+CsIqyTqs*Z+?t(K`W-KC78pzr#SP^4>#DDi$DoCJ*%BW1y
zJh_yK7C=Hs=TR1nYV_%_EJeeXA3OpS62(tdB$CEfCsvLnra>A2qUY2|3^(1ab8~z5
zYJAw^G*ptzu*_qVQGZ#~qt<`7xdet=_qabGAi9^Xf9VK>6G58Q3b?KC^wjdqr7k|`
zwXC;tb~}wX7b4~&pTx>Ip?a!*lj`ZJVnh&s*#Lq?CgV}u09kARx2WFM7!4i@H%ekv
z35pj)e=TpQ-<}1xB#pN2G0eW7DH1B?0!0gO_B*7Db@JE}G>eW;ByE4(VxcF9t(9dh
zny-i5R~N*gS}!u0hNJ{)(z<mEwaIG$cNRCWtJHX<p23zG=(=Y=ps4ebd06GL)JD;l
z8l#m(tV&+QU({?4?SR~=;fh$)PSD31BCM=r4I>a(DgqAujgJ~}2~R}^VD8lQ4-_Ho
z0DCILq;k)20Xti6z~g^C$$VqUGbgFYy2_3YoXjBd%yEfYXbVWUkJc}w@2Ko_kV9{K
zTr!TM>HNwnNa(Y=dbw1zu8CNVudw^o5;B6skHQ6s2KT-J+?7p6&<B4>*Cx248leOz
zQf#ABSlAM8w_%0y+8Fa{%38{*W-(GpG_awJzm{ADc}TzF3T%HE@NaHN!D}jG>B>gW
zCYSC46H^jG@(?vx>5u3XleiYW?Tq$=$4rhsyRGTBX{hphHB2=%NMKn30Yc0buMrF!
zS%D`|B(1j%ntE)5FO*4|v9y%ZJZ&Qzth(1khUeBavD@C~weeb#CuSv)7}F9KRDM<e
z00FSv@J`2rz_ov|S5FmuIgyr%PPWVbDJIs_I!wV+Re(^@k@!j6Zcd)dd`9N}Idm*F
zE{R=Nno9Dwmo#yuI!NHHfM&5{xCYDq6K%E)9qKA7XO6outTVoVkqK04A$=?9_B<W$
zwXAW#WDO=-@~i?_wW6|xT_*b}H`L*Q3OM?4j9RnDx#fSUnv*2yIpoZqRr0<PIbBLF
zn_Nd`Q**zh17Z2U$+ABey4tcjeD^49nwkcZWt%aR$euxFA|MrBMPg7WC2y(8Hy6NG
zJW1(X-=gbjbBw*=m20GWd1|TM-7AYPm~}+k!sws^x|m!NrrL2E)4gFVal~MfNT^~}
zktam+B&UCn5I`D*h1hCRcx}hFANBMdQQ>7}EM0w*Q}upXn3)!;nX4*7MKC&S9dx{G
zHK`6R3G|<xFQ&Xf%_$<zXeu)Go(#$uqjhx}6;Ry?09Z1e?l|0w90torPo8zvcl~XW
zSILy;l$6U)Qk3!eF=ns_lF}oArDtDJvw)1DhM<2|#Lwcw^uA%3wE3S?GStlUwV83Q
zSdhf3%KD;2Ru=@Z1oCaeTNRF%&uj9mr86h2dVHF)Qxpi5D;(=EM)MNhqNv+~ZDJ3x
z0~!WrL*OIisYeYQvh@a8PO#F&nJ}t?IOUAY%#wa13zw5h4Q(8N*eEM-fauQ^^A3=u
zgR6f!c;$4kPgx>U!sI4n9X^j#svlmM%N+o-62u<XII>Xvp>hhm>bEXYml{gA+2~@K
ztn_saAq{p}6;L_c+>v8&Ma{78v+3@arOnJ4-dZ|U>fg<p1-r7stWkhac4EqZn}7%#
z?^EhNl*w|GgE#3eq#1+-C1<8zFi%LSAWwfXYbjM$V`U+m@e9}+;5~m+ndg$zI-Z%!
zGwhc&k5p@uR<Z?-+Cs|=n-W|yi#&j^Csx-Ow^sO5)Qw*8WgQovWC<{8sWR%C8BkN0
zfO2&{(x<31D_`+jd)!bmzPYREypCFFDrxf=s@hRgK*gY8bzuu6s(i<I<SN!HqUwJx
z2)-WA<A;kG<t0r$Wp#ah1ZFwiQRkGTF(N{Ni;{HPNE(lDFJJ%<LwIk|d9Q+=8E0Kp
zophx{om<s;XO=2ur$vs620}z@>eSMCWn7Y+yedc_gH07QlylOQmUO4m`s9?lmSrWC
z>@8#JB$93WV?XKdw0DUa4Rpsa>neY=vJ`@ru4!a~MP(kBVnZtcidBFhHIH&}YREio
z$@-GAqo+F3lO&kc$rY;F@Yw>eC1;5pi{8z4AA}G!5(WXUg3`y;@K$8qdz?d4MbmFB
zMqiaw)A^CjIi5w5d7{(_ypu$x<lL0FAw{+Btf8XoioToZc{-Msl01%?p<jP0L=pKB
zx+HS7+!+~>RFw3Q>$czwN>TJb*_)!O==zf)>TLEnE2C<mGD#A_t(qw$jw3kM*K|j)
z4c|!wFv0V>p9%WcGoPqBW1$nNB$8t+qFR}h`3m5%SR-*E^}Wkz05$bC*a{5?LYl7-
zs(KSM>GjMi78;eLH3%byc^!W|*qePz%9{o)x|JP>rgX2_UpuCsKI;Cgre^p+Dj=kZ
zOqCh{89?U0OqygKSkwy*xF=!{d~BJo+7CX<rlpIex_2PU%S6<4)wLBcq><A`Ew9Yf
z>;z!yDqW45!uK~$yX~Kx(^F-cUQHDwvR(>`sUKL7v|uyS3pMvU8C8Eka!!yea2aJU
zO?bzaO#Ly^eIwWHOPN)N@}-Jc;En)XA`K(ea?H(epo4IrSmNuQ_)D0_NYeFJhEYpV
zStOo9%{?p?D$5CF0KCa7009<X<z^s{NGv1!P<WjaPt+OxepxeARK)8|R}6&FqKzpM
zO9P`cY)QBz++O!Mnd*P9wB;UamwEO1c1IR)^oveK9Zbyem(s;F&lUWp09+)5iwkJn
zU@`iSvv-S>F}`bCk!1BB6y9PaImI{>?$$(&eCGv;YgvWHqitJ~Hctxv$a%j^S5x&r
zjBh0*m6b6^@KIHUXb`v}R+2J9-_i=BSaM5nGRiL$dV;g4sCs{zmRbbMGS#aZA%T?B
z)6_dNMA}I66=avt(gyO&E}I2VFI9d%PYp8XxprwZQ=+vz5G<y6X=G&!=0vZkAr>yG
z0Mc0Lwgb_tlz3&~wRJ{c@b%Hj4J*k>R%%)WWKb@VhAhpzh)ZeQ$Sr*>rwp(0n;<#Q
zl^<GX*{(|{n$&-l*QS^W5b9Gd>hb!l29&m~yBuSES>nG`X91y?spoMd&r=FXX$s9_
z<`usR%6DE5d}dimtLtv7q3LL|xw6QnsE(QxSjo_&fv8@<G27VRYz6PDx|65hs3)q9
zX{hPgu%-zbO&hI_i0{I&V8ELJ$pBw;dT*(E!>VhbY_ET(XOfZ@_0-J1Olf--^?nj9
zdn$r=8)H-HUk17#rxB&9$)L<?V<I6wU`0fV!G+mS{o0Z87aF_sjSH!gm6j?fVPu9V
zK(nCGNgxX_`SX4VBLLSV%>Fv2O=PTwC20hb5g7mwMS&Mmb#Z=3#G%f5b1iqJ6m+66
zZx|2>z+ZoDPTIC0js36R6^eRIH7f)&q(WzJyO5AVufK0?{$E@tQC&$Kg<4u<H$InE
zV!NHLq+j#j_P}Zxo?T5%B~7WIsn<HHObJmoqD8?_EJcUtJ@FAO1azh-A+*fwB@oJh
zs-0u;8=H+FT-%aGx5EDb#8#pRX{1J#=V0vKln8%yY!n-v!PxEG5n+Q=r1MCVMFjA$
zR5Y?Wx)n{dItyI>6VAf-+SmxX?6yh*&r=SQF4IaKY{c7}ZU&MDn*e{IT8@eQ#HfZg
zg_&Vj)E&mHRwZt4U>N&a!`l<o^GNm86{H_IN=RmmA4>&Y^z&<wZh5`#aStUhloxg~
zwZ(rrs;Mt)`hd3<+*<wc4Ct1+hN<U<jijK2Di)8@gsHG=k5=Py0KXR)btOyHYMOZF
zr&-UIL~KIJ%dva+wY3Gl#@(=C9H|q=vN9t|!Z1l7k;%6qhS*gSnhG|kok4_ofnthJ
z{C$WZivUOmlg<H!MMXNr1toEspotvEXqbPqwx9yD0maAHweaINjzgKkUS_9^##_vt
zTU3f~CzyHE*SI{|+ngjTXlKmngox2XvIb8nl~^r_vN0D{^;{pF$i1+JVOg2FRY_jc
z%Nn!!Pb})rsBdH}cVp@|7Z$&@unA^R{{Rl9fg`Abu3$D+Qp*u$1jgZjDs9cUxZZ!w
zypBsvCSyn>bn?cd7@tA26$@gkvJfs$H@U=A&s9+K>Q|cZ30b2pWFc*GNwD<ow>q}I
zCE)71wvL9Hw#^+4uM}h;lrUTBlKg>TVSXE)z|T8LD<Xj@9ZM={o-?E-yWN5w*0|f;
zo-8n=nKdq5SE6{7O&W)gB9r$(T>5{7$hq5YI6L18XHrzlDX92E<TRShXvgkAHr4an
zQk_F!-!X(EGtgI5&6LYMB+~%NJwo{`5yhAV)J2%;7qJH0U?ZoF7+ocx{7%8Za@LJT
zn-&}Q1nu+M*20w38HQ&;UzpO=$0?;rC6Y;<Rg7c>^AhAS-otC|ckP8F59faw+hE~d
zK#_wO+mbi7t&ZewdkunOmJu7Tm}7Y?(D|rs=|WtV2K@a+kN&s^8C26vG_a|XC|)Sb
zM=Faojqae|Pd5Y|?0t>wb3*aSBTnvBJq@lfECAb4+p^xn{{T_GD&GrP9YYH0dFP3d
zEFvmpjhDF8ZT|qNxV4EPcsPHUIrC3XPgX@lEj{6vZ83%_E<ru)4e!VF373=M>7%5M
zMTunc+_aFfR$nn~^&;!@wa0rHxvJ>0^{$QsO*Lgq!XZ$o0F8y2NG<6Ajb_5%JK=oK
zFj?k?Xo3Y@a7E?Fs;L_Kih=4kDi3Z+JKGi7#+XRyK`fqBOf^L=`oVt}^)VI}18;tI
zz%hwsrH!g29<;hh(<;byU6pj)5-;-_Zb!=x4HaE-(^JaI2tcr;$S(R*lEr(ss>fk-
zYu^bx^VG#vRXrNcYS7BcWoFch*9Cynb76m18*|1LsE%43ni(glrMrS8Tm9r_#a1(>
z*FS}?wT7E*z&Fk^{F;BWB!?}}GHB+NTtZ~0Ev_~74Go}=Q7Qd(abk^UhV?nD132l-
z)=ZZ%hK#)(M3r(+E80R`1VoT7S$!|Q#@{7Xnh8Yn!eyqYE2>fpm@Mji#0qyLC<kM0
zdU>^^)2HhnD5}q>-CmA4ujgyAWRb0RUUdUz1F^BdQ)Qs>3YveM)l_ErHekX=g`QX^
zrgxB`yAq7ae^ERREDy0I>I~YLt#)nEjg(Qy6^yj?3si_~MWkRGL1qWkK~2<Laf=Je
zvWm}!4)-D$Bc5-(R<Isu+Q6Fuusa?s3B|W6=$yMWo`!;0(pV;nSY(Jyt0K0hTNh9d
zCgArRTbu<}n>~NabGbzHxs@G7RPvgFw9*Mf9U+`Xp!#^!KmZNxdnm+e=2SCH9U5ja
z4~3{`s+vU#%O&rpu|LW^Lf70_acSzW0(u9fBcrE>rYN32E<pmb$aD*WO0~f@X4Pxn
z{F|Isvu_AGDk@kjtNL~T89d3RhfpwHLN&9N^%HZ=t_^>M$iOe0<=r7!H^EjV4O=KN
z(@9YjdKeHnc`xYz9UxnC`C}~W$`1yiqLQkq>1pfon97<;3a5ryKwwN!vg$=GtPt86
zZbqwIk!jAnEXe9Bqy87uIZ`~7b4fv6Q1v=qbdV$^V_6lwOc)Y)3QdVN#6>sRzar{P
z;*yW5vkZTaE2^%mwNJp()KjDb@PH(eW{d(&+wS#Ue-|W%dsCkD-7O4I^p{)as}snL
zSCeK{cN0Rvh>u!;JtVDw03BAjBFi<?Jts{wE^VB6gOF*=soohGsv(`BSv^RPtfHX>
zbocdGbBnHX;2%L6icE_x=}JhRmbDFIMLV~f9Cv>%i88g=Ft+`L_rCabn|KZ3gqc-E
zH9lEaMJ$CRSze{gg;!x4snUuFU&ZWCa!vxP(;sKvgQ;N-VbxRSHStXHO;1f(98kuI
z9;pqLnblQJqQ$`Uj^@I}mh?}7{SgqYn9&N#1Li>0b<HA?VCx`8a=NYo*b}fkTKU{Q
z71Mvq)Jc~3eOpyhj5Nubt9fFQQUqw6kt*D@i~!Z?7Wv&eAMCx5!$yOwI`X=vSf^@~
zr=^ZHjoFt@!a&S@E6srPfq<IN{9^FmqT+0}o(W~HX#&pKbPEt7!_v=Su1NOO4em)d
z#V)t)Gu0Wq6umLgT`NygvQ#P3S>$$N6>ooaDnj3aPT#IJ6X4H7<WxTmNs`lNGv*GB
z9F@*0suxbmpn+g@0NO&G^n-^G<kIxxe+#9`spwS3(@3=Os+iFCIz@>iNj}6|Ywj41
zX+h(Mh~9Q`*+p#$r1)VHx~w$O>03(QdFz#nqUiuIu&`?>CZ8uw=Zl(4mC)p01Dk&;
zNeqIfHZTTjUqN6rHz!N}8{KR%>RR05zc1;!s`@q(W=w_}I&-Sb%m`B3f)El5Hr4~@
zV@Rj)hQBtBuQ%%75f?DbW+?t$9U8+7YKtYQ*boVielDym2q2sXob%J3WHPo}n8jBP
zWk$;NaY!w8HgLp)xduUFus7SbCL{*T-#MaKE4OU{0#q~w05L$$zqi3P0x>`Z02M&$
zzqg(H2Z9f`d4&R#Fa-b$K=i-247LIbKm`B?K={A67xn_h3%B6`141+f02V;$zqek!
z2wo7kfkp$kF9iTXK)t`WZ~X{y5Vx;}1JN%906IX$zqheH340K?`^E#mF9iS>K<K}>
zy7mLg3%9U|38n<MF9!stF9iS=K<U4?{k;iw5VuQ51feel04PA$zqbzl33m{;Xom!`
zF9iU2K!?A#8Mg#@K!5GLbx<79(>99B;*0y@4hzfT?(PuWU4sS-?(VX<28ZATcMt9a
z51t?)grGOS_pQ4BeP4b5+`4sdy-(Ft&(!qH>8^gxsXjH`6S9H#+fr~)<V6$b#5U}k
zqZM^iAcW#C!D{`X7xzbmTHDv>ztw_jELYar_FoCu&eR%HSbyeF=jxVPVyc%Mh}m(1
zPeGD8l-KjvN|w8UKhcW1RUO<PAmfSaer(75i(rcOjKl98(y{A@VYs<hX{x54C8*P-
z$5M7efgKpVMSgf4Omwwjn*>X24!dW2RpaB;&1sq(8;hd}95w{e?;xxK{mpgPm8CZh
zvl<k#+$(*{mVdkWOxi{2j{5Md0@Z<#`Vhm6n;(yl84C%m-*b{2^&MojeT{T=I5^_c
zEbXH#$|03B`VK5E%f*;}>bg@%MVOQ}kCb1Kqg{Rv(??o&QDMS-!7lzc%!5Y{0JZrU
zpMbsRp|yllFTL!iaoDj6_=A!QR9bb54A0`xWk}gZ)PJBZvr<tNn``B*dssPl>PAve
zirn}7U@nq*6+=1nxQj;vm?e)drIHyDnxvGDNEuyY+`}sGH!@V*0Al)(@i}ppORh99
z57(*&FiG3>5>&6~&o`!%BvqrRiwW*-x{RdmIKbplV9-2ezHG7P%`)`$Ybhyfo><I@
zP(meri+@*}WV6g|<Kd7bVVH7~I;yH+oaFn@zH$hMLXfp(VL9lh%}mE-cKFDQW0w?P
ze6gQ?SGCr?6AnL3oDMW?^ycwjib3vOs_>n;5T=KYXez)Q;2cpnbad`~t48)&A05M+
zPs~F{nfbMii0}<cEB~Fj%KwT7HOd*8(ISugkbjA_oigiWAKb$~KOc_itqzo`VMy>=
zgHfD`Ew`QO3dS`R;@O<xc*$w-ud^*VdIp8|jzy<C#<C*+%c+uI!vRd0>Cx%84uUSE
zmCogax?WzRKD-!#dM0Uvk{?ubiM>^M7jJOOLHS-)r3&Q4b)cG7`^OKYb@RcP(iC;q
zSbuXT1aTw_OORl6++{+^MU|30x~5UU`+)9NgUuvfT!A(l_4OJ$1;mRJIP~hg;>bH~
z-ue=>G+cDMk|~$(`35j6;^UKAbA^wC6`Qze%cxRuX(DZ^4d$`9|Ml_5Ja>_GRui78
zG7G{{s`SSAM^wRq*#AdJK6=GDoEkk-C4USlB;_GPg^lTsC1`jpI*%Q)Tj29W8qFDQ
zLh&LG#m`trGKE{+xW5vKWl`JK5pVPYx1uvwCm;=W)mq0AG}>|7bz`&pg7pkO$PnCE
zx2bG8NDbGek0pIHLkLALWgfqQ8^_@M9rdpJATyaP_Tyk0Fnt*<`4KvOj2m_C4}aAV
z;pwxJd*@ql8YB#la*N!g{Y>s@OcsOoRm<y#>%4%fi5;3^z6Ky>Hj*Z98o)gUsO03D
z{q$9!bOKKgFfW5>{r0H!X+fWKUJsadGjO#A6WOy8nSXSPK4qoTRtRIc-Nzy-Qj~e5
zUzsZXm*PIcFYxRkdM2jk;5)0X^M4o(7p%tpx~LVPimFKq;}8Fa`&$!I?vrN7rrDo$
zI5kj3+B9mu5vH#~Gb5YA&4Ib_i=umQf&}KTcq=ONk9j{=Qn@8Fe8;GU4g$-K1&k(e
zmWr);8X}EKv}_x6`AKl452k*5IF3HHQQ$kmTMk5>=EPM>m8`VVYVhzp41b4tjEr=!
zhKnjH4xw`F+s+$$LNEa&_$Z!po80ueYdVr2j#@#nJq<S2vHUhhbNXr3w^26bP>!HE
z*!2{S`f+<1_HQ5a=PZnUw-%+A+lRNpZnYDD1JGBST>|G*20#q(TJ#fXaQfA{vwk^f
zJtvJ@M)4itmJdCtT-`*Hg@2&KaPtz3lNlUj<l{vOFJih1)ojArNwcBjnDQ`Gp3b7p
zjVi8nFwLSjowpvZfiGjB3|k*`f*^fX#<M8Ffh$u*=zz@qa6LXDIQH}0e#JG!7EVYi
zeRe9gMeJ*)c3gd{f5B@B;=<-~WvIlSzIHUO^Pmx8Nj}Fl1!WVXet*F&2)V{+KBn!^
zAa0_#uGmevdoiUn@l)%IqdQQ4KOF#bNXcWM_wlhH`FW}vH!4+~fVZf}rBD9d6n`l$
z#6Vzff96^W4!a3M(%Jr3`0n+AL>qZiDOP~lijHbJJ3fSS-BzxJU^K<XG=0%>OwT;U
z)&&JX8OAHE(JkY$SAPKMbw2L;#`;2v>cO^aq&U>R*?2zQe(LhFv2qo;@Nu<#xr&1<
z^ptX^r;w}}51-^=ra;USVd;F`w>F%>mMtsauK`jt@;lEr*cmSntyhpAmFo^Ar9|oN
z#X+F7<U(#!6ItHdSKeJlb4Wve38=}e5H`i;(9^;qRGE_MJb%6jeu<h1H=QUaoEZXz
zHfvC@)vs&GrJvU!nO<o<a7vPgWorD^#WF;XtzO_4MIQd<s3;9=F5^sOK&&0YL{EI`
zuaV@X8ab$I+d8&l6>CHjdzXk6%>zD{;U}R=i0H1;VC8T=Vf1K^AL2np$rc=^L77IX
zt5j^{67I6x=zrhND%}lp)3|{}7a0CPu7<y_-gdJ<lK_k>OsmOhFEuk8CV`lnb5k)j
z7SoswvMTJGBg}Mk$2}<0JC<m<D!FsRE+?7~PP16fYh0_EX{&20mW`whvzfgn5lyVz
z*2!w_Yf$P-MwZRja6OWmFqT$7Z_}}ry|qX7a>bEqYJc+da1Kdx?5sLZHX%zN5t_cE
zx6aC;8t5ta8aULf3)9RydJ{YB>EP_%-yMJMi~D$ZEc!vs(V`@U`bVx{W7?IX)RW`Y
zrF_7Bo9;{=nOhR(O!=Kfu12W?DBrkCp+nD6W|Ny^$#bG0JxJKi`p2yCEKi$`!OTF`
z;*y-)@_!&2KB1APigYSZ5=a<UH~FPr;S5Oxk+KXMI``PyvLHepG4fY$DEntiu}zfQ
zVa5%uGu{=#lN2|!^wrz>0lt6H!9I=;Skx}O6Pv&Va07L*{WfDTWc<P0Q(*vOCuuLa
z2|ElCLl_sZ%FANdAZ2h_KZ=qFC(2l4NYm<-27i61?X8jY;OB-uPMO%gg<Udibx-|_
zlx3dY`%D&R2_x-Uu>O_RU`M}$*OuTKpT(}9G`hybM*?o<Devys6ejNtz+d{r^_R}i
zP}P?48<t|#SDe<7JQ?r4uZC|~I@A%ZF~Z3BpVZIhy+_S2=U$gT5a`}7*(g>H=EZ@?
zw1395Tx2z%CTsM2?d9Zcp9xH_a5G86<QzkBGBleCLqfj^j<ayF5_lkL1%>?ih|jWg
zQctsjzf{p)$(l{^se5gbXusZc;q&8pt;p}0VWaQ9C4H8u3Ii%7Biv>u*B9$W<sMb{
z$o5}_v4!p)bkm!*Xa9suNq<B-io$7T#D4|tCeBfY1}D;g$o0qb{*eFeH2&1Y1{jV8
zyA0`MCFgXUy2Ok1h|^cI>p7NY3j82ox&94C?+6p&@s+1`VF}8s%l_fgY;)0Llk1?{
z?8VLTN!3UL!q|k<ncXhCbfhaxw??$z6H@U*7B4f$9<!W_IBU>8|0h+<W&1b8+J8Ey
z%T%Nc%DsY@Y_GR9J%g=}yA+%kBX?(PL+Oo+ZeRO^MBtO9u7D_EPM)^tfet3sfRuD~
zU=BB8Pdzn%4zb?o*i)ZpF|JmN=cTI^qdH0dNd~$kF3Xa{5}Aczs81$RlF!WHGJ&uD
z{D(r=qBbu-(yI!YVA92&{jnBchkv$(QJ|~mY~Mk&@^<f83%c#~iFXcp$$liGPIwMk
zPL279IM={#LE%`Cek{p4Rq@@_cPXk=>3jCXqlB9$_4@LbhSA>{fXvc=De1Sn;x|{{
zQO4cF+31D4em-ioHV&fZDZb#jh6!Q6TJzxuA(O@s$1KLP^8lLX_F=c~l7B1X+K&*i
zA7r9^wK@XO(mWR4Z9(zu+vDFOmU+;EuZ42eS3RKYQD5Xaj-wB-6~=)_=ff}-so@=n
z&yV>MblqkbTMR!`vR{z~F2blZm#^V_(CxrA&OZm*^o3lUA#T5gT-Du34Rs3mKuNvI
z7Vxdt6mu5h*kVICIW%b3!he`-SA%X<3(sD_UkqtVQV}M*vFW$M-W(wn!k--94T2NY
zZr#^65aWl=6aT0Y`xoRpJ*EL8GThBG8m9YP%vhlmm>1HgW944xN&G=q+o56lY_jdO
zn47L;sdOYq62+Mqk(hOaY%DLIIi3z@wJBD#L(n9Y+pcexFjO&Xb@o?aGH`z%hvj>)
z;Q8`NH!0UD#YY*f$4<U^^x$)eq&=>5AsxDk4vZe=3Ic=gP*-(uR|WyluzqF6GhneX
zQoM=ssqK<FEg!q^==Dte;cHOYryrtT>)IB7(=Ec4*k?-O`K}`RKSU@K!pz65%dj7e
z<k8tV?1$J~khjJf@xpv8iV=SnEri7S=QdBJMWwim_kT1Eb%_oK?{O`XJuAb8i_Gu6
z5Y7tit%n&jW)~k=hjVtEhzih|r!O??sUxe<7YjJ<2x>%p)jdnJZ>ce(B69zLp%{TO
zK`5Rpqzg1PBV*CxKXS;pA8f1%zuX0P53B_?VJB-npM<#-A+*RWM5ce{+l3yG)9g*l
zdCUl`zfxWjmx=`nXoxY8^;Dv@$r;#`Th68eXbovft5_|CnWps*G*_X4J`+M8>uAUh
z(>^!6Db&<gW2^jPRk}vKPh>C0EL`)={7L_y$o<$6M#iI4U!aa2SBmIBkh+DB!Z}P`
z@XoOv!%%6(#x&c#Xi9%>N<;RcO+BD~LB|bejsL5BN_yR<0HshsTC=LE%vw|T4maNg
zQGAlgiPOTnL4$nh{zP%_7`$}NjweroQG0IODz?GHvUfTIB_KduZO4)Idr{<D0XhED
ztoJho0?QE}18CLu*)84bW+M%cka_mH_p_B)ge?UaHYB>6Hiv&}+@uwWwPEEoO?QIm
z_x%l|%~kXteclG+Jb<z{STdCj0}{w4U<qTi(tIAw>kd=`*)~*ibR=D+&{-PAuvJb+
z7U>jkT)}O9u&Qi|dQU@Zyd{7_bUh3KZ8?KkfhRC*Yh42q*MJtaRGG(D8pzP@T~}4^
z^*~3$v&BC2HH&``y<Ae*3)fe}w0Wdm$wTQFY5?@|m04!LU@-{q`!W;CR)5x<^!WuK
zg1}PhKoNr}n<+~<HrAS?yXVfzbsjguS1WRBtY_grFPV;|sfU$Rg4`OmjW3H_uIUY2
zvEM1mqn~Ece21k7Un`fQ^`$EGG<@<DG)_=#T4xZlMh<_IQU!@JIIuRxn`1kEakZ~0
zZ6lXcI*>#gl0eESWk&nL+HnDr4aJE<&_;sNq61ZC?D=JKN&#6ZQ-e8Pis^p71@L`z
za>ub`s6#rdGL(L!|12Bm(2-ydXSIaolAil3;WG@wp@ujhLsSoLC_NG=STWHaBnmMz
z_y|&-B^-Zr7@LNDC-*{nYAS~+)0=el2u)WptxGC@)d1M74abMhQ*BgN%!5=Q%|@Z#
z4yR3+tr?Z12r1mF?20Uk8HB!A{8=u<A!s#8``UZ51639#WL10S3oK!k@PBO{Xq#jf
zHda43p$ujLF=Z|Cbl(f<NE5h6u_Q)|Rs(oGu{M7h*eD`;pzV!W82~ilSV>tPfL`ED
z1`T(Tmy(NO{y{YO%{aqZTTmM<g19>a_biGthQ3@e#bA=AWIgMb3C_lIC-=GYS?(9s
zn%#L^#p1+Tj0z<GlBkK5JbDjF`x2{?I3B{TFQw>+M!||Q4dtQEqoa*&=@nR&-JI{O
z7;t}mgc>wbT~O9gIx)!cg$~7tE$Ty;!SnM7g(1F-*?dra2N0;#77M9!B%B^>Ir{}5
zUegTEL;NqqgApK*ximA`rHPu9x>9&sO)ELTG$M2sO$+e|@28Qbwb6JOIu{z5X{cSE
zMQl9C9a+*PIx=o>P~>M?6(JvHjNmrQbxePi0Z=Or#3BSh!1_+To`IqqckY!(&j!|(
z)Rg^ljYX(HgXfN+rLwQvo>_wAn+U3qC?bIx8Z@Kp$C3Vlw4TdYDio(9rEOsA#7kL2
zPFWIBC5QN=YZ=jckUXl?g{N?&WzTC_2$}6d_V&iY91$0)m3kJEY-2;BxR_P%G`)YW
z(=@19K!TDoF+iqj{dB#QOy6Jnssovh3LxuspOpQmuqjEmXkCW$sNy)l7T`I;Zn7to
zhtZLbetx3adP3-vqWO<_81n_YZmma1CIfJnr&$J=N3Tqva)mN&x~h={H;TET;rQ6C
zXoFs5s5>MIuS}qW(Wo>byuKcW-5!5B*cxV<Z4FPz!iym9P1QF(xEMig!7+3jp&GdW
z72^D;ky_)3TOCeFCnFtZBS4fz_fu2j{RqbrvT-0qU(%Y!k`x;W&|ZEo=$*OQu$(2?
z_yJ2y%yHIK>%l65YQ#D*Ce}Umy7Xr`9hXuBMp%Nsh$B;JGQW%@{50heT8@7cfYUFB
zx1gn_vRDzJdZayRWuZqZfM#_=0w2|j)md~tiKL`<FiyS<lb2mgN**s-4-tq4+c&b`
zuHo_hdQI0j0)vvUXh^KI>bGD=d?&=JKNBS+JYt%V$MuB5Hta5#7c7eQ<r4Xa)dvhq
zObEd4@t{)B!U8WoA=X+*=p28_;=n8NAS;3yvSs=OSh6WhP*<4+^jkO5g@VnZ+N>B2
zdJ%40+*MkemXIZMO?gQ!8QX@?;$;xBTlGT48;*+<t^ndgXBHyV;}8=tF4QIRPzjr;
zr;}V}RKx}INb;Y_?ZE-fEHnxoq3E>c?bINV1v`MhM^BtGQG#JBG)8~SDo=KK5ZH`;
z*OtWju8t5XI%%wp!s)w(qLT=$T7^x49<SvX@VT$)asVH1;P2TIAdBA~njy8|soPV4
z2E+Cez|=>ar3<xJDhUs<6^O=3r`LZX+Y%zWBqi-ZljdKy#Z{p7N3I%^IE1er)Q~F5
zKDk-pv9|_)(ixQ~k9U7SiI;%LfGY?NXb0Mn#MNnz$&zB$G$g0Mfw}E~5wBE1()HMG
zY<ax4B*MXXjRip)qfljY@~}}j6zX{`Jc&>7&Q1x=nh<3x+X*PlYCE24@l)5iNA)I!
zC-X$qYml3TT;o(RC-Sy)(>z`!atHY{BO!ewyc#DrdM#ggSjc~O^hyu{>VmW*?>LoK
zo1mPO-^7oq8Uj3v$&h7!YeBK!WV7agfkik^I<-`?*-PmfzHt^>Y^W>(Hy1y~31XQX
z@~CnW`Ml4`iW}R34PGyMc#N<JH~s(-9J$=_Ab^n26kb_Lj$fO;G7)JfJFG7WP72wL
z%E}m&gNRlXJ1>79ZHx60<eUYL#P^jpEYgGDS2T?mwNwZDX{RBXCXfY`zy*Y(UP;9#
zBZ}Y}EMoDpZMcV6Se?QuF=(IUWJ=BZLwe%c!Z;8RR5E`*2$=8mSWcLK8yg#}>WNXY
z5=M%s_tLKgfppQGiNXE@nKE-~CB31%=Ii0DC%oWqGP-|Y+{jRd+Uv9geVUvq7Q8hs
ze@CEzOLOtzQ1&1YwM8Qpv;9fmn&jc93`AxQB?-4lK7HL?$Hvt;zDeS1gVl;V(#BuY
z$#3t~p%@c*1A3Ww&Uyo0N};isJL4VSbcNJUT~k`l*13v#B@x3))YiJ{Xg^Inrg^81
z6AtsO1*(6k%(K*;Qse9rCtk4;XWDkncg7Ep0S9iMvW%_OK4tWs#Do{z^MBHAp+J^l
z#R7967pc^MJ_c~XfA+K7y@3~j0~+1G<I!K|2{VLMhSoPu8P@gAWpC^rI2pT<DpuS|
zw~h64|H^?jV80@@YkD~rW{x<KZK9B5&?wQq)n|VT4DH)D18XH12U0<JRwafIW+wyr
z0ICH%f=wYD*Zp<urVTKhlrCYYy@M=RM-x%cF=iC^4ef^VluOZ^BV`_6nmFc>pCB<?
z*O3EcTU*vvd=^!CD&#;tgJvbHoVW#NRep8z)iDcep5RBx&F;$AMOdZH!rl>K)2X1b
zT62G8{A0(CF|W1ip0?2XPZCub>;#M5q0Mm;YG{R$kXzN!00~YgbX(jP(5PH0l@x_c
zH3-ag$!yE0QDBXgkD-!0JbLKcl(nK6x#~c!fX?#R=lGlr8_vSzq7XIisNeiL^<;4u
zE)fbglz+OIU3&GEiD3dJE;%Li5^9^T<hg$%2~nw@bN4b@T4J7l8gC`@8T%GpKv&Or
zD<r2$`USY;Sm#Zm?G#RC?mT&FIb8+qHHg9uW9=O0tw_Cv=|AXCB*i#V!MkeSebm7m
zO@o?B@aExdgG#-JQ)0kLGOd89w8|Kul3yj^7<&y6?>1pCsULc8lz(;pM<;{nXMTT`
z)i*UnpQ^tWD>#qPM=W21AvYK_KF@f7fFWl&2{N72LiGy}$*=KbQ=e=??@*i8(&rX^
zwIQ419QC}bs-0b7+^A6x><E5|;F#Kh-&Nx7LcK;YIT3*uA~w$Z<*%x#*^N0kY0W#D
zfwho9k=FsyGfcHHnhq((Uuk9d73P0@78myEG|N1O)abqSw=5T<R*Ar&C2+1NllhC{
ze4(#;9U1ZCTqzk_&HZEH4c(WYyKhoNU6`}FpCs;j9U;jdA766aOXhR=lnXi&js?^{
z@6J31?%3JBh`%J&N}_q6$1d_9%k}c+w-$Z9SgDo7JW%2`!jiJ#N!P1pigSOemA1by
z!w3r+=wM(*3GGU;S^-5=ks)!QAI;~lGlRByP)E3ymRob;wub<jgpI~%>3r$_=!|Gt
zY2?d}HOi(2YFXcP(lgxjkZCOzG~3*eV|rOz^{Jla*-U&ZwMS%nSX={<FIJxz=%>X>
zQYd)kJ>v2RY_V`^1;DZO4zzz}EpCcf{YQ(zx+SPAtV#m8<?M2?rzy&V;3pAP_vBjE
zlVi4sZ9N?r<jNvzjZsHD6Ku^ciPc_Zwv={(t@~TsTvjJHg@B)?yF(^{vDtYTY4@*E
zlJy`{KsP!eHGzS;od0GcRFS&&0mQ9~R(3$%+aT*;O4H_5Dn+hZbq#-atj_~;QstQ1
zFoun@iYNfmwF<I!S>$zp=7$mM1lUVyqGqbM54{9&$o~+Mr7W)+m~6<T;@2oKmkPP}
zQB3^qoV6+{ixonc7SlVQZ-~9O$Yd3@3`^@X<@o20H&3BXJ4&;aB5TTZ>k&}z3$}It
zp<g(GxFUUD4Wo7FCX9c*%3zhI##QP9y?8JJJ-JJY^9ltT`a*V;$aPeFBgL}>yLS?7
z#?hDvtu_Oa!v52|Rcd7GIE}J{wSk_<wO_1=E(mI4Mn%ybvcPRq9c1EDI53i)W0o4z
zahBeb9n|C?MLbkd<hEFW-N<;6et?f@+D9;B;MO^S;jZ9ISJQv%riW6b|K$uS4vdG)
zkLKbTaY|+}dd<oonmq(hSzv@S1{w7CXk0!$GDV&LnUd5c8s++p>^F6%i`jJ85x8cq
zR<KwowyDwBp17<2t~OjSyoy7sw)K+^#!PkMaZK!dzL+`o;~+c?0%5+De{sf%w?mGR
z8Ew>7RMG;HAYgw~Y;oHDZfSbOnot6H((=k{q_1G}V0HmubiA0envbP|hm*?+CekSS
zds0WCM8fnHg6Qs-ak5iS1j?0%f~}YOtEyHsb#Q&?b*oJwrIASxk??bWes!lOWM7mX
z07l($)z7y^DZittz!iP4LkL-e8CE-{v5Zi3Tdh*746A?Ddo3^?kqN#Am8x=8aNF^@
znZtwzm4Uf|<h#O?^?cO$R+W1=t^7;G^MKqlMhOw2BFyFAPi>6ftZ<Vl;0wBQJ73(p
zvjsa&Bb&e{IjRyqUMw`#6-12Qm>a22aJfkHlQ4&e>OC4)2hZLJ<b<%ilPm|kUPc<R
zFR(iE^4)*N@0}W`H>q?Xp3TWrc4)WK?}C!d#nAY|w^?{mkGX+y6lJoqYN9#vdL(rv
z`W_NbTc^iB;qnB`V&;DQMm7iRz6KW{vU$uTu13Gvw0jxR0Ue~R0He@M9JoK17RmIe
zM;bK!Mb!8BWX=0ig$Oy$D~S@%%<R>g2SpXu>jZz^o?utXA1Aw~7&~q!Ly1oP(I2*O
z;&MYZ$8zblF#g2HaD<@pqk&`=;kW|#MN$JVdIN49Sshij`r5FRcHZ$-KY5=;t>bRE
zprd~TdJIlP4tngp3?5MD*t0WtCxOHNJwo$?2{4NoG-5cAtcCHQ81YmrPn3LsN4IB3
zYaf4%6No!6z!(k0O@Pe&QptA}K_2ZaOmOHjh+w{TO&XFwLt}so1ak_<!tk_8OZlOJ
z@;@;iR6b!hwhVa0eOH>SAR&%nALp|BSxkXIsa6u~%3dlZK_SC7bnx{Rk30JVJ1(Cq
z?g0VS;XESmcC>3ljKPKWwr{owMBd7v)J}iSc;M$u?p!bqV5HE8kTT_$A9?cTWJy^|
z$aJKnra_Bk9)UIn0Z-wOKxmPkM_ru!<x-TDl&`y3tOMD-6U>n)8XG@OZJ1HG3n39o
z7vV!&oIchGpBHG|uwR9%p`(02|DSTITW%eN*pP*@!F{^l=qh=MXswdI-OQc<D)N5{
zDZ0aFmqUwFi?SO!8iwmhq|V(&XfAL#IRlNB@}&G5kLcsTTj+6nZamxz5YLJ+z}&f;
z$exh(k4g8w0&rfiM2Zkb(XhxKv@DsHfhJ!td5Hs{8@)#t&AjcvFl65~kEWN%0wc!5
z66jk0v7?--xm?EAiC|4-ZW`CgCXs)F7kuVNW|*27l^HT3mj)hdGMoNfN>o=a^Kh>3
zF`@9)%N{q?=uX8WNf9r}ycUzlx;~l3BB`7HQ}=&r(Su!76k}!6AN>&~x0xDJDgMYD
zvlR;D5WS)>>Xm6Sdex+n>IC(#9*!4lFmek{S=%$Z*<r3|P%d~al7@EB({O*lpYiZ;
z0h)(}{pB($XykLP*>70@(&0S10UK``m4?hq17mr~Q4@g(8L{Qtu4Gg6%M!Weh9LZK
zF)TQqYQzwgbcx@cG6Y$c^6DjbWqp0Yf0*zZ=*cH_64emfU{P#}F9@+yM+b6WjrWcz
z-imWaewTMnJ~UE`5$-(QhDU#qD1?*C;~b=r9bL?9KtJwmEA96KNVRlONJ}Bu@F0Iy
zMJvtAYXNbu!7#XQgHdYql3_A*q`;#;POwLUC`@#%)}S3iEZs?qK^5?2mxvaAhX?*O
zHOoTD`}IcX8RdAe*Q)c~=VRu3j)dfE#AdV~4k_lHgfO_j3bA3l!$E)blhFc4_%y|b
z!-(+FRAIih3z3t!Mxt=iifHy!M|~LLyzob<3D%(Ge~W~G2I`R}oxT@vrOf<$_noa^
zq2hN$**IUXQ*q7NO{5TcqI7tXp#H2wInfaLM7RZRgY3P+hVM~-79~aS65utd*4!oz
z1rbm@p;U*dnGxF>hUtIgZD1;>lo(UmMXC&%EFAeW%CR+Lrv^G-X(KBN<v<$J`NE2S
zp~`=%_)7Se%jDjlaEeXjb{LcA?*FGBMi327oi2{cTZHHq9y|9Amoj;Y9Ov=WpJ#sj
zYRx~Nl;D((9`z$nR7qY|Lv9w$l!u%K#W0W{lwSo(CtPFC%F}<jIyS${lHP4yt(*}B
zxI24mehU?AlPC)1w}#WDN9RcU9AmO1LEVRPTDFStu~nuxc^&mGm2M_l=r)K#*=w4m
zQ}LLMRiv`G&)S3s5D4sItn@7-J6P(hwX!rDv71AylSt1H4YzJyPl+wtRoO4d2{@s~
zdT^xCAz!|rJ57I1KVY;(C7BmlLm|a45M6rEjYcPUZ5)hTKGk;;>HUmX*n;n2t1`_Z
zIH<^F3bIE4SfeRQc@GDvE`54VDUqldGK76x{IC!F)WM5_(Z#_1Gx5tGxBoPkt#hL(
z*fgoX{j?Io$fCV8Pv{)UH&e?^M4loZxUWW<6fAv^<)?p^orl5`@GM7cNx1C0;37sY
zqhS^hjOmx`OH*(?)5LICNK@qGA>NJjqF<+2&_izi<sakm{?{52Ff|$)Y>NYI+MH4W
zEn&be&qs3vKb{F4R&+3JlMv7$_CJ%joKuc~rV3Ja$+A$-NEScjN-&P&?aV7hZ`pQg
z-IJ0Rj|+c|CvL-~O^jtk?k20#(ZKTyLlMM?Ggw1VPD84YagQ%Cbl{bW@McN}$^4t^
zhf*#P3h~#Ay{yPy?@wnf*XXj5PiL>d)_89&ynqN(4K257sP4#~21ra0kdZa&W%!O`
zJR`ByMUYfFi$a!wz(g&4i^IO%b*uf(#G2@!f^&a|qz1$G#q3_3po$2<+Z2jBKs>1d
z8r;i^L?DGqfO?70tmzR#A5FQ^X>sj^6k`+CBM$9|=s(onM)Ioq4qKaFfAoupdT)0!
zgH)Un0i^=Q3dZuo$WZ9-kR!+DFL-<{I6LX(LMdRgjA&-nBVJ%z;^r_U!J`ApLPDf{
zI7WZLfOxxQMl*nkL;{q7Ao;5d%YK=)>YMIE3K3<jroYwLyp$MR1g(4s1vNewNQL)K
z6_NQ|TFz1lH(d}hsNa^w%xVIcjzb7PPfbhd0o4)@_1U+Rc@AHdgRGl><Glzva~=Ir
z#NwCym_bEDq+B%`;>r0jOo;r`TKamq&5nP$G$XKBRzh7=(XzhDOnFfhg~;I&IV5a+
zNJ26nSIEK_b=^7%nG72{mW6vEvbV{sHnk6%uUz*1`OG!I$!jW%dm_h`K`8rX*!{(6
ztPrv;9H2Nrqp5TZczQ8RWcb}g!-1eRzV5mZ`C|ZK)AJ*>;GSe`7X{UJxtoXrJ*$6W
zLtNxPJtinOZLc_&3P7W298)qidTdzAQfd~|UTI4LM`-DxCa)J%MKOc{A1?Y%ISfS+
z-8*a<CS;*uo{~y|;UP1rkoU#L-l0G}{tF`;3?93Kl{Lwss<bebLo{8p50PDAvD??X
zs;T&u>xr632N8n@7)qp=rlQPZ;`)Eg0VZkves6T9pISCo$t5_EoR7ZM7O`d3#HA)o
zs^42DZdT{p+u1i1C1k0r=n(7WwchA_kCbd`f80#aaD7n9g3Wwq{ZuMQbDbWVd4Yk_
zLhz=C5~=D|H`y>hGCFv5$Bn5UYKP!$Mi}s(SKA7u(I-~uG$licLng&5D?NWdVG|By
zaJg@7a$oemI&E$oL|zM?W5IJ@Q@=da8Zw&|iR)>v7eqjI55gCD9gu>p;OGRVD8;lg
z*$Q?PShhETxBZXu!PFj$NDzD<ZjE{^4-#Fe?~PojQw$xSP*G-GzozhuRwWbRew$>6
zi7EHtsC+LAE_u8NfA}lSmid1z!68%B%x7f9=IuMt?>Rx6bl0SZ4<TPo%@R*LWHE0O
zBR;#nTjNw1X&<<hh^Sj_raR3)`EByy75@z2l&7SWB0$f}A=G}WwOvRs%e%z8DlTMx
zN&ah0MiNys1nG1ng2FY!HXcdO980neJ0_Q;Ano0V$oUaSu!eL()vA9v-7v_3ozF%%
zpo~{xGzpKq+TgFj-(Gt%7V&+aQLXR<yG)Os8KQCeHFxu&yHZ97C5$gqh1!;E`^x(E
z*Sr(n{*PdQd%AnV4Y$Q*YCPd%yd|9#en~ZjGYYr5`{BBR^cxjDvhQ^o=Angb*QmlK
z7PflyO90uCsSSf0<jj8zPm&>|S{r+*aA6vmehBOnL(M2a*8-zb%N+Vje?w}kgYi@K
zZ#sn3m!7_^$b72Ic);{^>to+yNg9JJEAVeJ=9+a}*;<0tf@6|$Ez09H-Iy8hb_)+k
zx@?yaNDX^p<y|PN_(As|@J(N=6L;PHc@Lz-O!kBHD6DKW@HBrd{=Mcyl$jyiIH6DF
zpO*=L+SrGw!R|ALJacCj*Ty)bf+c;K)LBb_bS}jMnb%6mw4GenWj=ViK>fY=cS926
zVQX}3E6&3a+h3Kdf_><`+Q5GL%RiJ(j5qxnkT!4HtTXS@GqpE_1w^O%a0<Au?%}(_
zgDHr1<&Z!XrrUq&ylkONHPuUY*4I?^2S3MDBWLouuTGoxL+>j5iP}DT)(Z*G*Xr|f
z#176(e_;f)4;nQ)m&+QcIXP@-(J$e8L1(AaMlfKPykE`LI*aW4ui6ILgO#Uk5>(4v
z0Y2;{f-CZxlG+~EBa!<0mpmpUlCmpYZTev;#?d>bpf7)9R3Y-oM~if4Waz%)H%vP^
zHiVxV`?M$m=EQ<Nvd0*f=Ypm*MKzdCws-_6h&V<y7t+;B(JB4wlB$iujZnMh`-Y4)
zbW|fJ@B`SbH!WV2H2`-&RKZ!FQ$pkZ_sVg3VKmv?(1G$=v?Uaz!Ku&akEGT5X`ebt
zgJem@`}cpQGqf063Mc#Gj+I6*@B99Sm&dX59<j;C!9sG$2Bzr|XaJv4j53TlV)E*3
zQwG>|en6YKqlkH8Snr?Lkz1E19IsuJc<R9HJdfCFF{T-@`};Ciz$Jz%nxpioMz~q-
z>MMjGm@_jx_hHO$T+V&a5!|@q50e)4N5Ua~aV&pj+=4Ng_j1`mp>=(K$u_>LxMByR
zYQ)Fa<z82cR41tZ8kooRAv*i?`3tWQHWM|whtT>&6L)M8?sS1N7tM)9^}$0p(&G&T
z#iM=r#}8Cp4dDdtlv9Yv+Q}5bu4l1LoRn2+iM($G%jN|=H51uK$n(H^0$fRSGa(x#
zidBD0D}j(-w6^P}od}3=^W_$=HE3aqv|N!S`RvenVnxfqv)Cc);h^2chs9UBK<M~G
zTK~JTcWn#26Au6oR&~3Dd9L4dby&qHdw4{91}76b2gcLK@vXz)Dg!!=mWjT#^o>7C
zW4H_^y3Sjv)`}~H8{F(nF8V%(OUSZ>UPgaDUK{o~AB*VoSuX$dwqHF_3u1olqo&r!
z<IN5`%YKJBEpUwS7uTp%k2_M7gpg{L5GSDryd8q^^8k4Tn>owoi4XmOpsH57CB)cD
z4wpk~dRZsF^wq(G49xdJ;Yvi=Rm94BdAyZu-xx80nLCMzqF27IYz~mQC(!Q5G30+v
zui8-HK#SI+Lc`a5P^@=|jZPsTB#Uw!0vqot(N;0n-%QF9uAK4?X`%8~@@IW*ykzRI
znlZKv`23#fw@#ST0gdFhoFyXptuSC}-7J2Xj$}0eMo}TPAQTs;)s)Z_>+fM<fPo+<
zoRCX4Kl@A7frlT3`|e{Acio)Cxd4Cn9v8aIY<icvk0`%2UKBZA#!}@NH!ME^i}X(v
z!msST&lYuXltJJ^0{%<`mdhUt1EtY}ewg?<m;OP^(C>WkpC_T7&Fd^$iY$^>S)PaP
z1IrN{|M?#b<9?vnF);NzNdr%G5A~Sx<W)B^-g<&5FuXdtq)amG^wHl?z3P9{Z@E*4
zmtb3G&l&yS0XAcL<f`Gb`Zi6dL+-z$?F5p7dxo6({a^7ZTUEz<W!$jm5z)l2Q1Cpv
zhvsZUTs6VWS-nHiM5TzhjA?24D<zG})47j+pAck~mKigBGdn*_6-&bp$jHzOe!EF8
zS@do7h(St45D8URl~?}e{pf#LWj|C4<Wx+NeFQUs%kxvIv(zkDEb%m!LzK!%X!nJG
z-c5{(E2+2#q?a4&^v?soYM$!7#}B7Eo}a16sKfgrIWLkjo?lc=DIY17wzD0RX+$Ft
zzk%u<Gf0K$dvXB>kS!hgqYjbwbBx-I4$-X~H|^)9D-JV2u%-&gKD2-9TtH8a-=Mj6
zI4nz(<;I7@6H(Edw)i}}XN^DX>~O=-Oj~Vv>n&ttFj+py7?>!zOfPGd5;jPvSK369
zFyCm9nOvbV(2h{uH=?iknpOfPh1yCTF$*|j%s~+IhWi)iB3BGg4?-gM+-Nh#t>psl
zRV|bUXg1kGXotP+^(=qoXRV~-8{E!)5$y1rm;8y$XOIGVhNKd#$L>rb#<qE*p*X|D
zEh%mx9ODTb;?+fz`fI#&?jh0X5ltT(-!HGN$KJ+Az4TcnzU-bF@XN@v?y@DCxbR|$
z|AwTMAhg?00~;M{DiyOF&cE2?k<MxnNN4$}D#eo;Y3d9*6R>|4Gon{xKYeV}41MjK
z-qQ%(%V87w1M*0dLZa6sDl9E&vDN&YaZvSr&^}&b5xG5q8gS)=yKKE6AH-Rx@iYQ7
z_N}9n$rxH!YPWBuGW5j;5*k;=C3o!#rZc;oC+$PnM&M#88Kcy?xPFYI9HMi<njP~B
z<?wvNjs+N3Qcr);#<f+YzIhzfN1NK6zC{s^)4maicD;JzYI;9>#~+trRC%Uil=i)c
zi<^yB#}OLdo?i9=O@=)6%)Q%!AJ&(SqN;xQ$=Sw0BEb-pBc;l>OI0R^S2Ejhw-uVI
z*pz<l!3hnscP}$Hm`DAs_2eCxF<lf!r&ES65*DTG$N7Id8vX5gosP>@^yxY!_*CWB
z01rVq3aP;fd{-E*z)_@Iq}D}-=m>qGWD0HcrqoEgymW*Vx5^62C$~~<A@68KcS(d@
z5`>m^)mGe`@>gspN)$dezkX?c(@yh^yY0_bcJY@S)aSRVYutk3_^tvxtNl0cfS(PA
zJIBskeRqG<;UNAa6t3@K4dLX}`f&t;B}nub!c9NUZK35n8@TDR&JQnr6&%*jIKB2s
z9Oc1{&Lxgz4q*QU@mB|p)6rHqG7w&RA6Mi>0@f2UJh6%KJ*7PxT(e>dBcphXu`y;c
z!<WT!DzK#Li-Jv@KtybE>9zs^LGo>``~k~*=V5;sn`t=)I@R>)GN>xsI_4LJSqQEu
zKH6SiDMg@M7~A-vFCtU^NFS0EW1vg1%k?7PI<gFHRG%^E=4sNdy_wV7Uy*r)1URQC
zPnMD$8f7(<DKYH4vIBBlwGerOi!Y9n=^r4a&ZJrAFCh?I%h54svaiflNS1h~MJ!s0
zG8}&xh(}9UdOsqMUG@65+5H8Vzk#i%Z7W)UhJzy%$uVaF$NfSdMxDs(aUI^XGvZ;e
zby0Un1{_iFY!fzpVJB1EQDvzpNy%#eiSt{#1D$tXYE!D@V~nbjlrm^<et9oT%3jG#
ziMb@Fr#)h(>+RiQgDn$zz*+zf68?`B`{RFY;gkHQf~$OGZ1lZtuN`7(haMU~ias!D
zS&WR407-FB(kx(!)CRz!N%?6?V^I*jVqma^r5rN&i}OHK!=Mv58U=J>9*()0I*^rq
zW_;j`A_6OLnO3Ca4dpZ8eaB^*yp@;!O7Zg{$Cn6*exL#{$~!v11ny%7ZA_+D$ew?M
z>`^O4U^S2(qno-v-g`hKaycmgz$RpCviCElG=3~{k3)@O@m6$UeWRKXj6X4&DaX-9
zn6Yn&j+W`%Y2g;%q)EQ8x+57NaWpaxSWExiz#aCy<{2PDfFuHC(QKP(nT390%cGHT
z7>qL!cj!idf4a7laZKh-&I8H+>J5LRYxLR+O0u|ZR<2cbd1w5gY}9|R*?WwlR(9%4
zYM53Ip`z}nUr5swVb#^75yX1;l^qX5HFn&2ErI{vK6<~Wjck@?Vmy`2u7C$ny-1Pm
zQuh8_%6yI<dH6XJ?H!nA@!ccEk`!H4&Pf+S$TUMAO2xr#ao9`ZZ}eoWlo5Z3t@AsU
zYupBTi|>zhkX+4U_mPRvs2F#Krj9gPWh$!}nV!pAX1s`>Q8Y<|u6nYEH+`^Tz_oTt
zV)i!KuaP<+79v5&hgf4#93qQu)sC9t2JC765l^lqNFYN&nHmH7<D<aOVoZQBe8d8T
z;zOi}#*qf;By%j?jE3?eNm75dyZ1^0*Ql?x*{twx%T+25$IdHNYNNZe+Iz(T%0qp8
zV&#1|3x!a2dx}gYlfRPHwCyEZNK`*>iNrYcSjHh@2>7@(c|e(^PDO+#q?;Ii`(cPQ
z9XU$K!^eu)L|zipVQm7J1Jq2G_oc;EpdyF~66rJVQks@qETX&v7f62;Z-qa2^#R3p
zNbKVCwiKptIquX0zB?GMa{e!#D1)}wWcXeo0Yt9TUuydo=>ltb2tVr-OL(dM>0qZ{
z*oSi2+~sDe%!P6%Hf4#WO-<81NmKf<;<H~;Vm$!5F)=dNW?{h{Pkwu%a{V71#Wvx|
zt2aB8Ck2Z!ljMmxeA$05RZXfiKk8Lab@!kHDy=lh7<?GQQDVxb%2$H*Y9JliLKzT~
zuIBq(t227yIn1(k<45>LRIy0W%n`Lz46V#ZP4se5(`3pn#_|uke!%EzCaX<bjge?=
zlGL`>uSAnsis-27LEI-*!?xuz3Q;MUV%s<{d7_ep3ZZ1k6Kj8ntXc?ap29q99li*i
zu@Sw6hW4MW0dNk(`0>Qb+Va;YR-aHkX2g*1X@*`u3C-sCp0?RM#FTS?lnUd0Q#?e%
z?czdk!Q3aE(FI18QNU{zGJGeT<R@iOU6ZS+c`OHE4XgNi)1ceO>ig1scbRW;8pOcA
zCjE)x2hYz|aK?WPpuJazcTHjzVTeT`Ig6=E(aW;B+QI9=f(cw#2(jpiS1pHo`)2T_
z$?VfG?q%d_D0Q;WDA#%sD9uDp61*op3|o}fX~|*#!$>wVl)$~pmbB__Ln4hts>cA0
z|KzS=2E&3v#;0j-jKES>3>^36-r(gcA6bcJnKxRU@0Wj&qHN=-NpijQoQVpeG+m(u
z_Ydn8=k(LQ^Fxph{0N?V*LF-PzvG1&xYBJyZCBqqIn<*qWoA>x+`qX@Cy?s`TQft1
zDwC@Q&S5Rv;2hD9``e$5Klxa=Uav?KzbtkQ9(Hv_mdU#Z)R*k9Ot=l@S;E?NS-(`2
z8lbUvSR{Xvl46JWYhJvg_%k%>Ic<mi{aDX-u0Y)F`}E!B@GnGlg?ClAA-!LLVXqtQ
z+SiVfgRG$4hK)!|mRI+V6#2hWSWu;0cNobJV$m^hXEcY~&fQ+*T!6ux5sne1u<N#T
z3V$1!I-&?_D^&WMq<;3NdQj46S#jnUEq+%LTjGC~O5rneFB{_?;BaUY#!cgBWh*i5
zW+gd=J${Fu6(XE8A~^=-pn_5B`qY@5)v0vF*2Xop6-0~Dljih|{|?X3Zh6i!KAA|!
zM1ShxQ!VZSrv)29anx>fq=K9&AbaSXhXni=a8N9*?NuZm>p%V_Xgw*x@2I14c&CQj
zWYT|G>m@69PsGb+`>S5CXok0%o4Zv-wvyKPn}Cd=QrTgmPjA^F-y*k&*Sdba&*tok
zV?72+<hn!9)>aTU$kEv?h)U$|gX(E)?BYxWgk`rpVQZuIqls0z%D!yLHRR@pghr0s
z;e9!P@!trmzME|(DLRzcLZbbGb9OFEfDV5kr~3t`{(%GGPo<<*kM?}IS=%9R7d7Lx
zTwiT<A1cyy$w-s_DYI{Ug0o1wdShRm-$?)F$jJ*?cP%D1np4!}7w{1sT$tUy>(8((
zyo@zj&v5Vl<(;Ai6~g0G=X&hrg>lj^ds&oLesIgMiqG{$Xxj`TxHob*9#SP*S#p2x
z_FOTNT1(~^&JI(n+1v49Y^&E8`=O?*Oum+zNd@|@dKmbe$@|!zKFaU4-LTxj<YH)7
ztoqT~YZGN#lPuB*-6ggyz$KRHhg$Lnn%WdV!sl6Q33$qmVclLo@L2kN`BDnH3n`&5
z9Nml0q#-fGKSJ2mJ|7~V6t=1s+#7#QCpP7wgn#BcUl$j0wQ~}w7ft1>5^w{OK6Ius
zC-yOUNM*VUaZ>YB|1J5e4+yAU4HA=VmQL8(cM^{b&Gsq%Zi=6@^LuNIyI=RhYq3jb
zTG8Gmzk6`Ae7>{wkX;_naRnAc1~r=Gd-K}He&?w~XE|AOzg2ygkL%OQIGlg<p{TIP
zkGEz0j9peln6l9&td{fMe_Gx{uoW21p_<W4vOle+CpMj7R6(akv$L-<RmFB_-)uZ}
zpsv^Gk#!dnwBcYf45%z(BdaVb;S($2jDzX!y+VhCIJ!yr5ie@8sqy8|F!AENSB6mw
zMujf-t&UG`&M&luj3aTp*2#a=UDG<zD|4aWet!v0pd=povlYQWrF#)Xei1`$fB(C>
z+oV~r<|n&~dHbpxs%q<KzXj}}N$}b2^WH(QYTmPW%*DEPmPu<|mW_~Xz*K9#<qnKz
zhNCa}^CGH8o`p(5LbgHcV5hHsnY*y->HQSEl{xu*b<)hUSjTErHW`1G`1^veLs;Ko
zmn>IdX&`qMYQqIVylM5(7PKs$Ow$$buVe<@bLWwnD7Us)pWANk?cKWr@^$>Tjz6OT
z?^c6+UOqp)+3)cG&K7^ZUH$I%F@;V1dRw{C7#rvLbaOKJ`AQu}^kL4Z|HFM>x9{^b
z^2wN$`18-i4~{M2A9jEJ4^yePe-#v&?LG~)rOy@NJoWnrz5g|8He<T;rgpc#S<BSW
z9{jSgD;D_MQ@rnFU}@KrOzt5sTl`^cZD;zj`|o^az;5vO)z?o$Z^6e`W&&SMDn!|w
z)OL6&oqo>p?-ssX{5E*ox(eRi3O-EyW7_y;{o@#;5&7-$U}b*|N3~b4x6#<pKv+jG
zzd-om-YEF>Zux|IJ7Cn;Z+{{)@>6o~uMb6!Tli~1{?{9C&wqA5_}iu{^8Hz!*<fP(
zr2a9ipZn}3c`fGd)#&EQZKJY!_ic;DeL1_A!+*6@HlB#Xu+@Iye|s{WH7y^69G(UT
zMuqzy{QnS~T;6~0o!u;~J-Gy|q1Ip<eoKg;0Jn{`B|pEl6+e{QnwQ%K%EJ!<^KiZQ
zaC5Y=^5Xp9;N-w*>FsLmZ1aCM1{e(H<K+eYNAmMQ|4V{-{`>mB0EO^?xFP=pln>1J
z?*^C$!q3YG2Lk^;>-vAg+tbU!<DV)1Hg+~1jux)}E4_bzYPPojoA>;;E1>^d{(l5?
zARiu(or4$XzXJ+@TH0C(+S*!JLm_++DA)ob0Oc3>7b4z&bg+g(A^iLTf`S%2mLLxs
zTTd=~oBuG!#lq9e#^e9|Q3Y8^82~^-Qd;?+0SNyN#3Bse{DNs=>>)5F+<fn+>!gTV
zE=j_$vd(`b2fEZRO5dJ_NlA9ETOo)$_A;z|kOkqC_=PP?RiB-iQKn?{*wSgl!Y@kE
z?;N~qqkWc(Dpivxw>)3-dlb4{>u{QAx^UCb>5LN~wjhW4n%m%jQ(C|>X(q;H=QX9C
z!!z<az9>Q&f^R;A7Yv4QYfenesjf_`TEtu?3Q$)rjGhiyW#lgx_FL~$bPce<^_yWz
z3i{@=DROXXGSAI?MS=s8AZCJU?1VDyjAcSw;8*3p<T3l7|2+PGJF@?S|H1#@fABx}
RAN;TJe*pn^{n`Nd3;@<MuucE~

delta 258336
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

diff --git a/spec/features/projects/pipeline_schedules_spec.rb b/spec/features/projects/pipeline_schedules_spec.rb
new file mode 100644
index 0000000000000..cdac4fe2111ff
--- /dev/null
+++ b/spec/features/projects/pipeline_schedules_spec.rb
@@ -0,0 +1,140 @@
+require 'spec_helper'
+
+feature 'Pipeline Schedules', :feature do
+  include PipelineSchedulesHelper
+  include WaitForAjax
+
+  let!(:project) { create(:project) }
+  let!(:pipeline_schedule) { create(:ci_pipeline_schedule, project: project) }
+  let(:scope) { nil }
+  let!(:user) { create(:user) }
+
+  before do
+    project.add_master(user)
+
+    login_as(user)
+    visit_page
+  end
+
+  describe 'GET /projects/pipeline_schedules' do
+    let(:visit_page) { visit_pipelines_schedules }
+
+    it 'avoids N + 1 queries' do
+      control_count = ActiveRecord::QueryRecorder.new { visit_pipelines_schedules }.count
+
+      create_list(:ci_pipeline_schedule, 2, project: project)
+
+      expect { visit_pipelines_schedules }.not_to exceed_query_limit(control_count)
+    end
+
+    describe 'The view' do
+      it 'displays the required information description' do
+        page.within('.pipeline-schedule-table-row') do
+          expect(page).to have_content('pipeline schedule')
+          expect(page).to have_link('master')
+          expect(page).to have_content('None')
+        end
+      end
+
+      it 'creates a new scheduled pipeline' do
+        click_link 'New Schedule'
+
+        expect(page).to have_content('Schedule a new pipeline')
+      end
+
+      it 'changes ownership of the pipeline' do
+        click_link 'Take ownership'
+        page.within('.pipeline-schedule-table-row') do
+          expect(page).not_to have_content('No owner')
+          expect(page).to have_link('John Doe')
+        end
+      end
+
+      it 'edits the pipeline' do
+        page.within('.pipeline-schedule-table-row') do
+          click_link 'Edit'
+        end
+
+        expect(page).to have_content('Edit Pipeline Schedule')
+      end
+
+      it 'deletes the pipeline' do
+        click_link 'Delete'
+
+        expect(page).not_to have_content('pipeline schedule')
+      end
+    end
+  end
+
+  describe 'POST /projects/pipeline_schedules/new', js: true do
+    let(:visit_page) { visit_new_pipeline_schedule }
+
+    it 'it creates a new scheduled pipeline' do
+      fill_in_schedule_form
+      save_pipeline_schedule
+
+      expect(page).to have_content('my fancy description')
+    end
+
+    it 'it prevents an invalid form from being submitted' do
+      save_pipeline_schedule
+
+      expect(page).to have_content('This field is required')
+    end
+  end
+
+  describe 'PATCH /projects/pipelines_schedules/:id/edit', js: true do
+    let(:visit_page) do
+      edit_pipeline_schedule
+    end
+
+    it 'it displays existing properties' do
+      description = find_field('schedule_description').value
+      expect(description).to eq('pipeline schedule')
+      expect(page).to have_button('master')
+      expect(page).to have_button('UTC')
+    end
+
+    it 'edits the scheduled pipeline' do
+      fill_in 'schedule_description', with: 'my brand new description'
+
+      save_pipeline_schedule
+
+      expect(page).to have_content('my brand new description')
+    end
+  end
+
+  def visit_new_pipeline_schedule
+    visit new_namespace_project_pipeline_schedule_path(project.namespace, project, pipeline_schedule)
+  end
+
+  def edit_pipeline_schedule
+    visit edit_namespace_project_pipeline_schedule_path(project.namespace, project, pipeline_schedule)
+  end
+
+  def visit_pipelines_schedules
+    visit namespace_project_pipeline_schedules_path(project.namespace, project, scope: scope)
+  end
+
+  def select_timezone
+    click_button 'Select a timezone'
+    click_link 'American Samoa'
+  end
+
+  def select_target_branch
+    click_button 'Select target branch'
+    click_link 'master'
+  end
+
+  def save_pipeline_schedule
+    click_button 'Save pipeline schedule'
+  end
+
+  def fill_in_schedule_form
+    fill_in 'schedule_description', with: 'my fancy description'
+    fill_in 'schedule_cron', with: '* 1 2 3 4'
+
+    select_timezone
+    select_target_branch
+  end
+end
diff --git a/spec/features/security/project/internal_access_spec.rb b/spec/features/security/project/internal_access_spec.rb
index 26879a77c487e..78a76d9c11246 100644
--- a/spec/features/security/project/internal_access_spec.rb
+++ b/spec/features/security/project/internal_access_spec.rb
@@ -3,7 +3,7 @@
 describe "Internal Project Access", feature: true  do
   include AccessMatchers
 
-  let(:project) { create(:project, :internal) }
+  set(:project) { create(:project, :internal) }
 
   describe "Project should be internal" do
     describe '#internal?' do
@@ -437,6 +437,20 @@
     end
   end
 
+  describe "GET /:project_path/pipeline_schedules" do
+    subject { namespace_project_pipeline_schedules_path(project.namespace, project) }
+
+    it { is_expected.to be_allowed_for(:admin) }
+    it { is_expected.to be_allowed_for(:owner).of(project) }
+    it { is_expected.to be_allowed_for(:master).of(project) }
+    it { is_expected.to be_allowed_for(:developer).of(project) }
+    it { is_expected.to be_allowed_for(:reporter).of(project) }
+    it { is_expected.to be_allowed_for(:guest).of(project) }
+    it { is_expected.to be_allowed_for(:user) }
+    it { is_expected.to be_denied_for(:external) }
+    it { is_expected.to be_denied_for(:visitor) }
+  end
+
   describe "GET /:project_path/environments" do
     subject { namespace_project_environments_path(project.namespace, project) }
 
diff --git a/spec/features/security/project/private_access_spec.rb b/spec/features/security/project/private_access_spec.rb
index 699ca4f724c4f..a66f6e0905591 100644
--- a/spec/features/security/project/private_access_spec.rb
+++ b/spec/features/security/project/private_access_spec.rb
@@ -3,7 +3,7 @@
 describe "Private Project Access", feature: true  do
   include AccessMatchers
 
-  let(:project) { create(:project, :private, public_builds: false) }
+  set(:project) { create(:project, :private, public_builds: false) }
 
   describe "Project should be private" do
     describe '#private?' do
@@ -478,6 +478,48 @@
     it { is_expected.to be_denied_for(:visitor) }
   end
 
+  describe "GET /:project_path/pipeline_schedules" do
+    subject { namespace_project_pipeline_schedules_path(project.namespace, project) }
+
+    it { is_expected.to be_allowed_for(:admin) }
+    it { is_expected.to be_allowed_for(:owner).of(project) }
+    it { is_expected.to be_allowed_for(:master).of(project) }
+    it { is_expected.to be_allowed_for(:developer).of(project) }
+    it { is_expected.to be_allowed_for(:reporter).of(project) }
+    it { is_expected.to be_denied_for(:guest).of(project) }
+    it { is_expected.to be_denied_for(:user) }
+    it { is_expected.to be_denied_for(:external) }
+    it { is_expected.to be_denied_for(:visitor) }
+  end
+
+  describe "GET /:project_path/pipeline_schedules/new" do
+    subject { new_namespace_project_pipeline_schedule_path(project.namespace, project) }
+
+    it { is_expected.to be_allowed_for(:admin) }
+    it { is_expected.to be_allowed_for(:owner).of(project) }
+    it { is_expected.to be_allowed_for(:master).of(project) }
+    it { is_expected.to be_allowed_for(:developer).of(project) }
+    it { is_expected.to be_denied_for(:reporter).of(project) }
+    it { is_expected.to be_denied_for(:guest).of(project) }
+    it { is_expected.to be_denied_for(:user) }
+    it { is_expected.to be_denied_for(:external) }
+    it { is_expected.to be_denied_for(:visitor) }
+  end
+
+  describe "GET /:project_path/environments/new" do
+    subject { new_namespace_project_pipeline_schedule_path(project.namespace, project) }
+
+    it { is_expected.to be_allowed_for(:admin) }
+    it { is_expected.to be_allowed_for(:owner).of(project) }
+    it { is_expected.to be_allowed_for(:master).of(project) }
+    it { is_expected.to be_allowed_for(:developer).of(project) }
+    it { is_expected.to be_denied_for(:reporter).of(project) }
+    it { is_expected.to be_denied_for(:guest).of(project) }
+    it { is_expected.to be_denied_for(:user) }
+    it { is_expected.to be_denied_for(:external) }
+    it { is_expected.to be_denied_for(:visitor) }
+  end
+
   describe "GET /:project_path/container_registry" do
     let(:container_repository) { create(:container_repository) }
 
diff --git a/spec/features/security/project/public_access_spec.rb b/spec/features/security/project/public_access_spec.rb
index 624f0d0f48590..5cd575500c329 100644
--- a/spec/features/security/project/public_access_spec.rb
+++ b/spec/features/security/project/public_access_spec.rb
@@ -3,7 +3,7 @@
 describe "Public Project Access", feature: true  do
   include AccessMatchers
 
-  let(:project) { create(:project, :public) }
+  set(:project) { create(:project, :public) }
 
   describe "Project should be public" do
     describe '#public?' do
@@ -257,6 +257,20 @@
     end
   end
 
+  describe "GET /:project_path/pipeline_schedules" do
+    subject { namespace_project_pipeline_schedules_path(project.namespace, project) }
+
+    it { is_expected.to be_allowed_for(:admin) }
+    it { is_expected.to be_allowed_for(:owner).of(project) }
+    it { is_expected.to be_allowed_for(:master).of(project) }
+    it { is_expected.to be_allowed_for(:developer).of(project) }
+    it { is_expected.to be_allowed_for(:reporter).of(project) }
+    it { is_expected.to be_allowed_for(:guest).of(project) }
+    it { is_expected.to be_allowed_for(:user) }
+    it { is_expected.to be_allowed_for(:external) }
+    it { is_expected.to be_allowed_for(:visitor) }
+  end
+
   describe "GET /:project_path/environments" do
     subject { namespace_project_environments_path(project.namespace, project) }
 
diff --git a/spec/features/triggers_spec.rb b/spec/features/triggers_spec.rb
index 783f330221cac..c1ae6db00c640 100644
--- a/spec/features/triggers_spec.rb
+++ b/spec/features/triggers_spec.rb
@@ -77,77 +77,6 @@
       expect(page.find('.flash-notice')).to have_content 'Trigger was successfully updated.'
       expect(page.find('.triggers-list')).to have_content new_trigger_title
     end
-
-    context 'scheduled triggers' do
-      let!(:trigger) do
-        create(:ci_trigger, owner: user, project: @project, description: trigger_title)
-      end
-
-      context 'enabling schedule' do
-        before do
-          visit edit_namespace_project_trigger_path(@project.namespace, @project, trigger)
-        end
-
-        scenario 'do fill form with valid data and save' do
-          find('#trigger_trigger_schedule_attributes_active').click
-          fill_in 'trigger_trigger_schedule_attributes_cron', with: '1 * * * *'
-          fill_in 'trigger_trigger_schedule_attributes_cron_timezone', with: 'UTC'
-          fill_in 'trigger_trigger_schedule_attributes_ref', with: 'master'
-          click_button 'Save trigger'
-
-          expect(page.find('.flash-notice')).to have_content 'Trigger was successfully updated.'
-        end
-
-        scenario 'do not fill form with valid data and save' do
-          find('#trigger_trigger_schedule_attributes_active').click
-          click_button 'Save trigger'
-
-          expect(page).to have_content 'The form contains the following errors'
-        end
-
-        context 'when GitLab time_zone is ActiveSupport::TimeZone format' do
-          before do
-            allow(Time).to receive(:zone)
-              .and_return(ActiveSupport::TimeZone['Eastern Time (US & Canada)'])
-          end
-
-          scenario 'do fill form with valid data and save' do
-            find('#trigger_trigger_schedule_attributes_active').click
-            fill_in 'trigger_trigger_schedule_attributes_cron', with: '1 * * * *'
-            fill_in 'trigger_trigger_schedule_attributes_cron_timezone', with: 'UTC'
-            fill_in 'trigger_trigger_schedule_attributes_ref', with: 'master'
-            click_button 'Save trigger'
-
-            expect(page.find('.flash-notice'))
-              .to have_content 'Trigger was successfully updated.'
-          end
-        end
-      end
-
-      context 'disabling schedule' do
-        before do
-          trigger.create_trigger_schedule(
-            project: trigger.project,
-            active: true,
-            ref: 'master',
-            cron: '1 * * * *',
-            cron_timezone: 'UTC')
-
-          visit edit_namespace_project_trigger_path(@project.namespace, @project, trigger)
-        end
-
-        scenario 'disable and save form' do
-          find('#trigger_trigger_schedule_attributes_active').click
-          click_button 'Save trigger'
-          expect(page.find('.flash-notice')).to have_content 'Trigger was successfully updated.'
-
-          visit edit_namespace_project_trigger_path(@project.namespace, @project, trigger)
-          checkbox = find_field('trigger_trigger_schedule_attributes_active')
-
-          expect(checkbox).not_to be_checked
-        end
-      end
-    end
   end
 
   describe 'trigger "Take ownership" workflow' do
diff --git a/spec/finders/pipeline_schedules_finder_spec.rb b/spec/finders/pipeline_schedules_finder_spec.rb
new file mode 100644
index 0000000000000..e184a87c9c725
--- /dev/null
+++ b/spec/finders/pipeline_schedules_finder_spec.rb
@@ -0,0 +1,41 @@
+require 'spec_helper'
+
+describe PipelineSchedulesFinder do
+  let(:project) { create(:empty_project) }
+
+  let!(:active_schedule) { create(:ci_pipeline_schedule, project: project) }
+  let!(:inactive_schedule) { create(:ci_pipeline_schedule, :inactive, project: project) }
+
+  subject { described_class.new(project).execute(params) }
+
+  describe "#execute" do
+    context 'when the scope is nil' do
+      let(:params) { { scope: nil } }
+
+      it 'selects all pipeline pipeline schedules' do
+        expect(subject.count).to be(2)
+        expect(subject).to include(active_schedule, inactive_schedule)
+      end
+    end
+
+    context 'when the scope is active' do
+      let(:params) { { scope: 'active' } }
+
+      it 'selects only active pipelines' do
+        expect(subject.count).to be(1)
+        expect(subject).to include(active_schedule)
+        expect(subject).not_to include(inactive_schedule)
+      end
+    end
+
+    context 'when the scope is inactve' do
+      let(:params) { { scope: 'inactive' } }
+
+      it 'selects only inactive pipelines' do
+        expect(subject.count).to be(1)
+        expect(subject).not_to include(active_schedule)
+        expect(subject).to include(inactive_schedule)
+      end
+    end
+  end
+end
diff --git a/spec/javascripts/pipeline_schedules/interval_pattern_input_spec.js b/spec/javascripts/pipeline_schedules/interval_pattern_input_spec.js
new file mode 100644
index 0000000000000..08fa6ca9057c4
--- /dev/null
+++ b/spec/javascripts/pipeline_schedules/interval_pattern_input_spec.js
@@ -0,0 +1,217 @@
+import Vue from 'vue';
+import IntervalPatternInput from '~/pipeline_schedules/components/interval_pattern_input';
+
+const IntervalPatternInputComponent = Vue.extend(IntervalPatternInput);
+const inputNameAttribute = 'schedule[cron]';
+
+const cronIntervalPresets = {
+  everyDay: '0 4 * * *',
+  everyWeek: '0 4 * * 0',
+  everyMonth: '0 4 1 * *',
+};
+
+window.gl = window.gl || {};
+
+window.gl.pipelineScheduleFieldErrors = {
+  updateFormValidityState: () => {},
+};
+
+describe('Interval Pattern Input Component', function () {
+  describe('when prop initialCronInterval is passed (edit)', function () {
+    describe('when prop initialCronInterval is custom', function () {
+      beforeEach(function () {
+        this.initialCronInterval = '1 2 3 4 5';
+        this.intervalPatternComponent = new IntervalPatternInputComponent({
+          propsData: {
+            initialCronInterval: this.initialCronInterval,
+          },
+        }).$mount();
+      });
+
+      it('is initialized as a Vue component', function () {
+        expect(this.intervalPatternComponent).toBeDefined();
+      });
+
+      it('prop initialCronInterval is set', function () {
+        expect(this.intervalPatternComponent.initialCronInterval).toBe(this.initialCronInterval);
+      });
+
+      it('sets showUnsetWarning to false', function (done) {
+        Vue.nextTick(() => {
+          expect(this.intervalPatternComponent.showUnsetWarning).toBe(false);
+          done();
+        });
+      });
+
+      it('does not render showUnsetWarning', function (done) {
+        Vue.nextTick(() => {
+          expect(this.intervalPatternComponent.$el.outerHTML).not.toContain('Schedule not yet set');
+          done();
+        });
+      });
+
+      it('sets isEditable to true', function (done) {
+        Vue.nextTick(() => {
+          expect(this.intervalPatternComponent.isEditable).toBe(true);
+          done();
+        });
+      });
+    });
+
+    describe('when prop initialCronInterval is preset', function () {
+      beforeEach(function () {
+        this.intervalPatternComponent = new IntervalPatternInputComponent({
+          propsData: {
+            inputNameAttribute,
+            initialCronInterval: '0 4 * * *',
+          },
+        }).$mount();
+      });
+
+      it('is initialized as a Vue component', function () {
+        expect(this.intervalPatternComponent).toBeDefined();
+      });
+
+      it('sets showUnsetWarning to false', function (done) {
+        Vue.nextTick(() => {
+          expect(this.intervalPatternComponent.showUnsetWarning).toBe(false);
+          done();
+        });
+      });
+
+      it('does not render showUnsetWarning', function (done) {
+        Vue.nextTick(() => {
+          expect(this.intervalPatternComponent.$el.outerHTML).not.toContain('Schedule not yet set');
+          done();
+        });
+      });
+
+      it('sets isEditable to false', function (done) {
+        Vue.nextTick(() => {
+          expect(this.intervalPatternComponent.isEditable).toBe(false);
+          done();
+        });
+      });
+    });
+  });
+
+  describe('when prop initialCronInterval is not passed (new)', function () {
+    beforeEach(function () {
+      this.intervalPatternComponent = new IntervalPatternInputComponent({
+        propsData: {
+          inputNameAttribute,
+        },
+      }).$mount();
+    });
+
+    it('is initialized as a Vue component', function () {
+      expect(this.intervalPatternComponent).toBeDefined();
+    });
+
+    it('prop initialCronInterval is set', function () {
+      const defaultInitialCronInterval = '';
+      expect(this.intervalPatternComponent.initialCronInterval).toBe(defaultInitialCronInterval);
+    });
+
+    it('sets showUnsetWarning to true', function (done) {
+      Vue.nextTick(() => {
+        expect(this.intervalPatternComponent.showUnsetWarning).toBe(true);
+        done();
+      });
+    });
+
+    it('renders showUnsetWarning to true', function (done) {
+      Vue.nextTick(() => {
+        expect(this.intervalPatternComponent.$el.outerHTML).toContain('Schedule not yet set');
+        done();
+      });
+    });
+
+    it('sets isEditable to true', function (done) {
+      Vue.nextTick(() => {
+        expect(this.intervalPatternComponent.isEditable).toBe(true);
+        done();
+      });
+    });
+  });
+
+  describe('User Actions', function () {
+    beforeEach(function () {
+      // For an unknown reason, Phantom.js doesn't trigger click events
+      // on radio buttons in a way Vue can register. So, we have to mount
+      // to a fixture.
+      setFixtures('<div id="my-mount"></div>');
+
+      this.initialCronInterval = '1 2 3 4 5';
+      this.intervalPatternComponent = new IntervalPatternInputComponent({
+        propsData: {
+          initialCronInterval: this.initialCronInterval,
+        },
+      }).$mount('#my-mount');
+    });
+
+    it('cronInterval is updated when everyday preset interval is selected', function (done) {
+      this.intervalPatternComponent.$el.querySelector('#every-day').click();
+
+      Vue.nextTick(() => {
+        expect(this.intervalPatternComponent.cronInterval).toBe(cronIntervalPresets.everyDay);
+        expect(this.intervalPatternComponent.$el.querySelector('.cron-interval-input').value).toBe(cronIntervalPresets.everyDay);
+        done();
+      });
+    });
+
+    it('cronInterval is updated when everyweek preset interval is selected', function (done) {
+      this.intervalPatternComponent.$el.querySelector('#every-week').click();
+
+      Vue.nextTick(() => {
+        expect(this.intervalPatternComponent.cronInterval).toBe(cronIntervalPresets.everyWeek);
+        expect(this.intervalPatternComponent.$el.querySelector('.cron-interval-input').value).toBe(cronIntervalPresets.everyWeek);
+
+        done();
+      });
+    });
+
+    it('cronInterval is updated when everymonth preset interval is selected', function (done) {
+      this.intervalPatternComponent.$el.querySelector('#every-month').click();
+
+      Vue.nextTick(() => {
+        expect(this.intervalPatternComponent.cronInterval).toBe(cronIntervalPresets.everyMonth);
+        expect(this.intervalPatternComponent.$el.querySelector('.cron-interval-input').value).toBe(cronIntervalPresets.everyMonth);
+        done();
+      });
+    });
+
+    it('only a space is added to cronInterval (trimmed later) when custom radio is selected', function (done) {
+      this.intervalPatternComponent.$el.querySelector('#every-month').click();
+      this.intervalPatternComponent.$el.querySelector('#custom').click();
+
+      Vue.nextTick(() => {
+        const intervalWithSpaceAppended = `${cronIntervalPresets.everyMonth} `;
+        expect(this.intervalPatternComponent.cronInterval).toBe(intervalWithSpaceAppended);
+        expect(this.intervalPatternComponent.$el.querySelector('.cron-interval-input').value).toBe(intervalWithSpaceAppended);
+        done();
+      });
+    });
+
+    it('text input is disabled when preset interval is selected', function (done) {
+      this.intervalPatternComponent.$el.querySelector('#every-month').click();
+
+      Vue.nextTick(() => {
+        expect(this.intervalPatternComponent.isEditable).toBe(false);
+        expect(this.intervalPatternComponent.$el.querySelector('.cron-interval-input').disabled).toBe(true);
+        done();
+      });
+    });
+
+    it('text input is enabled when custom is selected', function (done) {
+      this.intervalPatternComponent.$el.querySelector('#every-month').click();
+      this.intervalPatternComponent.$el.querySelector('#custom').click();
+
+      Vue.nextTick(() => {
+        expect(this.intervalPatternComponent.isEditable).toBe(true);
+        expect(this.intervalPatternComponent.$el.querySelector('.cron-interval-input').disabled).toBe(false);
+        done();
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/pipeline_schedules/pipeline_schedule_callout_spec.js b/spec/javascripts/pipeline_schedules/pipeline_schedule_callout_spec.js
new file mode 100644
index 0000000000000..1d05f37cb3615
--- /dev/null
+++ b/spec/javascripts/pipeline_schedules/pipeline_schedule_callout_spec.js
@@ -0,0 +1,91 @@
+import Vue from 'vue';
+import Cookies from 'js-cookie';
+import PipelineSchedulesCallout from '~/pipeline_schedules/components/pipeline_schedules_callout';
+
+const PipelineSchedulesCalloutComponent = Vue.extend(PipelineSchedulesCallout);
+const cookieKey = 'pipeline_schedules_callout_dismissed';
+
+describe('Pipeline Schedule Callout', () => {
+  describe('independent of cookies', () => {
+    beforeEach(() => {
+      this.calloutComponent = new PipelineSchedulesCalloutComponent().$mount();
+    });
+
+    it('the component can be initialized', () => {
+      expect(this.calloutComponent).toBeDefined();
+    });
+
+    it('correctly sets illustrationSvg', () => {
+      expect(this.calloutComponent.illustrationSvg).toContain('<svg');
+    });
+  });
+
+  describe(`when ${cookieKey} cookie is set`, () => {
+    beforeEach(() => {
+      Cookies.set(cookieKey, true);
+      this.calloutComponent = new PipelineSchedulesCalloutComponent().$mount();
+    });
+
+    it('correctly sets calloutDismissed to true', () => {
+      expect(this.calloutComponent.calloutDismissed).toBe(true);
+    });
+
+    it('does not render the callout', () => {
+      expect(this.calloutComponent.$el.childNodes.length).toBe(0);
+    });
+  });
+
+  describe('when cookie is not set', () => {
+    beforeEach(() => {
+      Cookies.remove(cookieKey);
+      this.calloutComponent = new PipelineSchedulesCalloutComponent().$mount();
+    });
+
+    it('correctly sets calloutDismissed to false', () => {
+      expect(this.calloutComponent.calloutDismissed).toBe(false);
+    });
+
+    it('renders the callout container', () => {
+      expect(this.calloutComponent.$el.querySelector('.bordered-box')).not.toBeNull();
+    });
+
+    it('renders the callout svg', () => {
+      expect(this.calloutComponent.$el.outerHTML).toContain('<svg');
+    });
+
+    it('renders the callout title', () => {
+      expect(this.calloutComponent.$el.outerHTML).toContain('Scheduling Pipelines');
+    });
+
+    it('renders the callout text', () => {
+      expect(this.calloutComponent.$el.outerHTML).toContain('runs pipelines in the future');
+    });
+
+    it('updates calloutDismissed when close button is clicked', (done) => {
+      this.calloutComponent.$el.querySelector('#dismiss-callout-btn').click();
+
+      Vue.nextTick(() => {
+        expect(this.calloutComponent.calloutDismissed).toBe(true);
+        done();
+      });
+    });
+
+    it('#dismissCallout updates calloutDismissed', (done) => {
+      this.calloutComponent.dismissCallout();
+
+      Vue.nextTick(() => {
+        expect(this.calloutComponent.calloutDismissed).toBe(true);
+        done();
+      });
+    });
+
+    it('is hidden when close button is clicked', (done) => {
+      this.calloutComponent.$el.querySelector('#dismiss-callout-btn').click();
+
+      Vue.nextTick(() => {
+        expect(this.calloutComponent.$el.childNodes.length).toBe(0);
+        done();
+      });
+    });
+  });
+});
diff --git a/spec/lib/gitlab/import_export/all_models.yml b/spec/lib/gitlab/import_export/all_models.yml
index baa81870e810d..688e731bf156c 100644
--- a/spec/lib/gitlab/import_export/all_models.yml
+++ b/spec/lib/gitlab/import_export/all_models.yml
@@ -101,6 +101,7 @@ pipelines:
 - cancelable_statuses
 - manual_actions
 - artifacts
+- pipeline_schedule
 statuses:
 - project
 - pipeline
@@ -112,9 +113,13 @@ triggers:
 - project
 - trigger_requests
 - owner
-- trigger_schedule
-trigger_schedule:
-- trigger
+pipeline_schedules:
+- project
+- owner
+- pipelines
+- last_pipeline
+pipeline_schedule:
+- pipelines
 deploy_keys:
 - user
 - deploy_keys_projects
@@ -221,7 +226,7 @@ project:
 - active_runners
 - variables
 - triggers
-- trigger_schedules
+- pipeline_schedules
 - environments
 - deployments
 - project_feature
diff --git a/spec/lib/gitlab/import_export/safe_model_attributes.yml b/spec/lib/gitlab/import_export/safe_model_attributes.yml
index a66086f8b4756..3af2a172e6dfa 100644
--- a/spec/lib/gitlab/import_export/safe_model_attributes.yml
+++ b/spec/lib/gitlab/import_export/safe_model_attributes.yml
@@ -188,6 +188,7 @@ Ci::Pipeline:
 - user_id
 - lock_version
 - auto_canceled_by_id
+- pipeline_schedule_id
 CommitStatus:
 - id
 - project_id
@@ -247,18 +248,19 @@ Ci::Trigger:
 - owner_id
 - description
 - ref
-Ci::TriggerSchedule:
+Ci::PipelineSchedule:
 - id
-- project_id
-- trigger_id
-- deleted_at
-- created_at
-- updated_at
+- description
+- ref
 - cron
 - cron_timezone
 - next_run_at
-- ref
+- project_id
+- owner_id
 - active
+- deleted_at
+- created_at
+- updated_at
 DeployKey:
 - id
 - user_id
diff --git a/spec/lib/gitlab/usage_data_spec.rb b/spec/lib/gitlab/usage_data_spec.rb
index bf1dfe7f41297..9046d5c413fe9 100644
--- a/spec/lib/gitlab/usage_data_spec.rb
+++ b/spec/lib/gitlab/usage_data_spec.rb
@@ -32,6 +32,7 @@
         ci_pipelines
         ci_runners
         ci_triggers
+        ci_pipeline_schedules
         deploy_keys
         deployments
         environments
diff --git a/spec/models/ci/pipeline_schedule_spec.rb b/spec/models/ci/pipeline_schedule_spec.rb
new file mode 100644
index 0000000000000..822b98c5f6c9a
--- /dev/null
+++ b/spec/models/ci/pipeline_schedule_spec.rb
@@ -0,0 +1,112 @@
+require 'spec_helper'
+
+describe Ci::PipelineSchedule, models: true do
+  it { is_expected.to belong_to(:project) }
+  it { is_expected.to belong_to(:owner) }
+
+  it { is_expected.to have_many(:pipelines) }
+
+  it { is_expected.to respond_to(:ref) }
+  it { is_expected.to respond_to(:cron) }
+  it { is_expected.to respond_to(:cron_timezone) }
+  it { is_expected.to respond_to(:description) }
+  it { is_expected.to respond_to(:next_run_at) }
+  it { is_expected.to respond_to(:deleted_at) }
+
+  describe 'validations' do
+    it 'does not allow invalid cron patters' do
+      pipeline_schedule = build(:ci_pipeline_schedule, cron: '0 0 0 * *')
+
+      expect(pipeline_schedule).not_to be_valid
+    end
+
+    it 'does not allow invalid cron patters' do
+      pipeline_schedule = build(:ci_pipeline_schedule, cron_timezone: 'invalid')
+
+      expect(pipeline_schedule).not_to be_valid
+    end
+  end
+
+  describe '#set_next_run_at' do
+    let!(:pipeline_schedule) { create(:ci_pipeline_schedule, :nightly) }
+
+    context 'when creates new pipeline schedule' do
+      let(:expected_next_run_at) do
+        Gitlab::Ci::CronParser.new(pipeline_schedule.cron, pipeline_schedule.cron_timezone).
+          next_time_from(Time.now)
+      end
+
+      it 'updates next_run_at automatically' do
+        expect(Ci::PipelineSchedule.last.next_run_at).to eq(expected_next_run_at)
+      end
+    end
+
+    context 'when updates cron of exsisted pipeline schedule' do
+      let(:new_cron) { '0 0 1 1 *' }
+
+      let(:expected_next_run_at) do
+        Gitlab::Ci::CronParser.new(new_cron, pipeline_schedule.cron_timezone).
+          next_time_from(Time.now)
+      end
+
+      it 'updates next_run_at automatically' do
+        pipeline_schedule.update!(cron: new_cron)
+
+        expect(Ci::PipelineSchedule.last.next_run_at).to eq(expected_next_run_at)
+      end
+    end
+  end
+
+  describe '#schedule_next_run!' do
+    let!(:pipeline_schedule) { create(:ci_pipeline_schedule, :nightly) }
+
+    context 'when reschedules after 10 days from now' do
+      let(:future_time) { 10.days.from_now }
+
+      let(:expected_next_run_at) do
+        Gitlab::Ci::CronParser.new(pipeline_schedule.cron, pipeline_schedule.cron_timezone).
+          next_time_from(future_time)
+      end
+
+      it 'points to proper next_run_at' do
+        Timecop.freeze(future_time) do
+          pipeline_schedule.schedule_next_run!
+
+          expect(pipeline_schedule.next_run_at).to eq(expected_next_run_at)
+        end
+      end
+    end
+  end
+
+  describe '#real_next_run' do
+    subject do
+      described_class.last.real_next_run(worker_cron: worker_cron,
+                                         worker_time_zone: worker_time_zone)
+    end
+
+    context 'when GitLab time_zone is UTC' do
+      before do
+        allow(Time).to receive(:zone)
+          .and_return(ActiveSupport::TimeZone[worker_time_zone])
+      end
+
+      let(:worker_time_zone) { 'UTC' }
+
+      context 'when cron_timezone is Eastern Time (US & Canada)' do
+        before do
+          create(:ci_pipeline_schedule, :nightly,
+                  cron_timezone: 'Eastern Time (US & Canada)')
+        end
+
+        let(:worker_cron) { '0 1 2 3 *' }
+
+        it 'returns the next time worker executes' do
+          expect(subject.min).to eq(0)
+          expect(subject.hour).to eq(1)
+          expect(subject.day).to eq(2)
+          expect(subject.month).to eq(3)
+        end
+      end
+    end
+  end
+end
diff --git a/spec/models/ci/pipeline_spec.rb b/spec/models/ci/pipeline_spec.rb
index 3b222ea1c3dc7..208c8cb1c3d41 100644
--- a/spec/models/ci/pipeline_spec.rb
+++ b/spec/models/ci/pipeline_spec.rb
@@ -13,6 +13,7 @@
   it { is_expected.to belong_to(:project) }
   it { is_expected.to belong_to(:user) }
   it { is_expected.to belong_to(:auto_canceled_by) }
+  it { is_expected.to belong_to(:pipeline_schedule) }
 
   it { is_expected.to have_many(:statuses) }
   it { is_expected.to have_many(:trigger_requests) }
diff --git a/spec/models/ci/trigger_schedule_spec.rb b/spec/models/ci/trigger_schedule_spec.rb
deleted file mode 100644
index 92447564d7c87..0000000000000
--- a/spec/models/ci/trigger_schedule_spec.rb
+++ /dev/null
@@ -1,108 +0,0 @@
-require 'spec_helper'
-
-describe Ci::TriggerSchedule, models: true do
-  it { is_expected.to belong_to(:project) }
-  it { is_expected.to belong_to(:trigger) }
-  it { is_expected.to respond_to(:ref) }
-
-  describe '#set_next_run_at' do
-    context 'when creates new TriggerSchedule' do
-      before do
-        trigger_schedule = create(:ci_trigger_schedule, :nightly)
-        @expected_next_run_at = Gitlab::Ci::CronParser.new(trigger_schedule.cron, trigger_schedule.cron_timezone)
-                                                      .next_time_from(Time.now)
-      end
-
-      it 'updates next_run_at automatically' do
-        expect(Ci::TriggerSchedule.last.next_run_at).to eq(@expected_next_run_at)
-      end
-    end
-
-    context 'when updates cron of exsisted TriggerSchedule' do
-      before do
-        trigger_schedule = create(:ci_trigger_schedule, :nightly)
-        new_cron = '0 0 1 1 *'
-        trigger_schedule.update!(cron: new_cron) # Subject
-        @expected_next_run_at = Gitlab::Ci::CronParser.new(new_cron, trigger_schedule.cron_timezone)
-                                                      .next_time_from(Time.now)
-      end
-
-      it 'updates next_run_at automatically' do
-        expect(Ci::TriggerSchedule.last.next_run_at).to eq(@expected_next_run_at)
-      end
-    end
-  end
-
-  describe '#schedule_next_run!' do
-    context 'when reschedules after 10 days from now' do
-      before do
-        trigger_schedule = create(:ci_trigger_schedule, :nightly)
-        time_future = Time.now + 10.days
-        allow(Time).to receive(:now).and_return(time_future)
-        trigger_schedule.schedule_next_run! # Subject
-        @expected_next_run_at = Gitlab::Ci::CronParser.new(trigger_schedule.cron, trigger_schedule.cron_timezone)
-                                                      .next_time_from(time_future)
-      end
-
-      it 'points to proper next_run_at' do
-        expect(Ci::TriggerSchedule.last.next_run_at).to eq(@expected_next_run_at)
-      end
-    end
-
-    context 'when cron is invalid' do
-      before do
-        trigger_schedule = create(:ci_trigger_schedule, :nightly)
-        trigger_schedule.cron = 'Invalid-cron'
-        trigger_schedule.schedule_next_run! # Subject
-      end
-
-      it 'sets nil to next_run_at' do
-        expect(Ci::TriggerSchedule.last.next_run_at).to be_nil
-      end
-    end
-
-    context 'when cron_timezone is invalid' do
-      before do
-        trigger_schedule = create(:ci_trigger_schedule, :nightly)
-        trigger_schedule.cron_timezone = 'Invalid-cron_timezone'
-        trigger_schedule.schedule_next_run! # Subject
-      end
-
-      it 'sets nil to next_run_at' do
-        expect(Ci::TriggerSchedule.last.next_run_at).to be_nil
-      end
-    end
-  end
-
-  describe '#real_next_run' do
-    subject do
-      Ci::TriggerSchedule.last.real_next_run(worker_cron: worker_cron,
-                                             worker_time_zone: worker_time_zone)
-    end
-
-    context 'when GitLab time_zone is UTC' do
-      before do
-        allow(Time).to receive(:zone)
-          .and_return(ActiveSupport::TimeZone[worker_time_zone])
-      end
-
-      let(:worker_time_zone) { 'UTC' }
-
-      context 'when cron_timezone is Eastern Time (US & Canada)' do
-        before do
-          create(:ci_trigger_schedule, :nightly,
-                  cron_timezone: 'Eastern Time (US & Canada)')
-        end
-
-        let(:worker_cron) { '0 1 2 3 *' }
-
-        it 'returns the next time worker executes' do
-          expect(subject.min).to eq(0)
-          expect(subject.hour).to eq(1)
-          expect(subject.day).to eq(2)
-          expect(subject.month).to eq(3)
-        end
-      end
-    end
-  end
-end
diff --git a/spec/models/ci/trigger_spec.rb b/spec/models/ci/trigger_spec.rb
index d26121018ce05..92c15c13c1843 100644
--- a/spec/models/ci/trigger_spec.rb
+++ b/spec/models/ci/trigger_spec.rb
@@ -7,7 +7,6 @@
     it { is_expected.to belong_to(:project) }
     it { is_expected.to belong_to(:owner) }
     it { is_expected.to have_many(:trigger_requests) }
-    it { is_expected.to have_one(:trigger_schedule) }
   end
 
   describe 'before_validation' do
diff --git a/spec/models/project_spec.rb b/spec/models/project_spec.rb
index 2fc8ffed80a09..429b3dd83afb3 100644
--- a/spec/models/project_spec.rb
+++ b/spec/models/project_spec.rb
@@ -73,6 +73,7 @@
     it { is_expected.to have_many(:notification_settings).dependent(:destroy) }
     it { is_expected.to have_many(:forks).through(:forked_project_links) }
     it { is_expected.to have_many(:uploads).dependent(:destroy) }
+    it { is_expected.to have_many(:pipeline_schedules).dependent(:destroy) }
 
     context 'after initialized' do
       it "has a project_feature" do
diff --git a/spec/workers/pipeline_schedule_worker_spec.rb b/spec/workers/pipeline_schedule_worker_spec.rb
new file mode 100644
index 0000000000000..91d5a16993fd2
--- /dev/null
+++ b/spec/workers/pipeline_schedule_worker_spec.rb
@@ -0,0 +1,51 @@
+require 'spec_helper'
+
+describe PipelineScheduleWorker do
+  subject { described_class.new.perform }
+
+  set(:project) { create(:project, :repository) }
+  set(:user) { create(:user) }
+
+  let!(:pipeline_schedule) do
+    create(:ci_pipeline_schedule, :nightly, project: project, owner: user)
+  end
+
+  before do
+    project.add_master(user)
+
+    stub_ci_pipeline_to_return_yaml_file
+  end
+
+  context 'when there is a scheduled pipeline within next_run_at' do
+    let(:next_run_at) { 2.days.ago }
+
+    before do
+      pipeline_schedule.update_column(:next_run_at, next_run_at)
+    end
+
+    it 'creates a new pipeline' do
+      expect { subject }.to change { project.pipelines.count }.by(1)
+    end
+
+    it 'updates the next_run_at field' do
+      subject
+
+      expect(pipeline_schedule.reload.next_run_at).to be > Time.now
+    end
+
+    it 'sets the schedule on the pipeline' do
+      subject
+      expect(project.pipelines.last.pipeline_schedule).to eq(pipeline_schedule)
+    end
+  end
+
+  context 'inactive schedule' do
+    before do
+      pipeline_schedule.update(active: false)
+    end
+
+    it 'does not creates a new pipeline' do
+      expect { subject }.not_to change { project.pipelines.count }
+    end
+  end
+end
diff --git a/spec/workers/trigger_schedule_worker_spec.rb b/spec/workers/trigger_schedule_worker_spec.rb
deleted file mode 100644
index 861bed4442ec5..0000000000000
--- a/spec/workers/trigger_schedule_worker_spec.rb
+++ /dev/null
@@ -1,73 +0,0 @@
-require 'spec_helper'
-
-describe TriggerScheduleWorker do
-  let(:worker) { described_class.new }
-
-  before do
-    stub_ci_pipeline_to_return_yaml_file
-  end
-
-  context 'when there is a scheduled trigger within next_run_at' do
-    let(:next_run_at) { 2.days.ago }
-
-    let!(:trigger_schedule) do
-      create(:ci_trigger_schedule, :nightly)
-    end
-
-    before do
-      trigger_schedule.update_column(:next_run_at, next_run_at)
-    end
-
-    it 'creates a new trigger request' do
-      expect { worker.perform }.to change { Ci::TriggerRequest.count }
-    end
-
-    it 'creates a new pipeline' do
-      expect { worker.perform }.to change { Ci::Pipeline.count }
-      expect(Ci::Pipeline.last).to be_pending
-    end
-
-    it 'updates next_run_at' do
-      worker.perform
-
-      expect(trigger_schedule.reload.next_run_at).not_to eq(next_run_at)
-    end
-
-    context 'inactive schedule' do
-      before do
-        trigger_schedule.update(active: false)
-      end
-
-      it 'does not create a new trigger' do
-        expect { worker.perform }.not_to change { Ci::TriggerRequest.count }
-      end
-    end
-  end
-
-  context 'when there are no scheduled triggers within next_run_at' do
-    before { create(:ci_trigger_schedule, :nightly) }
-
-    it 'does not create a new pipeline' do
-      expect { worker.perform }.not_to change { Ci::Pipeline.count }
-    end
-
-    it 'does not update next_run_at' do
-      expect { worker.perform }.not_to change { Ci::TriggerSchedule.last.next_run_at }
-    end
-  end
-
-  context 'when next_run_at is nil' do
-    before do
-      schedule = create(:ci_trigger_schedule, :nightly)
-      schedule.update_column(:next_run_at, nil)
-    end
-
-    it 'does not create a new pipeline' do
-      expect { worker.perform }.not_to change { Ci::Pipeline.count }
-    end
-
-    it 'does not update next_run_at' do
-      expect { worker.perform }.not_to change { Ci::TriggerSchedule.last.next_run_at }
-    end
-  end
-end
-- 
GitLab


From e4814ddc44cc2f37ecd86eb0390c388451243585 Mon Sep 17 00:00:00 2001
From: Timothy Andrew <mail@timothyandrew.net>
Date: Mon, 8 May 2017 10:24:10 +0000
Subject: [PATCH 357/363] Merge branch 'update-deps-licenses-for-9-2' into
 'master'

Update the vendor licenses file for 9.2

See merge request !11159
---
 vendor/licenses.csv | 297 +++++++++++++++++++++++---------------------
 1 file changed, 158 insertions(+), 139 deletions(-)

diff --git a/vendor/licenses.csv b/vendor/licenses.csv
index 6441df25fe1b2..a8e7f5e3ea970 100644
--- a/vendor/licenses.csv
+++ b/vendor/licenses.csv
@@ -3,7 +3,7 @@ abbrev,1.0.9,ISC
 accepts,1.3.3,MIT
 ace-rails-ap,4.1.2,MIT
 acorn,4.0.11,MIT
-acorn-dynamic-import,2.0.2,MIT
+acorn-dynamic-import,2.0.1,MIT
 acorn-jsx,3.0.1,MIT
 actionmailer,4.2.8,MIT
 actionpack,4.2.8,MIT
@@ -16,7 +16,7 @@ acts-as-taggable-on,4.0.0,MIT
 addressable,2.3.8,Apache 2.0
 after,0.8.2,MIT
 after_commit_queue,1.3.0,MIT
-ajv,4.11.5,MIT
+ajv,4.11.2,MIT
 ajv-keywords,1.5.1,MIT
 akismet,2.0.0,MIT
 align-text,0.1.4,MIT
@@ -29,7 +29,7 @@ ansi-regex,2.1.1,MIT
 ansi-styles,2.2.1,MIT
 anymatch,1.3.0,ISC
 append-transform,0.4.0,MIT
-aproba,1.1.1,ISC
+aproba,1.1.0,ISC
 are-we-there-yet,1.1.2,ISC
 arel,6.0.4,MIT
 argparse,1.0.9,MIT
@@ -43,7 +43,7 @@ array-uniq,1.0.3,MIT
 array-unique,0.2.1,MIT
 arraybuffer.slice,0.0.6,MIT
 arrify,1.0.1,MIT
-asana,0.4.0,MIT
+asana,0.6.0,MIT
 asciidoctor,1.5.3,MIT
 asciidoctor-plantuml,0.0.7,MIT
 asn1,0.2.3,MIT
@@ -62,8 +62,8 @@ aws-sign2,0.6.0,Apache 2.0
 aws4,1.6.0,MIT
 axiom-types,0.1.1,MIT
 babel-code-frame,6.22.0,MIT
-babel-core,6.24.0,MIT
-babel-generator,6.24.0,MIT
+babel-core,6.23.1,MIT
+babel-generator,6.23.0,MIT
 babel-helper-bindify-decorators,6.22.0,MIT
 babel-helper-builder-binary-assignment-operator-visitor,6.22.0,MIT
 babel-helper-call-delegate,6.22.0,MIT
@@ -78,10 +78,10 @@ babel-helper-regex,6.22.0,MIT
 babel-helper-remap-async-to-generator,6.22.0,MIT
 babel-helper-replace-supers,6.23.0,MIT
 babel-helpers,6.23.0,MIT
-babel-loader,6.4.1,MIT
+babel-loader,6.2.10,MIT
 babel-messages,6.23.0,MIT
 babel-plugin-check-es2015-constants,6.22.0,MIT
-babel-plugin-istanbul,4.1.1,New BSD
+babel-plugin-istanbul,4.0.0,New BSD
 babel-plugin-syntax-async-functions,6.13.0,MIT
 babel-plugin-syntax-async-generators,6.13.0,MIT
 babel-plugin-syntax-class-properties,6.13.0,MIT
@@ -127,13 +127,13 @@ babel-preset-es2017,6.22.0,MIT
 babel-preset-latest,6.24.0,MIT
 babel-preset-stage-2,6.22.0,MIT
 babel-preset-stage-3,6.22.0,MIT
-babel-register,6.24.0,MIT
-babel-runtime,6.23.0,MIT
+babel-register,6.23.0,MIT
+babel-runtime,6.22.0,MIT
 babel-template,6.23.0,MIT
 babel-traverse,6.23.1,MIT
 babel-types,6.23.0,MIT
 babosa,1.0.2,MIT
-babylon,6.16.1,MIT
+babylon,6.15.0,MIT
 backo2,1.0.2,MIT
 balanced-match,0.4.2,MIT
 base32,0.3.2,MIT
@@ -149,20 +149,20 @@ binary-extensions,1.8.0,MIT
 bindata,2.3.5,ruby
 blob,0.0.4,unknown
 block-stream,0.0.9,ISC
-bluebird,3.5.0,MIT
+bluebird,3.4.7,MIT
 bn.js,4.11.6,MIT
-body-parser,1.17.1,MIT
+body-parser,1.16.0,MIT
 boom,2.10.1,New BSD
 bootstrap-sass,3.3.6,MIT
 brace-expansion,1.1.6,MIT
 braces,1.8.5,MIT
-brorand,1.1.0,MIT
+brorand,1.0.7,MIT
 browser,2.2.0,MIT
 browserify-aes,1.0.6,MIT
 browserify-cipher,1.0.0,MIT
 browserify-des,1.0.0,MIT
 browserify-rsa,4.0.1,MIT
-browserify-sign,4.0.4,ISC
+browserify-sign,4.0.0,ISC
 browserify-zlib,0.1.4,MIT
 browserslist,1.7.7,MIT
 buffer,4.9.1,MIT
@@ -178,8 +178,8 @@ callsites,0.2.0,MIT
 camelcase,1.2.1,MIT
 caniuse-api,1.6.1,MIT
 caniuse-db,1.0.30000649,CC-BY-4.0
-carrierwave,0.11.2,MIT
-caseless,0.12.0,Apache 2.0
+carrierwave,1.0.0,MIT
+caseless,0.11.0,Apache 2.0
 cause,0.1,MIT
 center-align,0.1.3,MIT
 chalk,1.1.3,MIT
@@ -194,6 +194,7 @@ citrus,3.0.2,MIT
 clap,1.1.3,MIT
 cli-cursor,1.0.2,MIT
 cli-width,2.1.0,ISC
+clipboard,1.6.1,MIT
 cliui,2.1.0,ISC
 clone,1.0.2,MIT
 co,4.6.0,MIT
@@ -216,14 +217,14 @@ commondir,1.0.1,MIT
 component-bind,1.0.0,unknown
 component-emitter,1.2.1,MIT
 component-inherit,0.0.3,unknown
-compressible,2.0.10,MIT
+compressible,2.0.9,MIT
 compression,1.6.2,MIT
 compression-webpack-plugin,0.3.2,MIT
 concat-map,0.0.1,MIT
 concat-stream,1.6.0,MIT
 config-chain,1.1.11,MIT
 configstore,1.4.0,Simplified BSD
-connect,3.6.0,MIT
+connect,3.5.0,MIT
 connect-history-api-fallback,1.3.0,MIT
 connection_pool,2.2.1,MIT
 console-browserify,1.1.0,MIT
@@ -233,7 +234,7 @@ constants-browserify,1.0.0,MIT
 contains-path,0.1.0,MIT
 content-disposition,0.5.2,MIT
 content-type,1.0.2,MIT
-convert-source-map,1.5.0,MIT
+convert-source-map,1.3.0,MIT
 cookie,0.3.1,MIT
 cookie-signature,1.0.6,MIT
 core-js,2.4.1,MIT
@@ -254,13 +255,13 @@ cssesc,0.1.0,MIT
 cssnano,3.10.0,MIT
 csso,2.3.2,MIT
 custom-event,1.0.1,MIT
-d,1.0.0,MIT
-d3,3.5.17,New BSD
+d,0.1.1,MIT
+d3,3.5.11,New BSD
 d3_rails,3.5.11,MIT
 dashdash,1.14.1,MIT
 date-now,0.1.4,MIT
 de-indent,1.0.2,MIT
-debug,2.6.3,MIT
+debug,2.6.0,MIT
 decamelize,1.2.0,MIT
 deckar01-task_list,1.0.6,MIT
 deep-extend,0.4.1,MIT
@@ -271,6 +272,7 @@ defaults,1.0.3,MIT
 defined,1.0.0,MIT
 del,2.2.2,MIT
 delayed-stream,1.0.0,MIT
+delegate,3.1.2,MIT
 delegates,1.0.0,MIT
 depd,1.1.0,MIT
 des.js,1.0.0,MIT
@@ -283,8 +285,8 @@ di,0.0.1,MIT
 diff-lcs,1.2.5,"MIT,Perl Artistic v2,GNU GPL v2"
 diffie-hellman,5.0.2,MIT
 diffy,3.1.0,MIT
-doctrine,2.0.0,Apache 2.0
-document-register-element,1.4.1,MIT
+doctrine,1.5.0,BSD
+document-register-element,1.3.0,MIT
 dom-serialize,2.2.1,MIT
 dom-serializer,0.1.0,MIT
 domain-browser,1.1.7,MIT
@@ -294,7 +296,7 @@ domhandler,2.3.0,unknown
 domutils,1.5.1,unknown
 doorkeeper,4.2.0,MIT
 doorkeeper-openid_connect,1.1.2,MIT
-dropzone,4.3.0,MIT
+dropzone,4.2.0,MIT
 dropzonejs-rails,0.7.2,MIT
 duplexer,0.1.1,MIT
 duplexify,3.5.0,MIT
@@ -303,36 +305,36 @@ editorconfig,0.13.2,MIT
 ee-first,1.1.1,MIT
 ejs,2.5.6,Apache 2.0
 electron-to-chromium,1.3.3,ISC
-elliptic,6.4.0,MIT
+elliptic,6.3.3,MIT
 email_reply_trimmer,0.1.6,MIT
 emoji-unicode-version,0.2.1,MIT
 emojis-list,2.1.0,MIT
 encodeurl,1.0.1,MIT
 encryptor,3.0.0,MIT
 end-of-stream,1.0.0,MIT
-engine.io,1.8.3,MIT
-engine.io-client,1.8.3,MIT
+engine.io,1.8.2,MIT
+engine.io-client,1.8.2,MIT
 engine.io-parser,1.3.2,MIT
 enhanced-resolve,3.1.0,MIT
 ent,2.2.0,MIT
 entities,1.1.1,BSD-like
 equalizer,0.0.11,MIT
 errno,0.1.4,MIT
-error-ex,1.3.1,MIT
+error-ex,1.3.0,MIT
 erubis,2.7.0,MIT
-es5-ext,0.10.15,MIT
-es6-iterator,2.0.1,MIT
-es6-map,0.1.5,MIT
+es5-ext,0.10.12,MIT
+es6-iterator,2.0.0,MIT
+es6-map,0.1.4,MIT
 es6-promise,3.0.2,MIT
-es6-set,0.1.5,MIT
-es6-symbol,3.1.1,MIT
-es6-weak-map,2.0.2,MIT
+es6-set,0.1.4,MIT
+es6-symbol,3.1.0,MIT
+es6-weak-map,2.0.1,MIT
 escape-html,1.0.3,MIT
 escape-string-regexp,1.0.5,MIT
 escape_utils,1.1.1,MIT
 escodegen,1.8.1,Simplified BSD
 escope,3.6.0,Simplified BSD
-eslint,3.19.0,MIT
+eslint,3.15.0,MIT
 eslint-config-airbnb-base,10.0.1,MIT
 eslint-import-resolver-node,0.2.3,MIT
 eslint-import-resolver-webpack,0.8.1,MIT
@@ -341,37 +343,39 @@ eslint-plugin-filenames,1.1.0,MIT
 eslint-plugin-html,2.0.1,ISC
 eslint-plugin-import,2.2.0,MIT
 eslint-plugin-jasmine,2.2.0,MIT
-espree,3.4.1,Simplified BSD
-esprima,2.7.3,Simplified BSD
-esquery,1.0.0,BSD
+eslint-plugin-promise,3.5.0,ISC
+espree,3.4.0,Simplified BSD
+esprima,3.1.3,Simplified BSD
 esrecurse,4.1.0,Simplified BSD
 estraverse,4.1.1,Simplified BSD
 esutils,2.0.2,BSD
-etag,1.8.0,MIT
+etag,1.7.0,MIT
 eve-raphael,0.5.0,Apache 2.0
-event-emitter,0.3.5,MIT
+event-emitter,0.3.4,MIT
 event-stream,3.3.4,MIT
 eventemitter3,1.2.0,MIT
 events,1.1.1,MIT
 eventsource,0.1.6,MIT
 evp_bytestokey,1.0.0,MIT
-excon,0.52.0,MIT
+excon,0.55.0,MIT
 execjs,2.6.0,MIT
 exit-hook,1.1.1,MIT
 expand-braces,0.1.2,MIT
 expand-brackets,0.1.5,MIT
 expand-range,1.8.2,MIT
-express,4.15.2,MIT
+exports-loader,0.6.4,MIT
+express,4.14.1,MIT
 expression_parser,0.9.0,MIT
 extend,3.0.0,MIT
 extglob,0.3.2,MIT
 extlib,0.9.16,MIT
 extract-zip,1.5.0,Simplified BSD
 extsprintf,1.0.2,MIT
-faraday,0.9.2,MIT
-faraday_middleware,0.10.0,MIT
+faraday,0.11.0,MIT
+faraday_middleware,0.11.0.1,MIT
 faraday_middleware-multi_json,0.0.6,MIT
 fast-levenshtein,2.0.6,MIT
+fast_gettext,1.4.0,"MIT,ruby"
 fastparse,1.1.1,MIT
 faye-websocket,0.7.3,MIT
 fd-slicer,1.0.1,MIT
@@ -383,37 +387,37 @@ filename-regex,2.0.0,MIT
 fileset,2.0.3,MIT
 filesize,3.3.0,New BSD
 fill-range,2.2.3,MIT
-finalhandler,1.0.1,MIT
+finalhandler,0.5.1,MIT
 find-cache-dir,0.1.1,MIT
 find-root,0.1.2,MIT
 find-up,2.1.0,MIT
 flat-cache,1.2.2,MIT
 flatten,1.0.2,MIT
 flowdock,0.7.1,MIT
-fog-aws,0.11.0,MIT
-fog-core,1.42.0,MIT
+fog-aws,0.13.0,MIT
+fog-core,1.44.1,MIT
 fog-google,0.5.0,MIT
 fog-json,1.0.2,MIT
 fog-local,0.3.0,MIT
 fog-openstack,0.1.6,MIT
 fog-rackspace,0.1.1,MIT
-fog-xml,0.1.2,MIT
+fog-xml,0.1.3,MIT
 font-awesome-rails,4.7.0.1,"MIT,SIL Open Font License"
-for-in,1.0.2,MIT
-for-own,0.1.5,MIT
+for-in,0.1.6,MIT
+for-own,0.1.4,MIT
 forever-agent,0.6.1,Apache 2.0
 form-data,2.1.2,MIT
 formatador,0.2.5,MIT
 forwarded,0.1.0,MIT
-fresh,0.5.0,MIT
+fresh,0.3.0,MIT
 from,0.1.7,MIT
 fs-extra,1.0.0,MIT
 fs.realpath,1.0.0,ISC
 fsevents,,unknown
-fstream,1.0.11,ISC
+fstream,1.0.10,ISC
 fstream-ignore,1.0.5,ISC
 function-bind,1.1.0,MIT
-gauge,2.7.3,ISC
+gauge,2.7.2,ISC
 gemnasium-gitlab-service,0.2.6,MIT
 gemojione,3.0.1,MIT
 generate-function,2.0.0,MIT
@@ -421,7 +425,9 @@ generate-object-property,1.2.0,MIT
 get-caller-file,1.0.2,ISC
 get_process_mem,0.2.0,MIT
 getpass,0.1.6,MIT
-gitaly,0.5.0,MIT
+gettext_i18n_rails,1.8.0,MIT
+gettext_i18n_rails_js,1.2.0,MIT
+gitaly,0.6.0,MIT
 github-linguist,4.7.6,MIT
 github-markup,1.4.0,MIT
 gitlab-flowdock-git-hook,1.0.1,MIT
@@ -432,12 +438,13 @@ glob,7.1.1,ISC
 glob-base,0.3.0,MIT
 glob-parent,2.0.0,ISC
 globalid,0.3.7,MIT
-globals,9.17.0,MIT
+globals,9.14.0,MIT
 globby,5.0.0,MIT
 gollum-grit_adapter,1.0.1,MIT
 gollum-lib,4.2.1,MIT
 gollum-rugged_adapter,0.4.4,MIT
 gon,6.1.0,MIT
+good-listener,1.2.2,MIT
 google-api-client,0.8.7,Apache 2.0
 google-protobuf,3.2.0.2,New BSD
 googleauth,0.5.1,Apache 2.0
@@ -446,13 +453,12 @@ graceful-fs,4.1.11,ISC
 graceful-readlink,1.0.1,MIT
 grape,0.19.1,MIT
 grape-entity,0.6.0,MIT
-grpc,1.1.2,New BSD
+grpc,1.2.5,New BSD
 gzip-size,3.0.0,MIT
 hamlit,2.6.1,MIT
 handle-thing,1.2.5,MIT
 handlebars,4.0.6,MIT
-har-schema,1.0.5,ISC
-har-validator,4.2.1,ISC
+har-validator,2.0.6,ISC
 has,1.0.1,MIT
 has-ansi,2.0.0,MIT
 has-binary,0.1.7,MIT
@@ -463,14 +469,14 @@ hash-sum,1.0.2,MIT
 hash.js,1.0.3,MIT
 hasha,2.2.0,MIT
 hashie,3.5.5,MIT
+hashie-forbidden_attributes,0.1.1,MIT
 hawk,3.1.3,New BSD
 he,1.1.1,MIT
 health_check,2.6.0,MIT
 hipchat,1.5.2,MIT
-hmac-drbg,1.0.0,MIT
 hoek,2.16.3,New BSD
 home-or-tmp,2.0.0,MIT
-hosted-git-info,2.4.1,ISC
+hosted-git-info,2.2.0,ISC
 hpack.js,2.1.6,MIT
 html-comment-regex,1.1.1,MIT
 html-entities,1.2.0,MIT
@@ -481,7 +487,7 @@ htmlparser2,3.9.2,MIT
 http,0.9.8,MIT
 http-cookie,1.0.3,MIT
 http-deceiver,1.2.7,MIT
-http-errors,1.6.1,MIT
+http-errors,1.5.1,MIT
 http-form_data,1.0.1,MIT
 http-proxy,1.16.2,MIT
 http-proxy-middleware,0.17.4,MIT
@@ -495,7 +501,7 @@ ice_nine,0.11.2,MIT
 iconv-lite,0.4.15,MIT
 icss-replace-symbols,1.0.2,ISC
 ieee754,1.1.8,New BSD
-ignore,3.2.6,MIT
+ignore,3.2.2,MIT
 ignore-by-default,1.0.1,ISC
 immediate,3.0.6,MIT
 imurmurhash,0.1.4,MIT
@@ -507,16 +513,16 @@ influxdb,0.2.3,MIT
 inherits,2.0.3,ISC
 ini,1.3.4,ISC
 inquirer,0.12.0,MIT
-interpret,1.0.2,MIT
+interpret,1.0.1,MIT
 invariant,2.2.2,New BSD
 invert-kv,1.0.0,MIT
-ipaddr.js,1.3.0,MIT
+ipaddr.js,1.2.0,MIT
 ipaddress,0.8.3,MIT
 is-absolute,0.2.6,MIT
 is-absolute-url,2.1.0,MIT
 is-arrayish,0.2.1,MIT
 is-binary-path,1.0.1,MIT
-is-buffer,1.1.5,MIT
+is-buffer,1.1.4,MIT
 is-builtin-module,1.0.0,MIT
 is-dotfile,1.0.2,MIT
 is-equal-shallow,0.1.3,MIT
@@ -525,7 +531,7 @@ is-extglob,1.0.0,MIT
 is-finite,1.0.2,MIT
 is-fullwidth-code-point,1.0.0,MIT
 is-glob,2.0.1,MIT
-is-my-json-valid,2.16.0,MIT
+is-my-json-valid,2.15.0,MIT
 is-npm,1.0.0,MIT
 is-number,2.1.0,MIT
 is-path-cwd,1.0.0,MIT
@@ -546,31 +552,32 @@ is-utf8,0.2.1,MIT
 is-windows,0.2.0,MIT
 isarray,1.0.0,MIT
 isbinaryfile,3.0.2,MIT
-isexe,2.0.0,ISC
+isexe,1.1.2,ISC
 isobject,2.1.0,MIT
 isstream,0.1.2,MIT
 istanbul,0.4.5,New BSD
-istanbul-api,1.1.7,New BSD
-istanbul-lib-coverage,1.0.2,New BSD
-istanbul-lib-hook,1.0.5,New BSD
-istanbul-lib-instrument,1.7.0,New BSD
-istanbul-lib-report,1.0.0,New BSD
-istanbul-lib-source-maps,1.1.1,New BSD
-istanbul-reports,1.0.2,New BSD
+istanbul-api,1.1.1,New BSD
+istanbul-lib-coverage,1.0.1,New BSD
+istanbul-lib-hook,1.0.0,New BSD
+istanbul-lib-instrument,1.4.2,New BSD
+istanbul-lib-report,1.0.0-alpha.3,New BSD
+istanbul-lib-source-maps,1.1.0,New BSD
+istanbul-reports,1.0.1,New BSD
 jasmine-core,2.5.2,MIT
 jasmine-jquery,2.1.1,MIT
+jed,1.1.1,MIT
 jira-ruby,1.1.2,MIT
 jodid25519,1.0.2,MIT
-jquery,2.2.4,MIT
+jquery,2.2.1,MIT
 jquery-atwho-rails,1.3.2,MIT
 jquery-rails,4.1.1,MIT
-jquery-ujs,1.2.2,MIT
+jquery-ujs,1.2.1,MIT
 js-base64,2.1.9,BSD
 js-beautify,1.6.12,MIT
-js-cookie,2.1.4,MIT
+js-cookie,2.1.3,MIT
 js-tokens,3.0.1,MIT
 js-yaml,3.7.0,MIT
-jsbn,0.1.1,MIT
+jsbn,0.1.0,BSD
 jsesc,1.3.0,MIT
 json,1.8.6,ruby
 json-jwt,1.7.1,MIT
@@ -583,18 +590,18 @@ json5,0.5.1,MIT
 jsonfile,2.4.0,MIT
 jsonify,0.0.0,Public Domain
 jsonpointer,4.0.1,MIT
-jsprim,1.4.0,MIT
+jsprim,1.3.1,MIT
 jszip,3.1.3,(MIT OR GPL-3.0)
 jszip-utils,0.0.2,MIT or GPLv3
 jwt,1.5.6,MIT
 kaminari,0.17.0,MIT
-karma,1.6.0,MIT
-karma-coverage-istanbul-reporter,0.2.3,MIT
+karma,1.4.1,MIT
+karma-coverage-istanbul-reporter,0.2.0,MIT
 karma-jasmine,1.1.0,MIT
-karma-mocha-reporter,2.2.3,MIT
-karma-phantomjs-launcher,1.0.4,MIT
+karma-mocha-reporter,2.2.2,MIT
+karma-phantomjs-launcher,1.0.2,MIT
 karma-sourcemap-loader,0.3.7,MIT
-karma-webpack,2.0.3,MIT
+karma-webpack,2.0.2,MIT
 kew,0.7.0,Apache 2.0
 kgio,2.10.0,LGPL-2.1+
 kind-of,3.1.0,MIT
@@ -610,7 +617,8 @@ lie,3.1.1,MIT
 little-plugger,1.1.4,MIT
 load-json-file,1.1.0,MIT
 loader-runner,2.3.0,MIT
-loader-utils,0.2.17,MIT
+loader-utils,0.2.16,MIT
+locale,2.1.2,"ruby,LGPLv3+"
 locate-path,2.0.0,MIT
 lodash,4.17.4,MIT
 lodash._baseassign,3.2.0,MIT
@@ -638,16 +646,17 @@ lodash.snakecase,4.0.1,MIT
 lodash.uniq,4.5.0,MIT
 lodash.words,4.2.0,MIT
 log4js,0.6.38,Apache 2.0
-logging,2.1.0,MIT
+logging,2.2.2,MIT
 longest,1.0.1,MIT
 loofah,2.0.3,MIT
 loose-envify,1.3.1,MIT
 lowercase-keys,1.0.0,MIT
 lru-cache,3.2.0,ISC
 macaddress,0.2.8,MIT
-mail,2.6.4,MIT
+mail,2.6.5,MIT
 mail_room,0.9.1,MIT
 map-stream,0.1.0,unknown
+marked,0.3.6,MIT
 math-expression-evaluator,1.2.16,MIT
 media-typer,0.3.0,MIT
 memoist,0.15.0,MIT
@@ -658,17 +667,16 @@ methods,1.1.2,MIT
 micromatch,2.3.11,MIT
 miller-rabin,4.0.0,MIT
 mime,1.3.4,MIT
-mime-db,1.27.0,MIT
+mime-db,1.26.0,MIT
 mime-types,2.99.3,"MIT,Artistic-2.0,GPL-2.0"
 mimemagic,0.3.0,MIT
 mini_portile2,2.1.0,MIT
 minimalistic-assert,1.0.0,ISC
-minimalistic-crypto-utils,1.0.1,MIT
 minimatch,3.0.3,ISC
 minimist,0.0.8,MIT
 mkdirp,0.5.1,MIT
-moment,2.18.1,MIT
-mousetrap,1.6.1,Apache 2.0
+moment,2.17.1,MIT
+mousetrap,1.4.6,Apache 2.0
 mousetrap-rails,1.4.6,"MIT,Apache"
 ms,0.7.2,MIT
 multi_json,1.12.1,MIT
@@ -684,14 +692,15 @@ nested-error-stacks,1.0.2,MIT
 net-ldap,0.12.1,MIT
 net-ssh,3.0.1,MIT
 netrc,0.11.0,MIT
+node-ensure,0.0.0,MIT
 node-libs-browser,2.0.0,MIT
-node-pre-gyp,0.6.34,New BSD
+node-pre-gyp,0.6.33,New BSD
 node-zopfli,2.0.2,MIT
 nodemon,1.11.0,MIT
 nokogiri,1.6.8.1,MIT
-nopt,4.0.1,ISC
-normalize-package-data,2.3.6,Simplified BSD
-normalize-path,2.1.1,MIT
+nopt,3.0.6,ISC
+normalize-package-data,2.3.5,Simplified BSD
+normalize-path,2.0.1,MIT
 normalize-range,0.1.2,MIT
 normalize-url,1.9.1,MIT
 npmlog,4.0.2,ISC
@@ -700,13 +709,13 @@ number-is-nan,1.0.1,MIT
 numerizer,0.1.1,MIT
 oauth,0.5.1,MIT
 oauth-sign,0.8.2,Apache 2.0
-oauth2,1.2.0,MIT
+oauth2,1.3.1,MIT
 object-assign,4.1.1,MIT
 object-component,0.0.3,unknown
 object.omit,2.0.1,MIT
 obuf,1.1.1,MIT
 octokit,4.6.2,MIT
-oj,2.17.4,MIT
+oj,2.17.5,MIT
 omniauth,1.4.2,MIT
 omniauth-auth0,1.4.1,MIT
 omniauth-authentiq,0.3.0,MIT
@@ -727,7 +736,7 @@ omniauth-twitter,1.2.1,MIT
 omniauth_crowd,2.2.3,MIT
 on-finished,2.3.0,MIT
 on-headers,1.0.1,MIT
-once,1.4.0,ISC
+once,1.3.3,ISC
 onetime,1.1.0,MIT
 opener,1.4.3,(WTFPL OR MIT)
 opn,4.0.2,MIT
@@ -748,7 +757,7 @@ p-locate,2.0.0,MIT
 package-json,1.2.0,MIT
 pako,1.0.5,(MIT AND Zlib)
 paranoia,2.2.0,MIT
-parse-asn1,5.1.0,ISC
+parse-asn1,5.0.0,ISC
 parse-glob,3.0.4,MIT
 parse-json,2.2.0,MIT
 parsejson,0.0.3,MIT
@@ -762,10 +771,10 @@ path-is-inside,1.0.2,(WTFPL OR MIT)
 path-parse,1.0.5,MIT
 path-to-regexp,0.1.7,MIT
 path-type,1.1.0,MIT
-pause-stream,0.0.11,"Apache2,MIT"
+pause-stream,0.0.11,"MIT,Apache2"
 pbkdf2,3.0.9,MIT
+pdfjs-dist,1.8.252,Apache 2.0
 pend,1.2.0,MIT
-performance-now,0.2.0,MIT
 pg,0.18.4,"BSD,ruby,GPL"
 phantomjs-prebuilt,2.1.14,Apache 2.0
 pify,2.3.0,MIT
@@ -775,6 +784,7 @@ pinkie-promise,2.0.1,MIT
 pkg-dir,1.0.0,MIT
 pkg-up,1.0.0,MIT
 pluralize,1.2.1,MIT
+po_to_json,1.0.1,MIT
 portfinder,1.0.13,MIT
 posix-spawn,0.3.11,"MIT,LGPL"
 postcss,5.2.16,MIT
@@ -818,12 +828,13 @@ premailer,1.8.6,New BSD
 premailer-rails,1.9.2,MIT
 prepend-http,1.0.4,MIT
 preserve,0.2.0,MIT
+prismjs,1.6.0,MIT
 private,0.1.7,MIT
 process,0.11.9,MIT
 process-nextick-args,1.0.7,MIT
 progress,1.1.8,MIT
 proto-list,1.2.4,ISC
-proxy-addr,1.1.4,MIT
+proxy-addr,1.1.3,MIT
 prr,0.0.0,MIT
 ps-tree,1.1.0,MIT
 pseudomap,1.0.2,ISC
@@ -832,7 +843,7 @@ punycode,1.4.1,MIT
 pyu-ruby-sasl,0.0.3.3,MIT
 q,1.5.0,MIT
 qjobs,1.1.5,MIT
-qs,6.4.0,New BSD
+qs,6.2.0,New BSD
 query-string,4.3.2,MIT
 querystring,0.2.0,MIT
 querystring-es3,0.2.1,MIT
@@ -857,15 +868,16 @@ randomatic,1.1.6,MIT
 randombytes,2.0.3,MIT
 range-parser,1.2.0,MIT
 raphael,2.2.7,MIT
+raven-js,3.15.0,Simplified BSD
 raw-body,2.2.0,MIT
 raw-loader,0.5.1,MIT
-rc,1.2.1,(BSD-2-Clause OR MIT OR Apache-2.0)
+rc,1.1.6,(BSD-2-Clause OR MIT OR Apache-2.0)
 rdoc,4.2.2,ruby
 react-dev-utils,0.5.2,New BSD
 read-all-stream,3.1.0,MIT
 read-pkg,1.1.0,MIT
 read-pkg-up,1.0.1,MIT
-readable-stream,2.0.6,MIT
+readable-stream,2.2.2,MIT
 readdirp,2.1.0,MIT
 readline2,1.0.1,MIT
 recaptcha,3.0.0,MIT
@@ -873,7 +885,7 @@ rechoir,0.6.2,MIT
 recursive-open-struct,1.0.0,MIT
 recursive-readdir,2.1.1,MIT
 redcarpet,3.4.0,MIT
-redis,3.2.2,MIT
+redis,3.3.3,MIT
 redis-actionpack,5.0.1,MIT
 redis-activesupport,5.0.1,MIT
 redis-namespace,1.5.2,MIT
@@ -883,18 +895,17 @@ redis-store,1.2.0,MIT
 reduce-css-calc,1.3.0,MIT
 reduce-function-call,1.0.2,MIT
 regenerate,1.3.2,MIT
-regenerator-runtime,0.10.3,MIT
+regenerator-runtime,0.10.1,MIT
 regenerator-transform,0.9.8,BSD
 regex-cache,0.4.3,MIT
 regexpu-core,2.0.0,MIT
 registry-url,3.1.0,MIT
 regjsgen,0.2.0,MIT
 regjsparser,0.1.5,BSD
-remove-trailing-separator,1.0.1,ISC
 repeat-element,1.1.2,MIT
 repeat-string,1.6.1,MIT
 repeating,2.0.1,MIT
-request,2.81.0,Apache 2.0
+request,2.79.0,Apache 2.0
 request-progress,2.0.1,MIT
 request_store,1.3.1,MIT
 require-directory,2.1.1,MIT
@@ -902,14 +913,14 @@ require-from-string,1.2.1,MIT
 require-main-filename,1.0.1,ISC
 require-uncached,1.0.3,MIT
 requires-port,1.0.0,MIT
-resolve,1.3.2,MIT
+resolve,1.2.0,MIT
 resolve-from,1.0.1,MIT
 responders,2.3.0,MIT
 rest-client,2.0.0,MIT
 restore-cursor,1.0.1,MIT
 retriable,1.4.1,MIT
 right-align,0.1.3,MIT
-rimraf,2.6.1,ISC
+rimraf,2.5.4,ISC
 rinku,2.0.0,ISC
 ripemd160,1.0.1,New BSD
 rotp,2.1.2,MIT
@@ -919,6 +930,7 @@ rqrcode-rails3,0.1.7,MIT
 ruby-fogbugz,0.2.1,MIT
 ruby-prof,0.16.2,Simplified BSD
 ruby-saml,1.4.1,MIT
+ruby_parser,3.8.4,MIT
 rubyntlm,0.5.2,MIT
 rubypants,0.2.0,BSD
 rufus-scheduler,3.1.10,MIT
@@ -934,23 +946,25 @@ sawyer,0.8.1,MIT
 sax,1.2.2,ISC
 securecompare,1.0.0,MIT
 seed-fu,2.3.6,MIT
+select,1.1.2,MIT
 select-hose,2.0.0,MIT
 select2,3.5.2-browserify,unknown
 select2-rails,3.5.9.3,MIT
 semver,5.3.0,ISC
 semver-diff,2.1.0,MIT
-send,0.15.1,MIT
+send,0.14.2,MIT
 sentry-raven,2.4.0,Apache 2.0
 serve-index,1.8.0,MIT
-serve-static,1.12.1,MIT
+serve-static,1.11.2,MIT
 set-blocking,2.0.0,ISC
 set-immediate-shim,1.0.1,MIT
 setimmediate,1.0.5,MIT
-setprototypeof,1.0.3,ISC
+setprototypeof,1.0.2,ISC
 settingslogic,2.0.9,MIT
+sexp_processor,4.8.0,MIT
 sha.js,2.4.8,MIT
-shelljs,0.7.7,New BSD
-sidekiq,4.2.7,LGPL
+shelljs,0.7.6,New BSD
+sidekiq,5.0.0,LGPL
 sidekiq-cron,0.4.4,MIT
 sidekiq-limit_fetch,3.4.0,MIT
 sigmund,1.0.1,ISC
@@ -961,16 +975,16 @@ slash,1.0.0,MIT
 slice-ansi,0.0.4,MIT
 slide,1.1.6,ISC
 sntp,1.0.9,BSD
-socket.io,1.7.3,MIT
+socket.io,1.7.2,MIT
 socket.io-adapter,0.5.0,MIT
-socket.io-client,1.7.3,MIT
+socket.io-client,1.7.2,MIT
 socket.io-parser,2.3.1,MIT
 sockjs,0.3.18,MIT
 sockjs-client,1.0.1,MIT
 sort-keys,1.1.2,MIT
 source-list-map,0.1.8,MIT
 source-map,0.5.6,New BSD
-source-map-support,0.4.14,MIT
+source-map-support,0.4.11,MIT
 spdx-correct,1.0.2,Apache 2.0
 spdx-expression-parse,1.0.4,(MIT AND CC-BY-3.0)
 spdx-license-ids,1.2.2,Unlicense
@@ -980,7 +994,8 @@ split,0.3.3,MIT
 sprintf-js,1.0.3,New BSD
 sprockets,3.7.1,MIT
 sprockets-rails,3.2.0,MIT
-sshpk,1.11.0,MIT
+sql.js,0.4.0,MIT
+sshpk,1.10.2,MIT
 state_machines,0.4.0,MIT
 state_machines-activemodel,0.4.0,MIT
 state_machines-activerecord,0.4.0,MIT
@@ -988,7 +1003,7 @@ stats-webpack-plugin,0.4.3,MIT
 statuses,1.3.1,MIT
 stream-browserify,2.0.1,MIT
 stream-combiner,0.0.4,MIT
-stream-http,2.7.0,MIT
+stream-http,2.6.3,MIT
 stream-shift,1.0.0,MIT
 strict-uri-encode,1.1.0,MIT
 string-length,1.0.1,MIT
@@ -998,16 +1013,17 @@ stringex,2.5.2,MIT
 stringstream,0.0.5,MIT
 strip-ansi,3.0.1,MIT
 strip-bom,2.0.0,MIT
-strip-json-comments,2.0.1,MIT
-supports-color,3.2.3,MIT
+strip-json-comments,1.0.4,MIT
+supports-color,0.2.0,MIT
 svgo,0.7.2,MIT
 sys-filesystem,1.1.6,Artistic 2.0
 table,3.8.3,New BSD
 tapable,0.2.6,MIT
 tar,2.2.1,ISC
-tar-pack,3.4.0,Simplified BSD
+tar-pack,3.3.0,Simplified BSD
 temple,0.7.7,MIT
-test-exclude,4.0.3,ISC
+test-exclude,4.0.0,ISC
+text,1.3.1,MIT
 text-table,0.2.0,MIT
 thor,0.19.4,MIT
 thread_safe,0.3.6,Apache 2.0
@@ -1021,7 +1037,8 @@ timeago.js,2.0.5,MIT
 timed-out,2.0.0,MIT
 timers-browserify,2.0.2,MIT
 timfel-krb5-auth,0.8.3,LGPL
-tmp,0.0.31,MIT
+tiny-emitter,1.1.0,MIT
+tmp,0.0.28,MIT
 to-array,0.1.4,MIT
 to-arraybuffer,1.0.1,MIT
 to-fast-properties,1.0.2,MIT
@@ -1034,10 +1051,10 @@ trim-right,1.0.1,MIT
 truncato,0.7.8,MIT
 tryit,1.0.3,MIT
 tty-browserify,0.0.0,MIT
-tunnel-agent,0.6.0,Apache 2.0
+tunnel-agent,0.4.3,Apache 2.0
 tweetnacl,0.14.5,Unlicense
 type-check,0.3.2,MIT
-type-is,1.6.15,MIT
+type-is,1.6.14,MIT
 typedarray,0.0.6,MIT
 tzinfo,1.2.2,MIT
 u2f,0.2.1,MIT
@@ -1060,17 +1077,18 @@ uniqs,2.0.0,MIT
 unpipe,1.0.0,MIT
 update-notifier,0.5.0,Simplified BSD
 url,0.11.0,MIT
+url-loader,0.5.8,MIT
 url-parse,1.0.5,MIT
 url_safe_base64,0.2.2,MIT
 user-home,2.0.0,MIT
-useragent,2.1.13,MIT
+useragent,2.1.12,MIT
 util,0.10.3,MIT
 util-deprecate,1.0.2,MIT
 utils-merge,1.0.0,MIT
 uuid,3.0.1,MIT
 validate-npm-package-license,3.0.1,Apache 2.0
 validates_hostname,1.0.6,MIT
-vary,1.1.1,MIT
+vary,1.1.0,MIT
 vendors,1.0.1,MIT
 verror,1.3.6,MIT
 version_sorter,2.1.0,MIT
@@ -1085,30 +1103,31 @@ vue-loader,11.3.4,MIT
 vue-resource,0.9.3,MIT
 vue-style-loader,2.0.5,MIT
 vue-template-compiler,2.2.6,MIT
-vue-template-es2015-compiler,1.5.2,MIT
+vue-template-es2015-compiler,1.5.1,MIT
 warden,1.2.6,MIT
 watchpack,1.3.1,MIT
 wbuf,1.7.2,MIT
 webpack,2.3.3,MIT
-webpack-bundle-analyzer,2.3.1,MIT
-webpack-dev-middleware,1.10.1,MIT
+webpack-bundle-analyzer,2.3.0,MIT
+webpack-dev-middleware,1.10.0,MIT
 webpack-dev-server,2.4.2,MIT
 webpack-rails,0.9.10,MIT
-webpack-sources,0.1.5,MIT
+webpack-sources,0.1.4,MIT
 websocket-driver,0.6.5,MIT
 websocket-extensions,0.1.1,MIT
 whet.extend,0.9.9,MIT
-which,1.2.14,ISC
+which,1.2.12,ISC
 which-module,1.0.0,ISC
 wide-align,1.1.0,ISC
 wikicloth,0.8.1,MIT
 window-size,0.1.0,MIT
-wordwrap,1.0.0,MIT
+wordwrap,0.0.2,MIT/X11
+worker-loader,0.8.0,MIT
 wrap-ansi,2.1.0,MIT
 wrappy,1.0.2,ISC
 write,0.2.1,MIT
 write-file-atomic,1.3.1,ISC
-ws,1.1.2,MIT
+ws,1.1.1,MIT
 wtf-8,1.0.0,MIT
 xdg-basedir,2.0.0,MIT
 xmlhttprequest-ssl,1.5.3,MIT
-- 
GitLab


From 07adf34ce022a5d84770ac9f605df9fafee85b8d Mon Sep 17 00:00:00 2001
From: Timothy Andrew <mail@timothyandrew.net>
Date: Mon, 8 May 2017 10:46:24 +0000
Subject: [PATCH 358/363] Merge branch 'update-templates-for-9-2' into 'master'

Update gitignore, dockerfile, and license templates for 9.2

See merge request !11158
---
 vendor/Dockerfile/OpenJDK-alpine.Dockerfile   |  8 ++++++
 vendor/Dockerfile/OpenJDK.Dockerfile          |  8 ++++++
 vendor/Dockerfile/Python-alpine.Dockerfile    | 19 +++++++++++++
 vendor/Dockerfile/Python.Dockerfile           | 22 +++++++++++++++
 vendor/gitignore/Global/Archives.gitignore    |  1 +
 vendor/gitignore/Global/JetBrains.gitignore   |  3 +++
 .../Global/MicrosoftOffice.gitignore          |  2 +-
 vendor/gitignore/Magento.gitignore            | 27 +++++++++++++++++++
 vendor/gitignore/Python.gitignore             |  4 +++
 vendor/gitignore/Qt.gitignore                 |  1 +
 vendor/gitignore/UnrealEngine.gitignore       |  5 ++++
 11 files changed, 99 insertions(+), 1 deletion(-)
 create mode 100644 vendor/Dockerfile/OpenJDK-alpine.Dockerfile
 create mode 100644 vendor/Dockerfile/OpenJDK.Dockerfile
 create mode 100644 vendor/Dockerfile/Python-alpine.Dockerfile
 create mode 100644 vendor/Dockerfile/Python.Dockerfile

diff --git a/vendor/Dockerfile/OpenJDK-alpine.Dockerfile b/vendor/Dockerfile/OpenJDK-alpine.Dockerfile
new file mode 100644
index 0000000000000..ee853d9cfd26f
--- /dev/null
+++ b/vendor/Dockerfile/OpenJDK-alpine.Dockerfile
@@ -0,0 +1,8 @@
+FROM openjdk:8-alpine
+
+COPY . /usr/src/myapp
+WORKDIR /usr/src/myapp
+
+RUN javac Main.java
+
+CMD ["java", "Main"]
diff --git a/vendor/Dockerfile/OpenJDK.Dockerfile b/vendor/Dockerfile/OpenJDK.Dockerfile
new file mode 100644
index 0000000000000..8a2ae62d93bf2
--- /dev/null
+++ b/vendor/Dockerfile/OpenJDK.Dockerfile
@@ -0,0 +1,8 @@
+FROM openjdk:9
+
+COPY . /usr/src/myapp
+WORKDIR /usr/src/myapp
+
+RUN javac Main.java
+
+CMD ["java", "Main"]
diff --git a/vendor/Dockerfile/Python-alpine.Dockerfile b/vendor/Dockerfile/Python-alpine.Dockerfile
new file mode 100644
index 0000000000000..59ac9f504dec5
--- /dev/null
+++ b/vendor/Dockerfile/Python-alpine.Dockerfile
@@ -0,0 +1,19 @@
+FROM python:3.6-alpine
+
+# Edit with mysql-client, postgresql-client, sqlite3, etc. for your needs.
+# Or delete entirely if not needed.
+RUN apk --no-cache add postgresql-client
+
+WORKDIR /usr/src/app
+
+COPY requirements.txt /usr/src/app/
+RUN pip install --no-cache-dir -r requirements.txt
+
+COPY . /usr/src/app
+
+# For Django
+EXPOSE 8000
+CMD ["python", "manage.py", "runserver", "0.0.0.0:8000"]
+
+# For some other command
+# CMD ["python", "app.py"]
diff --git a/vendor/Dockerfile/Python.Dockerfile b/vendor/Dockerfile/Python.Dockerfile
new file mode 100644
index 0000000000000..7c43ad990609f
--- /dev/null
+++ b/vendor/Dockerfile/Python.Dockerfile
@@ -0,0 +1,22 @@
+FROM python:3.6
+
+# Edit with mysql-client, postgresql-client, sqlite3, etc. for your needs.
+# Or delete entirely if not needed.
+RUN apt-get update \
+    && apt-get install -y --no-install-recommends \
+        postgresql-client \
+    && rm -rf /var/lib/apt/lists/*
+
+WORKDIR /usr/src/app
+
+COPY requirements.txt /usr/src/app/
+RUN pip install --no-cache-dir -r requirements.txt
+
+COPY . /usr/src/app
+
+# For Django
+EXPOSE 8000
+CMD ["python", "manage.py", "runserver", "0.0.0.0:8000"]
+
+# For some other command
+# CMD ["python", "app.py"]
diff --git a/vendor/gitignore/Global/Archives.gitignore b/vendor/gitignore/Global/Archives.gitignore
index e9eda68baf2e6..f440b808d982b 100644
--- a/vendor/gitignore/Global/Archives.gitignore
+++ b/vendor/gitignore/Global/Archives.gitignore
@@ -5,6 +5,7 @@
 *.rar
 *.zip
 *.gz
+*.tgz
 *.bzip
 *.bz2
 *.xz
diff --git a/vendor/gitignore/Global/JetBrains.gitignore b/vendor/gitignore/Global/JetBrains.gitignore
index a5d4cc86d33d5..ff23445e2b0d7 100644
--- a/vendor/gitignore/Global/JetBrains.gitignore
+++ b/vendor/gitignore/Global/JetBrains.gitignore
@@ -19,6 +19,9 @@
 .idea/**/gradle.xml
 .idea/**/libraries
 
+# CMake
+cmake-build-debug/
+
 # Mongo Explorer plugin:
 .idea/**/mongoSettings.xml
 
diff --git a/vendor/gitignore/Global/MicrosoftOffice.gitignore b/vendor/gitignore/Global/MicrosoftOffice.gitignore
index cb89174566014..0c203662d3908 100644
--- a/vendor/gitignore/Global/MicrosoftOffice.gitignore
+++ b/vendor/gitignore/Global/MicrosoftOffice.gitignore
@@ -13,4 +13,4 @@
 ~$*.ppt*
 
 # Visio autosave temporary files
-*.~vsdx
+*.~vsd*
diff --git a/vendor/gitignore/Magento.gitignore b/vendor/gitignore/Magento.gitignore
index b282f5cf54703..6f1fa22399234 100644
--- a/vendor/gitignore/Magento.gitignore
+++ b/vendor/gitignore/Magento.gitignore
@@ -3,14 +3,41 @@
 #--------------------------#
 
 /app/etc/local.xml
+
 /media/*
 !/media/.htaccess
+
+!/media/customer
+/media/customer/*
 !/media/customer/.htaccess
+
+!/media/dhl
+/media/dhl/*
 !/media/dhl/logo.jpg
+
+!/media/downloadable
+/media/downloadable/*
 !/media/downloadable/.htaccess
+
+!/media/xmlconnect
+/media/xmlconnect/*
+
+!/media/xmlconnect/custom
+/media/xmlconnect/custom/*
 !/media/xmlconnect/custom/ok.gif
+
+!/media/xmlconnect/original
+/media/xmlconnect/original/*
 !/media/xmlconnect/original/ok.gif
+
+!/media/xmlconnect/system
+/media/xmlconnect/system/*
 !/media/xmlconnect/system/ok.gif
+
 /var/*
 !/var/.htaccess
+
+!/var/package
+/var/package/*
 !/var/package/*.xml
+
diff --git a/vendor/gitignore/Python.gitignore b/vendor/gitignore/Python.gitignore
index ff65a437185fe..768d5f400bb1d 100644
--- a/vendor/gitignore/Python.gitignore
+++ b/vendor/gitignore/Python.gitignore
@@ -89,9 +89,13 @@ ENV/
 
 # Spyder project settings
 .spyderproject
+.spyproject
 
 # Rope project settings
 .ropeproject
 
 # mkdocs documentation
 /site
+
+# mypy
+.mypy_cache/
diff --git a/vendor/gitignore/Qt.gitignore b/vendor/gitignore/Qt.gitignore
index c7659c24f386b..6732e72091c00 100644
--- a/vendor/gitignore/Qt.gitignore
+++ b/vendor/gitignore/Qt.gitignore
@@ -20,6 +20,7 @@
 *.qbs.user.*
 *.moc
 moc_*.cpp
+moc_*.h
 qrc_*.cpp
 ui_*.h
 Makefile*
diff --git a/vendor/gitignore/UnrealEngine.gitignore b/vendor/gitignore/UnrealEngine.gitignore
index 2f096001fec46..6c6e1c327fd14 100644
--- a/vendor/gitignore/UnrealEngine.gitignore
+++ b/vendor/gitignore/UnrealEngine.gitignore
@@ -54,6 +54,11 @@ Binaries/*
 # Builds
 Build/*
 
+# Whitelist PakBlacklist-<BuildConfiguration>.txt files
+!Build/*/
+Build/*/**
+!Build/*/PakBlacklist*.txt
+
 # Don't ignore icon files in Build
 !Build/**/*.ico
 
-- 
GitLab


From 5c03b64b9d22f0d521b77389349caf0268f6e85c Mon Sep 17 00:00:00 2001
From: Timothy Andrew <mail@timothyandrew.net>
Date: Mon, 8 May 2017 17:03:48 +0000
Subject: [PATCH 359/363] Merge branch 'update-guides-for-9-2' into 'master'

Update guides for 9.2

See merge request !11157
---
 doc/install/installation.md |   4 +-
 doc/update/9.1-to-9.2.md    | 288 ++++++++++++++++++++++++++++++++++++
 2 files changed, 290 insertions(+), 2 deletions(-)
 create mode 100644 doc/update/9.1-to-9.2.md

diff --git a/doc/install/installation.md b/doc/install/installation.md
index dc807d93bbb9d..5615b2a534bae 100644
--- a/doc/install/installation.md
+++ b/doc/install/installation.md
@@ -289,9 +289,9 @@ sudo usermod -aG redis git
 ### Clone the Source
 
     # Clone GitLab repository
-    sudo -u git -H git clone https://gitlab.com/gitlab-org/gitlab-ce.git -b 9-1-stable gitlab
+    sudo -u git -H git clone https://gitlab.com/gitlab-org/gitlab-ce.git -b 9-2-stable gitlab
 
-**Note:** You can change `9-1-stable` to `master` if you want the *bleeding edge* version, but never install master on a production server!
+**Note:** You can change `9-2-stable` to `master` if you want the *bleeding edge* version, but never install master on a production server!
 
 ### Configure It
 
diff --git a/doc/update/9.1-to-9.2.md b/doc/update/9.1-to-9.2.md
new file mode 100644
index 0000000000000..19db6e5763efa
--- /dev/null
+++ b/doc/update/9.1-to-9.2.md
@@ -0,0 +1,288 @@
+# From 9.1 to 9.2
+
+Make sure you view this update guide from the tag (version) of GitLab you would
+like to install. In most cases this should be the highest numbered production
+tag (without rc in it). You can select the tag in the version dropdown at the
+top left corner of GitLab (below the menu bar).
+
+If the highest number stable branch is unclear please check the
+[GitLab Blog](https://about.gitlab.com/blog/archives.html) for installation
+guide links by version.
+
+### 1. Stop server
+
+```bash
+sudo service gitlab stop
+```
+
+### 2. Backup
+
+```bash
+cd /home/git/gitlab
+
+sudo -u git -H bundle exec rake gitlab:backup:create RAILS_ENV=production
+```
+
+### 3. Update Ruby
+
+NOTE: GitLab 9.0 and higher only support Ruby 2.3.x and dropped support for Ruby 2.1.x. Be
+sure to upgrade your interpreter if necessary.
+
+You can check which version you are running with `ruby -v`.
+
+Download and compile Ruby:
+
+```bash
+mkdir /tmp/ruby && cd /tmp/ruby
+curl --remote-name --progress https://cache.ruby-lang.org/pub/ruby/2.3/ruby-2.3.3.tar.gz
+echo '1014ee699071aa2ddd501907d18cbe15399c997d ruby-2.3.3.tar.gz' | shasum -c - && tar xzf ruby-2.3.3.tar.gz
+cd ruby-2.3.3
+./configure --disable-install-rdoc
+make
+sudo make install
+```
+
+Install Bundler:
+
+```bash
+sudo gem install bundler --no-ri --no-rdoc
+```
+
+### 4. Update Node
+
+GitLab now runs [webpack](http://webpack.js.org) to compile frontend assets and
+it has a minimum requirement of node v4.3.0.
+
+You can check which version you are running with `node -v`. If you are running
+a version older than `v4.3.0` you will need to update to a newer version.  You
+can find instructions to install from community maintained packages or compile
+from source at the nodejs.org website.
+
+<https://nodejs.org/en/download/>
+
+
+Since 8.17, GitLab requires the use of yarn `>= v0.17.0` to manage
+JavaScript dependencies.
+
+```bash
+curl --location https://yarnpkg.com/install.sh | bash -
+```
+
+More information can be found on the [yarn website](https://yarnpkg.com/en/docs/install).
+
+### 5. Get latest code
+
+```bash
+cd /home/git/gitlab
+
+sudo -u git -H git fetch --all
+sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically
+```
+
+For GitLab Community Edition:
+
+```bash
+cd /home/git/gitlab
+
+sudo -u git -H git checkout 9-2-stable
+```
+
+OR
+
+For GitLab Enterprise Edition:
+
+```bash
+cd /home/git/gitlab
+
+sudo -u git -H git checkout 9-2-stable-ee
+```
+
+### 6. Update gitlab-shell
+
+```bash
+cd /home/git/gitlab-shell
+
+sudo -u git -H git fetch --all --tags
+sudo -u git -H git checkout v$(</home/git/gitlab/GITLAB_SHELL_VERSION)
+sudo -u git -H bin/compile
+```
+
+### 7. Update gitlab-workhorse
+
+Install and compile gitlab-workhorse. This requires
+[Go 1.5](https://golang.org/dl) which should already be on your system from
+GitLab 8.1. GitLab-Workhorse uses [GNU Make](https://www.gnu.org/software/make/).
+If you are not using Linux you may have to run `gmake` instead of
+`make` below.
+
+```bash
+cd /home/git/gitlab-workhorse
+
+sudo -u git -H git fetch --all --tags
+sudo -u git -H git checkout v$(</home/git/gitlab/GITLAB_WORKHORSE_VERSION)
+sudo -u git -H make
+```
+
+### 8. Update configuration files
+
+#### New configuration options for `gitlab.yml`
+
+There might be configuration options available for [`gitlab.yml`][yaml]. View them with the command below and apply them manually to your current `gitlab.yml`:
+
+```sh
+cd /home/git/gitlab
+
+git diff origin/9-1-stable:config/gitlab.yml.example origin/9-2-stable:config/gitlab.yml.example
+```
+
+#### Nginx configuration
+
+Ensure you're still up-to-date with the latest NGINX configuration changes:
+
+```sh
+cd /home/git/gitlab
+
+# For HTTPS configurations
+git diff origin/9-1-stable:lib/support/nginx/gitlab-ssl origin/9-2-stable:lib/support/nginx/gitlab-ssl
+
+# For HTTP configurations
+git diff origin/9-1-stable:lib/support/nginx/gitlab origin/9-2-stable:lib/support/nginx/gitlab
+```
+
+If you are using Strict-Transport-Security in your installation to continue using it you must enable it in your Nginx
+configuration as GitLab application no longer handles setting it.
+
+If you are using Apache instead of NGINX please see the updated [Apache templates].
+Also note that because Apache does not support upstreams behind Unix sockets you
+will need to let gitlab-workhorse listen on a TCP port. You can do this
+via [/etc/default/gitlab].
+
+[Apache templates]: https://gitlab.com/gitlab-org/gitlab-recipes/tree/master/web-server/apache
+[/etc/default/gitlab]: https://gitlab.com/gitlab-org/gitlab-ce/blob/9-2-stable/lib/support/init.d/gitlab.default.example#L38
+
+#### SMTP configuration
+
+If you're installing from source and use SMTP to deliver mail, you will need to add the following line
+to config/initializers/smtp_settings.rb:
+
+```ruby
+ActionMailer::Base.delivery_method = :smtp
+```
+
+See [smtp_settings.rb.sample] as an example.
+
+[smtp_settings.rb.sample]: https://gitlab.com/gitlab-org/gitlab-ce/blob/9-1-stable/config/initializers/smtp_settings.rb.sample#L13
+
+#### Init script
+
+There might be new configuration options available for [`gitlab.default.example`][gl-example]. View them with the command below and apply them manually to your current `/etc/default/gitlab`:
+
+```sh
+cd /home/git/gitlab
+
+git diff origin/9-1-stable:lib/support/init.d/gitlab.default.example origin/9-2-stable:lib/support/init.d/gitlab.default.example
+```
+
+Ensure you're still up-to-date with the latest init script changes:
+
+```bash
+cd /home/git/gitlab
+
+sudo cp lib/support/init.d/gitlab /etc/init.d/gitlab
+```
+
+For Ubuntu 16.04.1 LTS:
+
+```bash
+sudo systemctl daemon-reload
+```
+
+### 9. Install libs, migrations, etc.
+
+```bash
+cd /home/git/gitlab
+
+# MySQL installations (note: the line below states '--without postgres')
+sudo -u git -H bundle install --without postgres development test --deployment
+
+# PostgreSQL installations (note: the line below states '--without mysql')
+sudo -u git -H bundle install --without mysql development test --deployment
+
+# Optional: clean up old gems
+sudo -u git -H bundle clean
+
+# Run database migrations
+sudo -u git -H bundle exec rake db:migrate RAILS_ENV=production
+
+# Update node dependencies and recompile assets
+sudo -u git -H bundle exec rake yarn:install gitlab:assets:clean gitlab:assets:compile RAILS_ENV=production NODE_ENV=production
+
+# Clean up cache
+sudo -u git -H bundle exec rake cache:clear RAILS_ENV=production
+```
+
+**MySQL installations**: Run through the `MySQL strings limits` and `Tables and data conversion to utf8mb4` [tasks](../install/database_mysql.md).
+
+### 10. Optional: install Gitaly
+
+Gitaly is still an optional component of GitLab. If you want to save time
+during your 9.2 upgrade **you can skip this step**.
+
+If you have not yet set up Gitaly then follow [Gitaly section of the installation
+guide](../install/installation.md#install-gitaly).
+
+#### Compile Gitaly
+
+```shell
+cd /home/git/gitaly
+sudo -u git -H git fetch --all --tags
+sudo -u git -H git checkout v$(</home/git/gitlab/GITALY_SERVER_VERSION)
+sudo -u git -H make
+```
+
+### 11. Start application
+
+```bash
+sudo service gitlab start
+sudo service nginx restart
+```
+
+### 12. Check application status
+
+Check if GitLab and its environment are configured correctly:
+
+```bash
+cd /home/git/gitlab
+
+sudo -u git -H bundle exec rake gitlab:env:info RAILS_ENV=production
+```
+
+To make sure you didn't miss anything run a more thorough check:
+
+```bash
+cd /home/git/gitlab
+
+sudo -u git -H bundle exec rake gitlab:check RAILS_ENV=production
+```
+
+If all items are green, then congratulations, the upgrade is complete!
+
+## Things went south? Revert to previous version (9.1)
+
+### 1. Revert the code to the previous version
+
+Follow the [upgrade guide from 9.0 to 9.1](9.0-to-9.1.md), except for the
+database migration (the backup is already migrated to the previous version).
+
+### 2. Restore from the backup
+
+```bash
+cd /home/git/gitlab
+
+sudo -u git -H bundle exec rake gitlab:backup:restore RAILS_ENV=production
+```
+
+If you have more than one backup `*.tar` file(s) please add `BACKUP=timestamp_of_backup` to the command above.
+
+[yaml]: https://gitlab.com/gitlab-org/gitlab-ce/blob/9-2-stable/config/gitlab.yml.example
+[gl-example]: https://gitlab.com/gitlab-org/gitlab-ce/blob/9-2-stable/lib/support/init.d/gitlab.default.example
-- 
GitLab


From 848466ee6c42186b4e8a3c49027438254c476ae4 Mon Sep 17 00:00:00 2001
From: Sean McGivern <sean@mcgivern.me.uk>
Date: Mon, 8 May 2017 11:06:54 +0000
Subject: [PATCH 360/363] Merge branch 'sh-fix-almost-there-spec-mysql' into
 'master'

Fix sub-second timing comparison error for Devise confirmation period

Closes gitlab-ee#2362

See merge request !11156
---
 app/models/user.rb | 9 +++++++++
 1 file changed, 9 insertions(+)

diff --git a/app/models/user.rb b/app/models/user.rb
index accaa91b8056e..4e5f94683b826 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -1000,6 +1000,15 @@ def send_devise_notification(notification, *args)
     devise_mailer.send(notification, self, *args).deliver_later
   end
 
+  # This works around a bug in Devise 4.2.0 that erroneously causes a user to
+  # be considered active in MySQL specs due to a sub-second comparison
+  # issue. For more details, see: https://gitlab.com/gitlab-org/gitlab-ee/issues/2362#note_29004709
+  def confirmation_period_valid?
+    return false if self.class.allow_unconfirmed_access_for == 0.days
+
+    super
+  end
+
   def ensure_external_user_rights
     return unless external?
 
-- 
GitLab


From 3d6086d0ee5d16e77be5f83c0b19d6fe4d4b6c43 Mon Sep 17 00:00:00 2001
From: Sean McGivern <sean@mcgivern.me.uk>
Date: Mon, 8 May 2017 12:16:15 +0000
Subject: [PATCH 361/363] Merge branch
 'fix-notes_on_personal_snippets_spec-timeago-assertion-ce' into 'master'

Fix notes_on_personal_snippets_spec

Closes #31938

See merge request !11160
---
 spec/features/snippets/notes_on_personal_snippets_spec.rb | 4 +++-
 1 file changed, 3 insertions(+), 1 deletion(-)

diff --git a/spec/features/snippets/notes_on_personal_snippets_spec.rb b/spec/features/snippets/notes_on_personal_snippets_spec.rb
index 957baac02eb00..698eb46573fc1 100644
--- a/spec/features/snippets/notes_on_personal_snippets_spec.rb
+++ b/spec/features/snippets/notes_on_personal_snippets_spec.rb
@@ -78,9 +78,11 @@
       end
 
       page.within("#notes-list li#note_#{snippet_notes[0].id}") do
+        edited_text = find('.edited-text')
+
         expect(page).to have_css('.note_edited_ago')
         expect(page).to have_content('new content')
-        expect(find('.note_edited_ago').text).to match(/less than a minute ago/)
+        expect(edited_text).to have_selector('.note_edited_ago')
       end
     end
   end
-- 
GitLab


From 9632e01d986390282353017bdf31534c9c5eb924 Mon Sep 17 00:00:00 2001
From: Sean McGivern <sean@mcgivern.me.uk>
Date: Mon, 8 May 2017 11:51:26 +0000
Subject: [PATCH 362/363] Merge branch
 'Add-index_redirect_routes_path_for_link-migration-to-setup_postgresql' into
 'master'

Add index_redirect_routes_path_for_link migration to setup_postgresql

See merge request !11165
---
 lib/tasks/migrate/setup_postgresql.rake | 1 +
 1 file changed, 1 insertion(+)

diff --git a/lib/tasks/migrate/setup_postgresql.rake b/lib/tasks/migrate/setup_postgresql.rake
index 1e00b47303d5f..4108cee08b4d0 100644
--- a/lib/tasks/migrate/setup_postgresql.rake
+++ b/lib/tasks/migrate/setup_postgresql.rake
@@ -4,6 +4,7 @@ require Rails.root.join('db/migrate/20151007120511_namespaces_projects_path_lowe
 require Rails.root.join('db/migrate/20151008110232_add_users_lower_username_email_indexes')
 require Rails.root.join('db/migrate/20161212142807_add_lower_path_index_to_routes')
 require Rails.root.join('db/migrate/20170317203554_index_routes_path_for_like')
+require Rails.root.join('db/migrate/20170503185032_index_redirect_routes_path_for_like')
 
 desc 'GitLab | Sets up PostgreSQL'
 task setup_postgresql: :environment do
-- 
GitLab


From fa0596a9ca5e81498e4485637ac53dec227f5c72 Mon Sep 17 00:00:00 2001
From: Timothy Andrew <mail@timothyandrew.net>
Date: Tue, 9 May 2017 06:11:39 +0000
Subject: [PATCH 363/363] Merge branch `ce_upstream` into `master`

- https://gitlab.slack.com/archives/C0XM5UU6B/p1494309973105903
- `ce_upstream` has was green at ef0b3a56892896a56193a581ee0174716a4b996e,
  but the MR was _not_ merged in, and has since been refreshed (and thus
  has more conflicts to fix).
- For the purposes of `9-2-stable-ee`, we don't need the _latest_
  `ce_upstream`, or the latest `ee/master`, simply the newest version
  since `9-2-stable-ee` was branched off.
- We merge `ef0b3a` into the latest version of `ee/master` that it
  merges to without conflicts, and then pick that merge commit into
  `9-2-stable-ee`
---
 .eslintignore                                 |   1 +
 .gitignore                                    |   3 +
 .gitlab-ci.yml                                |  35 +-
 CHANGELOG.md                                  |  35 +
 CONTRIBUTING.md                               | 219 +++++--
 Gemfile                                       |   6 +
 Gemfile.lock                                  |  19 +
 PROCESS.md                                    |  95 ++-
 app/assets/javascripts/autosave.js            |  44 +-
 .../behaviors/gl_emoji/unicode_support_map.js |  17 +-
 .../javascripts/behaviors/quick_submit.js     |   2 +-
 .../blob/balsamiq/balsamiq_viewer.js          | 114 ++++
 .../javascripts/blob/balsamiq_viewer.js       |   6 +
 .../blob/file_template_mediator.js            |   6 +-
 .../blob/file_template_selector.js            |  10 +-
 .../blob/target_branch_dropdown.js            |   4 +-
 .../javascripts/blob/template_selector.js     |   7 +-
 .../template_selectors/ci_yaml_selector.js    |   2 +-
 .../template_selectors/dockerfile_selector.js |   2 +-
 .../template_selectors/gitignore_selector.js  |   2 +-
 .../template_selectors/license_selector.js    |  13 +-
 .../blob/template_selectors/type_selector.js  |   2 +-
 .../javascripts/boards/boards_bundle.js       |   5 +-
 .../boards/components/board_new_issue.js      |   1 +
 .../boards/components/board_sidebar.js        |  61 +-
 .../boards/components/issue_card_inner.js     | 104 ++-
 .../boards/components/new_list_dropdown.js    |   4 +-
 .../javascripts/boards/models/assignee.js     |  12 +
 app/assets/javascripts/boards/models/issue.js |  34 +-
 app/assets/javascripts/boards/models/list.js  |   5 +-
 app/assets/javascripts/boards/models/user.js  |  12 -
 .../javascripts/boards/stores/boards_store.js |   4 +-
 .../components/limit_warning_component.js     |   4 +-
 .../components/stage_code_component.js        |   4 +-
 .../components/stage_issue_component.js       |   4 +-
 .../components/stage_plan_component.js        |   4 +-
 .../components/stage_production_component.js  |   4 +-
 .../components/stage_review_component.js      |   4 +-
 .../components/stage_staging_component.js     |   2 +-
 .../components/total_time_component.js        |   8 +-
 .../cycle_analytics/cycle_analytics_bundle.js |   3 +
 .../cycle_analytics_service.js                |   2 +-
 .../cycle_analytics/cycle_analytics_store.js  |  17 +-
 .../deploy_keys/components/action_btn.vue     |  54 ++
 .../deploy_keys/components/app.vue            | 102 +++
 .../deploy_keys/components/key.vue            |  80 +++
 .../deploy_keys/components/keys_panel.vue     |  52 ++
 .../javascripts/deploy_keys/eventhub.js       |   3 +
 app/assets/javascripts/deploy_keys/index.js   |  21 +
 .../javascripts/deploy_keys/service/index.js  |  34 +
 .../javascripts/deploy_keys/store/index.js    |   9 +
 app/assets/javascripts/dispatcher.js          |   6 +
 app/assets/javascripts/droplab/constants.js   |   3 +
 app/assets/javascripts/droplab/drop_down.js   |   2 +-
 app/assets/javascripts/droplab/utils.js       |  16 +-
 .../recent_searches_dropdown_content.js       |  12 +-
 .../filtered_search/dropdown_hint.js          |   2 +-
 .../filtered_search_manager.js                |  11 +-
 .../filtered_search_visual_tokens.js          |  43 +-
 .../filtered_search/recent_searches_root.js   |   7 +-
 .../services/recent_searches_service.js       |  14 +
 .../services/recent_searches_service_error.js |  11 +
 app/assets/javascripts/gfm_auto_complete.js   | 169 +++--
 app/assets/javascripts/gl_dropdown.js         |  56 +-
 .../issuable/auto_width_dropdown_select.js    |  38 ++
 .../javascripts/issuable/issuable_bundle.js   |   1 -
 .../components/collapsed_state.js             |  60 --
 .../components/comparison_pane.js             |  82 ---
 .../components/estimate_only_pane.js          |  19 -
 .../time_tracking/components/help_state.js    |  30 -
 .../components/no_tracking_pane.js            |  12 -
 .../components/spent_only_pane.js             |  19 -
 .../time_tracking/components/time_tracker.js  | 135 ----
 .../time_tracking/time_tracking_bundle.js     |  66 --
 app/assets/javascripts/issue_status_select.js |   4 +-
 .../javascripts/issues_bulk_assignment.js     |   3 +
 app/assets/javascripts/labels_select.js       |   7 +-
 app/assets/javascripts/lib/utils/accessor.js  |  47 ++
 .../javascripts/lib/utils/ajax_cache.js       |  32 +
 .../javascripts/lib/utils/common_utils.js     |   8 +
 app/assets/javascripts/locale/de/app.js       |   1 +
 app/assets/javascripts/locale/en/app.js       |   1 +
 app/assets/javascripts/locale/es/app.js       |   1 +
 app/assets/javascripts/locale/index.js        |  70 ++
 app/assets/javascripts/main.js                |   1 -
 app/assets/javascripts/members.js             |   4 +-
 .../javascripts/merge_request_widget.js       |   2 +-
 app/assets/javascripts/milestone_select.js    |   5 +-
 app/assets/javascripts/namespace_select.js    |   3 +-
 app/assets/javascripts/new_branch_form.js     |  16 +
 app/assets/javascripts/notes.js               | 345 ++++++++--
 .../pipelines/services/pipelines_service.js   |   2 +-
 app/assets/javascripts/project.js             |   3 +-
 .../protected_branch_access_dropdown.js       |   5 +-
 .../protected_branch_dropdown.js              |   3 +-
 .../protected_tag_access_dropdown.js          |   4 +-
 .../protected_tags/protected_tag_dropdown.js  |   4 +-
 app/assets/javascripts/raven/index.js         |  16 +
 app/assets/javascripts/raven/raven_config.js  | 100 +++
 .../components/assignees/assignee_title.js    |  41 ++
 .../sidebar/components/assignees/assignees.js | 224 +++++++
 .../components/assignees/sidebar_assignees.js |  82 +++
 .../time_tracking/collapsed_state.js          |  97 +++
 .../time_tracking/comparison_pane.js          |  98 +++
 .../time_tracking/estimate_only_pane.js       |  17 +
 .../components/time_tracking/help_state.js    |  44 ++
 .../time_tracking/no_tracking_pane.js         |  10 +
 .../time_tracking/sidebar_time_tracking.js    |  51 ++
 .../time_tracking/spent_only_pane.js          |  15 +
 .../components/time_tracking/time_tracker.js  | 163 +++++
 app/assets/javascripts/sidebar/event_hub.js   |   3 +
 .../sidebar/services/sidebar_service.js       |  28 +
 .../javascripts/sidebar/sidebar_bundle.js     |  24 +
 .../javascripts/sidebar/sidebar_mediator.js   |  38 ++
 .../sidebar/stores/sidebar_store.js           |  52 ++
 .../javascripts/signin_tabs_memoizer.js       |  12 +-
 app/assets/javascripts/subbable_resource.js   |  51 --
 app/assets/javascripts/subscription_select.js |   4 +-
 app/assets/javascripts/users_select.js        | 403 +++++++++---
 .../javascripts/vue_shared/translate.js       |  42 ++
 app/assets/javascripts/weight_select.js       |   5 +-
 .../stylesheets/framework/animations.scss     |  28 +
 app/assets/stylesheets/framework/avatar.scss  |  11 +
 .../stylesheets/framework/dropdowns.scss      |  16 +-
 app/assets/stylesheets/framework/files.scss   |  12 +
 app/assets/stylesheets/framework/filters.scss |  14 +-
 app/assets/stylesheets/framework/lists.scss   |   1 +
 .../stylesheets/framework/variables.scss      |   2 +
 app/assets/stylesheets/pages/boards.scss      |  73 ++-
 .../stylesheets/pages/cycle_analytics.scss    |  31 +-
 app/assets/stylesheets/pages/diff.scss        |   9 +-
 app/assets/stylesheets/pages/issuable.scss    | 125 +++-
 app/assets/stylesheets/pages/labels.scss      |   2 +-
 .../stylesheets/pages/merge_requests.scss     |   4 +
 app/assets/stylesheets/pages/notes.scss       |  39 +-
 app/assets/stylesheets/pages/pipelines.scss   |  27 +
 app/assets/stylesheets/pages/todos.scss       |   3 +-
 .../admin/application_settings_controller.rb  |   2 +
 app/controllers/admin/services_controller.rb  |   2 +
 app/controllers/application_controller.rb     |  10 +
 app/controllers/concerns/issuable_actions.rb  |   1 +
 .../concerns/issuable_collections.rb          |   2 +-
 .../dashboard/labels_controller.rb            |   2 +-
 app/controllers/groups/labels_controller.rb   |   2 +-
 .../omniauth_callbacks_controller.rb          |   2 +-
 app/controllers/profiles_controller.rb        |   3 +-
 .../projects/artifacts_controller.rb          |  28 +-
 .../projects/boards/issues_controller.rb      |   2 +-
 .../projects/deploy_keys_controller.rb        |  18 +-
 app/controllers/projects/issues_controller.rb |   8 +-
 app/controllers/projects/labels_controller.rb |   2 +-
 .../projects/pipelines_controller.rb          |  40 +-
 app/finders/issuable_finder.rb                |   2 +-
 app/finders/issues_finder.rb                  |  19 +-
 app/helpers/application_helper.rb             |  16 +-
 app/helpers/blob_helper.rb                    |   6 +-
 app/helpers/boards_helper.rb                  |   1 +
 app/helpers/builds_helper.rb                  |  12 +
 app/helpers/form_helper.rb                    |  32 +
 app/helpers/gitlab_routing_helper.rb          |   2 +
 app/helpers/issuables_helper.rb               |  10 +
 app/helpers/sorting_helper.rb                 |   8 +
 app/helpers/system_note_helper.rb             |   1 +
 app/mailers/emails/issues.rb                  |   6 +-
 app/models/application_setting.rb             |   4 +
 app/models/blob.rb                            |   1 +
 app/models/blob_viewer/balsamiq.rb            |  12 +
 app/models/ci/artifact_blob.rb                |  35 +
 app/models/concerns/elastic/issues_search.rb  |  11 +-
 app/models/concerns/elastic/notes_search.rb   |   2 +-
 app/models/concerns/issuable.rb               |  29 +-
 app/models/concerns/milestoneish.rb           |   2 +-
 app/models/global_milestone.rb                |   6 +-
 app/models/issue.rb                           |  35 +-
 app/models/issue_assignee.rb                  |  29 +
 app/models/merge_request.rb                   |  31 +
 app/models/milestone.rb                       |   5 +-
 app/models/note.rb                            |   6 +
 app/models/snippet.rb                         |   5 +
 app/models/system_note_metadata.rb            |   2 +-
 app/models/user.rb                            |   8 +-
 .../settings/deploy_keys_presenter.rb         |  11 +
 app/serializers/README.md                     | 325 ++++++++++
 app/serializers/analytics_stage_entity.rb     |   1 +
 app/serializers/analytics_summary_entity.rb   |   5 +-
 app/serializers/deploy_key_entity.rb          |  14 +
 app/serializers/deploy_key_serializer.rb      |   3 +
 app/serializers/issuable_entity.rb            |   1 -
 app/serializers/issue_entity.rb               |   1 +
 app/serializers/label_entity.rb               |   1 +
 app/serializers/label_serializer.rb           |   7 +
 app/serializers/merge_request_entity.rb       |   1 +
 app/serializers/project_entity.rb             |  14 +
 app/services/issuable/bulk_update_service.rb  |   6 +-
 app/services/issuable_base_service.rb         |  39 +-
 app/services/issues/base_service.rb           |  19 +
 app/services/issues/export_csv_service.rb     |   6 +-
 app/services/issues/update_service.rb         |  14 +-
 .../members/authorized_destroy_service.rb     |  15 +-
 .../merge_requests/assign_issues_service.rb   |   4 +-
 app/services/merge_requests/base_service.rb   |   5 +
 app/services/merge_requests/update_service.rb |   5 +-
 .../notification_recipient_service.rb         |   7 +-
 app/services/notification_service.rb          |  29 +-
 .../projects/propagate_service_template.rb    | 103 +++
 .../slash_commands/interpret_service.rb       |  33 +-
 app/services/system_note_service.rb           |  55 ++
 app/services/todo_service.rb                  |   4 +-
 .../application_settings/_form.html.haml      |  18 +-
 app/views/discussions/_notes.html.haml        |   1 +
 app/views/errors/omniauth_error.html.haml     |  21 +-
 app/views/issues/_issue.atom.builder          |  15 +-
 app/views/layouts/_head.html.haml             |   3 +
 app/views/layouts/application.html.haml       |   4 +-
 app/views/layouts/devise.html.haml            |   1 -
 app/views/layouts/devise_empty.html.haml      |   1 -
 app/views/layouts/oauth_error.html.haml       | 127 ++++
 .../_reassigned_issuable_email.text.erb       |   6 -
 app/views/notify/new_issue_email.html.haml    |   4 +-
 app/views/notify/new_issue_email.text.erb     |   2 +-
 .../new_mention_in_issue_email.text.erb       |   2 +-
 .../notify/reassigned_issue_email.html.haml   |  11 +-
 .../notify/reassigned_issue_email.text.erb    |   7 +-
 .../reassigned_merge_request_email.html.haml  |  10 +-
 .../reassigned_merge_request_email.text.erb   |   7 +-
 app/views/profiles/show.html.haml             |   5 +
 .../projects/artifacts/_tree_file.html.haml   |   9 +-
 app/views/projects/artifacts/file.html.haml   |  33 +
 .../projects/blob/viewers/_balsamiq.html.haml |   4 +
 .../boards/components/_board.html.haml        |   4 +-
 .../components/sidebar/_assignee.html.haml    |  46 +-
 app/views/projects/branches/new.html.haml     |  12 +-
 app/views/projects/compare/_form.html.haml    |   4 +-
 .../cycle_analytics/_empty_stage.html.haml    |   2 +-
 .../cycle_analytics/_no_access.html.haml      |   4 +-
 .../projects/cycle_analytics/show.html.haml   |  53 +-
 .../projects/deploy_keys/_index.html.haml     |  23 +-
 .../projects/group_links/_index.html.haml     |   4 +-
 app/views/projects/issues/_issue.html.haml    |   4 +-
 .../projects/merge_requests/_show.html.haml   |   2 +-
 app/views/projects/notes/_actions.html.haml   |   6 +-
 .../projects/pipelines/_with_tabs.html.haml   |  19 +-
 .../projects/project_members/_index.html.haml |   2 +-
 .../_create_protected_branch.html.haml        |   4 +-
 .../_update_protected_branch.html.haml        |   4 +-
 .../protected_tags/_dropdown.html.haml        |   2 +-
 .../protected_tags/_update_protected_tag.haml |   2 +-
 .../settings/repository/show.html.haml        |   4 +
 app/views/projects/tags/index.html.haml       |  17 +-
 .../_ref_dropdown.html.haml                   |   4 +-
 app/views/shared/errors/_graphic_422.svg      |   1 +
 .../shared/issuable/_assignees.html.haml      |  15 +
 .../shared/issuable/_participants.html.haml   |   8 +-
 .../shared/issuable/_search_bar.html.haml     |   7 +-
 app/views/shared/issuable/_sidebar.html.haml  |  94 +--
 .../issuable/form/_branch_chooser.html.haml   |  10 +-
 .../issuable/form/_issue_assignee.html.haml   |  30 +
 .../form/_merge_request_assignee.html.haml    |  31 +
 .../shared/issuable/form/_metadata.html.haml  |  28 +-
 app/views/shared/members/_requests.html.haml  |   2 +-
 .../shared/milestones/_issuable.html.haml     |   8 +-
 .../shared/milestones/_labels_tab.html.haml   |  14 +-
 app/views/shared/notes/_edit_form.html.haml   |   2 +-
 app/views/shared/notes/_note.html.haml        |   2 +-
 app/views/shared/snippets/_header.html.haml   |   2 +-
 app/views/snippets/notes/_actions.html.haml   |   6 +-
 app/workers/elastic_indexer_worker.rb         |   2 +-
 .../propagate_service_template_worker.rb      |  21 +
 .../24883-build-failure-summary-page.yml      |   4 +
 .../unreleased/27614-instant-comments.yml     |   4 +
 changelogs/unreleased/29145-oauth-422.yml     |   4 +
 .../30007-done-todo-hover-state.yml           |   4 +
 .../30903-vertically-align-mini-pipeline.yml  |   4 +
 ...-network-graph-sorted-by-date-and-topo.yml |   4 +
 .../31689-request-access-spacing.yml          |   4 +
 .../31760-add-tooltips-to-note-actions.yml    |   4 +
 changelogs/unreleased/31810-commit-link.yml   |   4 +
 .../add_system_note_for_editing_issuable.yml  |   4 +
 changelogs/unreleased/balsalmiq-support.yml   |   4 +
 .../unreleased/deploy-keys-load-async.yml     |   4 +
 .../unreleased/dm-artifact-blob-viewer.yml    |   4 +
 .../unreleased/fix-admin-integrations.yml     |   4 +
 .../unreleased/implement-i18n-support.yml     |   4 +
 .../unreleased/issue-boards-no-avatar.yml     |   4 +
 .../issue-title-description-realtime.yml      |   4 +
 .../merge-request-poll-json-endpoint.yml      |   4 +
 .../mrchrisw-import-shell-timeout.yml         |   4 +
 ...rometheus-integration-test-setting-fix.yml |   4 +
 changelogs/unreleased/tags-sort-default.yml   |   4 +
 .../update-issue-board-cards-design.yml       |   4 +
 .../unreleased/winh-visual-token-labels.yml   |   4 +
 config/application.rb                         |   3 +
 config/gitlab.yml.example                     |   3 +
 config/initializers/1_settings.rb             |   1 +
 config/initializers/fast_gettext.rb           |   5 +
 .../initializers/gettext_rails_i18n_patch.rb  |  42 ++
 config/locales/de.yml                         | 219 +++++++
 config/locales/es.yml                         | 217 +++++++
 config/routes/project.rb                      |   2 +
 config/sidekiq_queues.yml                     |   2 +-
 config/webpack.config.js                      |  25 +-
 db/fixtures/development/09_issues.rb          |   2 +-
 db/fixtures/development/20_burndown.rb        |   2 +-
 ...0320171632_create_issue_assignees_table.rb |  40 ++
 .../20170320173259_migrate_assignees.rb       |  52 ++
 ...3035209_add_preferred_language_to_users.rb |  16 +
 ...ited_at_and_last_edited_by_id_to_issues.rb |  14 +
 ...and_last_edited_by_id_to_merge_requests.rb |  14 +
 ...ientside_sentry_to_application_settings.rb |  33 +
 db/schema.rb                                  |  21 +-
 doc/api/issues.md                             | 104 ++-
 doc/development/README.md                     |   1 +
 doc/development/build_test_package.md         |  35 +
 doc/development/code_review.md                |  41 +-
 doc/development/fe_guide/droplab/droplab.md   |   2 +
 doc/development/fe_guide/style_guide_js.md    | 602 ++++++++++--------
 doc/development/fe_guide/vue.md               |  15 +
 doc/user/project/integrations/webhooks.md     |  12 +
 features/project/builds/artifacts.feature     |   3 +-
 features/project/deploy_keys.feature          |   6 +
 features/steps/dashboard/dashboard.rb         |   2 +-
 features/steps/dashboard/todos.rb             |   2 +-
 features/steps/group/milestones.rb            |   4 +-
 features/steps/groups.rb                      |   4 +-
 features/steps/project/builds/artifacts.rb    |  15 +-
 features/steps/project/deploy_keys.rb         |  16 +-
 features/steps/project/merge_requests.rb      |   7 +
 features/steps/shared/note.rb                 |   4 +
 lib/api/api.rb                                |   5 +
 lib/api/entities.rb                           |   6 +-
 lib/api/helpers/common_helpers.rb             |  13 +
 lib/api/issues.rb                             |   9 +-
 lib/api/settings.rb                           |   5 +
 lib/api/v3/entities.rb                        |   7 +
 lib/api/v3/issues.rb                          |  31 +-
 lib/api/v3/merge_requests.rb                  |   2 +-
 lib/api/v3/milestones.rb                      |   4 +-
 lib/banzai/reference_parser/issue_parser.rb   |   2 +-
 lib/github/import.rb                          |   2 +-
 .../chat_commands/presenters/issue_base.rb    |   2 +-
 .../ci/build/artifacts/metadata/entry.rb      |   6 +
 lib/gitlab/cycle_analytics/base_stage.rb      |   2 +-
 lib/gitlab/cycle_analytics/code_stage.rb      |   8 +-
 lib/gitlab/cycle_analytics/issue_stage.rb     |   8 +-
 lib/gitlab/cycle_analytics/plan_stage.rb      |   8 +-
 .../cycle_analytics/production_stage.rb       |   8 +-
 lib/gitlab/cycle_analytics/review_stage.rb    |   8 +-
 lib/gitlab/cycle_analytics/staging_stage.rb   |   8 +-
 lib/gitlab/cycle_analytics/summary/base.rb    |   2 +-
 lib/gitlab/cycle_analytics/summary/commit.rb  |   4 +
 lib/gitlab/cycle_analytics/summary/deploy.rb  |   4 +
 lib/gitlab/cycle_analytics/summary/issue.rb   |   2 +-
 lib/gitlab/cycle_analytics/test_stage.rb      |   8 +-
 lib/gitlab/fogbugz_import/importer.rb         |  18 +-
 lib/gitlab/git/repository.rb                  |  17 +-
 lib/gitlab/github_import/issue_formatter.rb   |   2 +-
 lib/gitlab/gon_helper.rb                      |   2 +
 lib/gitlab/google_code_import/importer.rb     |  14 +-
 lib/gitlab/i18n.rb                            |  26 +
 lib/gitlab/import_export/relation_factory.rb  |   2 +-
 lib/gitlab/prometheus.rb                      |   6 +
 lib/gitlab/shell.rb                           |   4 +-
 lib/tasks/gettext.rake                        |  14 +
 locale/de/gitlab.po                           | 207 ++++++
 locale/de/gitlab.po.time_stamp                |   0
 locale/en/gitlab.po                           | 207 ++++++
 locale/en/gitlab.po.time_stamp                |   0
 locale/es/gitlab.po                           | 208 ++++++
 locale/es/gitlab.po.time_stamp                |   0
 locale/gitlab.pot                             | 208 ++++++
 package.json                                  |   4 +
 scripts/static-analysis                       |   1 +
 .../admin/services_controller_spec.rb         |  32 +
 .../dashboard/todos_controller_spec.rb        |   2 +-
 .../projects/artifacts_controller_spec.rb     | 188 ++++++
 .../projects/boards/issues_controller_spec.rb |   2 +-
 .../projects/deploy_keys_controller_spec.rb   |  66 ++
 .../projects/issues_controller_spec.rb        |   8 +-
 .../merge_requests_controller_spec.rb         |   2 +-
 .../projects/pipelines_controller_spec.rb     |  36 ++
 spec/features/atom/dashboard_issues_spec.rb   |   8 +-
 spec/features/atom/issues_spec.rb             |   8 +-
 .../boards/board_with_milestone_spec.rb       |   2 +-
 spec/features/boards/boards_spec.rb           |   2 +-
 spec/features/boards/modal_filter_spec.rb     |   2 +-
 spec/features/boards/sidebar_spec.rb          |  35 +-
 spec/features/cycle_analytics_spec.rb         |  19 +
 .../dashboard/issuables_counter_spec.rb       |   6 +-
 spec/features/dashboard/issues_spec.rb        |   7 +-
 spec/features/dashboard_issues_spec.rb        |   4 +-
 .../features/gitlab_flavored_markdown_spec.rb |   4 +-
 spec/features/issues/award_emoji_spec.rb      |   2 +-
 spec/features/issues/csv_spec.rb              |   2 +-
 .../filtered_search/filter_issues_spec.rb     |  10 +-
 .../filter_issues_weight_spec.rb              |   2 +-
 spec/features/issues/form_spec.rb             |  69 +-
 spec/features/issues/issue_sidebar_spec.rb    |  15 +
 spec/features/issues/update_issues_spec.rb    |   2 +-
 spec/features/issues_spec.rb                  |  42 +-
 .../merge_requests/assign_issues_spec.rb      |   2 +-
 .../merge_requests/user_posts_notes_spec.rb   |   1 +
 .../user_uses_slash_commands_spec.rb          |   1 +
 spec/features/milestones/show_spec.rb         |   4 +-
 spec/features/projects/artifacts/file_spec.rb |  59 ++
 spec/features/projects/audit_events_spec.rb   |   2 +-
 .../branches/new_branch_ref_dropdown_spec.rb  |  48 ++
 spec/features/projects/deploy_keys_spec.rb    |  12 +-
 .../projects/issuable_templates_spec.rb       |   4 +-
 .../projects/pipelines/pipeline_spec.rb       |  53 ++
 spec/features/raven_js_spec.rb                |  23 +
 spec/features/search_spec.rb                  |   2 +-
 spec/features/unsubscribe_links_spec.rb       |   2 +-
 spec/finders/issues_finder_spec.rb            |  12 +-
 spec/fixtures/api/schemas/issue.json          |  50 +-
 .../api/schemas/public_api/v4/issues.json     |  17 +-
 spec/helpers/issuables_helper_spec.rb         |  17 +
 spec/javascripts/autosave_spec.js             | 134 ++++
 .../gl_emoji/unicode_support_map_spec.js      |  47 ++
 .../blob/balsamiq/balsamiq_viewer_spec.js     | 342 ++++++++++
 spec/javascripts/boards/board_card_spec.js    |   8 +-
 spec/javascripts/boards/board_list_spec.js    |   1 +
 spec/javascripts/boards/boards_store_spec.js  |  19 +-
 spec/javascripts/boards/issue_card_spec.js    | 131 +++-
 spec/javascripts/boards/issue_spec.js         |  76 ++-
 spec/javascripts/boards/list_spec.js          |  25 +-
 spec/javascripts/boards/mock_data.js          |   3 +-
 spec/javascripts/boards/modal_store_spec.js   |  12 +-
 .../limit_warning_component_spec.js           |   3 +
 .../deploy_keys/components/action_btn_spec.js |  70 ++
 .../deploy_keys/components/app_spec.js        | 142 +++++
 .../deploy_keys/components/key_spec.js        |  92 +++
 .../deploy_keys/components/keys_panel_spec.js |  70 ++
 spec/javascripts/droplab/constants_spec.js    |   6 +
 spec/javascripts/droplab/drop_down_spec.js    |   4 +-
 .../recent_searches_dropdown_content_spec.js  |  20 +
 .../filtered_search_manager_spec.js           |  34 +
 .../filtered_search_visual_tokens_spec.js     | 101 +++
 .../recent_searches_root_spec.js              |  31 +
 .../recent_searches_service_error_spec.js     |  18 +
 .../services/recent_searches_service_spec.js  |  95 ++-
 spec/javascripts/fixtures/deploy_keys.rb      |  36 ++
 spec/javascripts/fixtures/labels.rb           |  56 ++
 .../helpers/user_mock_data_helper.js          |  16 +
 .../javascripts/issuable_time_tracker_spec.js | 250 ++++----
 spec/javascripts/lib/utils/accessor_spec.js   |  78 +++
 spec/javascripts/lib/utils/ajax_cache_spec.js | 129 ++++
 .../lib/utils/common_utils_spec.js            |  11 +
 spec/javascripts/notes_spec.js                | 192 +++++-
 spec/javascripts/raven/index_spec.js          |  42 ++
 spec/javascripts/raven/raven_config_spec.js   | 276 ++++++++
 .../sidebar/assignee_title_spec.js            |  80 +++
 spec/javascripts/sidebar/assignees_spec.js    | 272 ++++++++
 spec/javascripts/sidebar/mock_data.js         | 109 ++++
 .../sidebar/sidebar_assignees_spec.js         |  45 ++
 .../sidebar/sidebar_bundle_spec.js            |  42 ++
 .../sidebar/sidebar_mediator_spec.js          |  40 ++
 .../sidebar/sidebar_service_spec.js           |  32 +
 .../javascripts/sidebar/sidebar_store_spec.js |  80 +++
 spec/javascripts/signin_tabs_memoizer_spec.js |  90 +++
 spec/javascripts/subbable_resource_spec.js    |  63 --
 spec/javascripts/vue_shared/translate_spec.js |  90 +++
 .../lib/banzai/filter/redactor_filter_spec.rb |   2 +-
 .../ci/build/artifacts/metadata/entry_spec.rb |  11 +
 .../elastic/project_search_results_spec.rb    |   2 +-
 .../lib/gitlab/elastic/search_results_spec.rb |   4 +-
 spec/lib/gitlab/git/repository_spec.rb        |   2 +-
 .../github_import/issue_formatter_spec.rb     |  10 +-
 .../google_code_import/importer_spec.rb       |   2 +-
 .../health_checks/fs_shards_check_spec.rb     |  12 +-
 .../health_checks/simple_check_shared.rb      |   6 +-
 spec/lib/gitlab/i18n_spec.rb                  |  27 +
 spec/lib/gitlab/import_export/all_models.yml  |   6 +-
 .../import_export/project_tree_saver_spec.rb  |   2 +-
 .../import_export/safe_model_attributes.yml   |   4 +
 .../lib/gitlab/project_search_results_spec.rb |   2 +-
 spec/lib/gitlab/prometheus_spec.rb            |  30 +
 spec/lib/gitlab/search_results_spec.rb        |   4 +-
 spec/lib/gitlab/shell_spec.rb                 |  41 ++
 spec/mailers/notify_spec.rb                   |  10 +-
 spec/models/ci/artifact_blob_spec.rb          |  44 ++
 spec/models/concerns/elastic/issue_spec.rb    |   7 +-
 spec/models/concerns/issuable_spec.rb         | 109 +---
 spec/models/concerns/milestoneish_spec.rb     |   6 +-
 spec/models/event_spec.rb                     |   4 +-
 spec/models/issue_collection_spec.rb          |   2 +-
 spec/models/issue_spec.rb                     |  91 ++-
 spec/models/merge_request_spec.rb             |  91 ++-
 spec/models/network/graph_spec.rb             |  21 +-
 .../prometheus_service_spec.rb                |   2 +-
 spec/models/user_spec.rb                      |   8 +
 spec/policies/issue_policy_spec.rb            |   8 +-
 spec/requests/api/issues_spec.rb              |  70 +-
 spec/requests/api/v3/issues_spec.rb           |  31 +-
 .../projects/artifacts_controller_spec.rb     | 117 ----
 spec/serializers/deploy_key_entity_spec.rb    |  38 ++
 spec/serializers/label_serializer_spec.rb     |  46 ++
 .../issuable/bulk_update_service_spec.rb      |  62 +-
 spec/services/issues/close_service_spec.rb    |   2 +-
 spec/services/issues/create_service_spec.rb   |  86 ++-
 .../issues/export_csv_service_spec.rb         |   6 +-
 spec/services/issues/update_service_spec.rb   |  70 +-
 .../authorized_destroy_service_spec.rb        |   6 +-
 .../assign_issues_service_spec.rb             |  10 +-
 .../merge_requests/create_service_spec.rb     |  82 ++-
 .../merge_requests/update_service_spec.rb     |  55 ++
 .../notes/slash_commands_service_spec.rb      |  31 +-
 spec/services/notification_service_spec.rb    |  80 +--
 .../projects/autocomplete_service_spec.rb     |   2 +-
 .../propagate_service_template_spec.rb        | 103 +++
 .../slash_commands/interpret_service_spec.rb  |  86 ++-
 spec/services/system_note_service_spec.rb     |  61 ++
 spec/services/todo_service_spec.rb            |  26 +-
 spec/services/users/destroy_service_spec.rb   |   4 +-
 ...issuable_slash_commands_shared_examples.rb |   5 +-
 .../import_export/export_file_helper.rb       |   2 +-
 spec/support/prometheus_helpers.rb            |   4 +
 ...issuable_create_service_shared_examples.rb |  52 --
 ..._service_slash_commands_shared_examples.rb |  18 +-
 ...issuable_update_service_shared_examples.rb |  48 --
 spec/support/test_env.rb                      |   1 +
 spec/support/time_tracking_shared_examples.rb |  13 +-
 .../projects/tags/index.html.haml_spec.rb     |  20 +
 .../elastic_commit_indexer_worker_spec.rb     |   3 +-
 .../propagate_service_template_worker_spec.rb |  29 +
 yarn.lock                                     |  31 +-
 525 files changed, 13038 insertions(+), 2700 deletions(-)
 create mode 100644 app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
 create mode 100644 app/assets/javascripts/blob/balsamiq_viewer.js
 create mode 100644 app/assets/javascripts/boards/models/assignee.js
 delete mode 100644 app/assets/javascripts/boards/models/user.js
 create mode 100644 app/assets/javascripts/deploy_keys/components/action_btn.vue
 create mode 100644 app/assets/javascripts/deploy_keys/components/app.vue
 create mode 100644 app/assets/javascripts/deploy_keys/components/key.vue
 create mode 100644 app/assets/javascripts/deploy_keys/components/keys_panel.vue
 create mode 100644 app/assets/javascripts/deploy_keys/eventhub.js
 create mode 100644 app/assets/javascripts/deploy_keys/index.js
 create mode 100644 app/assets/javascripts/deploy_keys/service/index.js
 create mode 100644 app/assets/javascripts/deploy_keys/store/index.js
 create mode 100644 app/assets/javascripts/filtered_search/services/recent_searches_service_error.js
 create mode 100644 app/assets/javascripts/issuable/auto_width_dropdown_select.js
 delete mode 100644 app/assets/javascripts/issuable/issuable_bundle.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/collapsed_state.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/comparison_pane.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/estimate_only_pane.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/help_state.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/no_tracking_pane.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/spent_only_pane.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/components/time_tracker.js
 delete mode 100644 app/assets/javascripts/issuable/time_tracking/time_tracking_bundle.js
 create mode 100644 app/assets/javascripts/lib/utils/accessor.js
 create mode 100644 app/assets/javascripts/lib/utils/ajax_cache.js
 create mode 100644 app/assets/javascripts/locale/de/app.js
 create mode 100644 app/assets/javascripts/locale/en/app.js
 create mode 100644 app/assets/javascripts/locale/es/app.js
 create mode 100644 app/assets/javascripts/locale/index.js
 create mode 100644 app/assets/javascripts/raven/index.js
 create mode 100644 app/assets/javascripts/raven/raven_config.js
 create mode 100644 app/assets/javascripts/sidebar/components/assignees/assignee_title.js
 create mode 100644 app/assets/javascripts/sidebar/components/assignees/assignees.js
 create mode 100644 app/assets/javascripts/sidebar/components/assignees/sidebar_assignees.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/collapsed_state.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/comparison_pane.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/estimate_only_pane.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/help_state.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/no_tracking_pane.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/spent_only_pane.js
 create mode 100644 app/assets/javascripts/sidebar/components/time_tracking/time_tracker.js
 create mode 100644 app/assets/javascripts/sidebar/event_hub.js
 create mode 100644 app/assets/javascripts/sidebar/services/sidebar_service.js
 create mode 100644 app/assets/javascripts/sidebar/sidebar_bundle.js
 create mode 100644 app/assets/javascripts/sidebar/sidebar_mediator.js
 create mode 100644 app/assets/javascripts/sidebar/stores/sidebar_store.js
 delete mode 100644 app/assets/javascripts/subbable_resource.js
 create mode 100644 app/assets/javascripts/vue_shared/translate.js
 create mode 100644 app/models/blob_viewer/balsamiq.rb
 create mode 100644 app/models/ci/artifact_blob.rb
 create mode 100644 app/models/issue_assignee.rb
 create mode 100644 app/serializers/README.md
 create mode 100644 app/serializers/deploy_key_entity.rb
 create mode 100644 app/serializers/deploy_key_serializer.rb
 create mode 100644 app/serializers/label_serializer.rb
 create mode 100644 app/serializers/project_entity.rb
 create mode 100644 app/services/projects/propagate_service_template.rb
 create mode 100644 app/views/layouts/oauth_error.html.haml
 delete mode 100644 app/views/notify/_reassigned_issuable_email.text.erb
 create mode 100644 app/views/projects/artifacts/file.html.haml
 create mode 100644 app/views/projects/blob/viewers/_balsamiq.html.haml
 rename app/views/{projects/compare => shared}/_ref_dropdown.html.haml (50%)
 create mode 100644 app/views/shared/errors/_graphic_422.svg
 create mode 100644 app/views/shared/issuable/_assignees.html.haml
 create mode 100644 app/views/shared/issuable/form/_issue_assignee.html.haml
 create mode 100644 app/views/shared/issuable/form/_merge_request_assignee.html.haml
 create mode 100644 app/workers/propagate_service_template_worker.rb
 create mode 100644 changelogs/unreleased/24883-build-failure-summary-page.yml
 create mode 100644 changelogs/unreleased/27614-instant-comments.yml
 create mode 100644 changelogs/unreleased/29145-oauth-422.yml
 create mode 100644 changelogs/unreleased/30007-done-todo-hover-state.yml
 create mode 100644 changelogs/unreleased/30903-vertically-align-mini-pipeline.yml
 create mode 100644 changelogs/unreleased/30973-network-graph-sorted-by-date-and-topo.yml
 create mode 100644 changelogs/unreleased/31689-request-access-spacing.yml
 create mode 100644 changelogs/unreleased/31760-add-tooltips-to-note-actions.yml
 create mode 100644 changelogs/unreleased/31810-commit-link.yml
 create mode 100644 changelogs/unreleased/add_system_note_for_editing_issuable.yml
 create mode 100644 changelogs/unreleased/balsalmiq-support.yml
 create mode 100644 changelogs/unreleased/deploy-keys-load-async.yml
 create mode 100644 changelogs/unreleased/dm-artifact-blob-viewer.yml
 create mode 100644 changelogs/unreleased/fix-admin-integrations.yml
 create mode 100644 changelogs/unreleased/implement-i18n-support.yml
 create mode 100644 changelogs/unreleased/issue-boards-no-avatar.yml
 create mode 100644 changelogs/unreleased/issue-title-description-realtime.yml
 create mode 100644 changelogs/unreleased/merge-request-poll-json-endpoint.yml
 create mode 100644 changelogs/unreleased/mrchrisw-import-shell-timeout.yml
 create mode 100644 changelogs/unreleased/prometheus-integration-test-setting-fix.yml
 create mode 100644 changelogs/unreleased/tags-sort-default.yml
 create mode 100644 changelogs/unreleased/update-issue-board-cards-design.yml
 create mode 100644 changelogs/unreleased/winh-visual-token-labels.yml
 create mode 100644 config/initializers/fast_gettext.rb
 create mode 100644 config/initializers/gettext_rails_i18n_patch.rb
 create mode 100644 config/locales/de.yml
 create mode 100644 config/locales/es.yml
 create mode 100644 db/migrate/20170320171632_create_issue_assignees_table.rb
 create mode 100644 db/migrate/20170320173259_migrate_assignees.rb
 create mode 100644 db/migrate/20170413035209_add_preferred_language_to_users.rb
 create mode 100644 db/migrate/20170503021915_add_last_edited_at_and_last_edited_by_id_to_issues.rb
 create mode 100644 db/migrate/20170503022548_add_last_edited_at_and_last_edited_by_id_to_merge_requests.rb
 create mode 100644 db/migrate/20170504102911_add_clientside_sentry_to_application_settings.rb
 create mode 100644 doc/development/build_test_package.md
 create mode 100644 lib/api/helpers/common_helpers.rb
 create mode 100644 lib/gitlab/i18n.rb
 create mode 100644 lib/tasks/gettext.rake
 create mode 100644 locale/de/gitlab.po
 create mode 100644 locale/de/gitlab.po.time_stamp
 create mode 100644 locale/en/gitlab.po
 create mode 100644 locale/en/gitlab.po.time_stamp
 create mode 100644 locale/es/gitlab.po
 create mode 100644 locale/es/gitlab.po.time_stamp
 create mode 100644 locale/gitlab.pot
 create mode 100644 spec/controllers/projects/artifacts_controller_spec.rb
 create mode 100644 spec/controllers/projects/deploy_keys_controller_spec.rb
 create mode 100644 spec/features/projects/artifacts/file_spec.rb
 create mode 100644 spec/features/projects/branches/new_branch_ref_dropdown_spec.rb
 create mode 100644 spec/features/raven_js_spec.rb
 create mode 100644 spec/javascripts/autosave_spec.js
 create mode 100644 spec/javascripts/behaviors/gl_emoji/unicode_support_map_spec.js
 create mode 100644 spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
 create mode 100644 spec/javascripts/deploy_keys/components/action_btn_spec.js
 create mode 100644 spec/javascripts/deploy_keys/components/app_spec.js
 create mode 100644 spec/javascripts/deploy_keys/components/key_spec.js
 create mode 100644 spec/javascripts/deploy_keys/components/keys_panel_spec.js
 create mode 100644 spec/javascripts/filtered_search/recent_searches_root_spec.js
 create mode 100644 spec/javascripts/filtered_search/services/recent_searches_service_error_spec.js
 create mode 100644 spec/javascripts/fixtures/deploy_keys.rb
 create mode 100644 spec/javascripts/fixtures/labels.rb
 create mode 100644 spec/javascripts/helpers/user_mock_data_helper.js
 create mode 100644 spec/javascripts/lib/utils/accessor_spec.js
 create mode 100644 spec/javascripts/lib/utils/ajax_cache_spec.js
 create mode 100644 spec/javascripts/raven/index_spec.js
 create mode 100644 spec/javascripts/raven/raven_config_spec.js
 create mode 100644 spec/javascripts/sidebar/assignee_title_spec.js
 create mode 100644 spec/javascripts/sidebar/assignees_spec.js
 create mode 100644 spec/javascripts/sidebar/mock_data.js
 create mode 100644 spec/javascripts/sidebar/sidebar_assignees_spec.js
 create mode 100644 spec/javascripts/sidebar/sidebar_bundle_spec.js
 create mode 100644 spec/javascripts/sidebar/sidebar_mediator_spec.js
 create mode 100644 spec/javascripts/sidebar/sidebar_service_spec.js
 create mode 100644 spec/javascripts/sidebar/sidebar_store_spec.js
 delete mode 100644 spec/javascripts/subbable_resource_spec.js
 create mode 100644 spec/javascripts/vue_shared/translate_spec.js
 create mode 100644 spec/lib/gitlab/i18n_spec.rb
 create mode 100644 spec/models/ci/artifact_blob_spec.rb
 delete mode 100644 spec/requests/projects/artifacts_controller_spec.rb
 create mode 100644 spec/serializers/deploy_key_entity_spec.rb
 create mode 100644 spec/serializers/label_serializer_spec.rb
 create mode 100644 spec/services/projects/propagate_service_template_spec.rb
 delete mode 100644 spec/support/services/issuable_create_service_shared_examples.rb
 create mode 100644 spec/views/projects/tags/index.html.haml_spec.rb
 create mode 100644 spec/workers/propagate_service_template_worker_spec.rb

diff --git a/.eslintignore b/.eslintignore
index c742b08c00540..1605e483e9e63 100644
--- a/.eslintignore
+++ b/.eslintignore
@@ -7,3 +7,4 @@
 /vendor/
 karma.config.js
 webpack.config.js
+/app/assets/javascripts/locale/**/*.js
diff --git a/.gitignore b/.gitignore
index 9862ee53d4b73..ccaf475c4df16 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,5 +1,7 @@
 *.log
 *.swp
+*.mo
+*.edit.po
 .DS_Store
 .bundle
 .chef
@@ -55,3 +57,4 @@ eslint-report.html
 /shared/*
 /.gitlab_workhorse_secret
 /webpack-report/
+/locale/**/LC_MESSAGES
diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index 6389aeb84648a..71b6d5d841382 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -26,6 +26,7 @@ before_script:
   - source scripts/prepare_build.sh
 
 stages:
+- build
 - prepare
 - test
 - post-test
@@ -142,6 +143,28 @@ stages:
   <<: *only-master-and-ee-or-mysql
   <<: *except-docs
 
+# Trigger a package build on omnibus-gitlab repository
+
+build-package:
+  services: []
+  variables:
+    SETUP_DB: "false"
+    USE_BUNDLE_INSTALL: "false"
+  stage: build
+  when: manual
+  script:
+    # If no branch in omnibus is specified, trigger pipeline against master
+    - if [ -z "$OMNIBUS_BRANCH" ] ; then export OMNIBUS_BRANCH=master ;fi
+    - echo "token=${BUILD_TRIGGER_TOKEN}" > version_details
+    - echo "ref=${OMNIBUS_BRANCH}" >> version_details
+    - echo "variables[ALTERNATIVE_SOURCES]=true" >> version_details
+    - echo "variables[GITLAB_VERSION]=${CI_COMMIT_SHA}" >> version_details
+    # Collect version details of all components
+    - for f in *_VERSION; do echo "variables[$f]=$(cat $f)" >> version_details; done
+    # Trigger the API and pass values collected above as parameters to it
+    - cat version_details | tr '\n' '&' | curl -X POST https://gitlab.com/api/v4/projects/20699/trigger/pipeline --data-binary @-
+    - rm version_details
+
 # Prepare and merge knapsack tests
 knapsack:
   <<: *knapsack-state
@@ -397,18 +420,6 @@ rake karma:
     paths:
     - coverage-javascript/
 
-bundler:audit:
-  stage: test
-  <<: *ruby-static-analysis
-  <<: *dedicated-runner
-  only:
-    - master@gitlab-org/gitlab-ce
-    - master@gitlab-org/gitlab-ee
-    - master@gitlab/gitlabhq
-    - master@gitlab/gitlab-ee
-  script:
-    - "bundle exec bundle-audit check --update --ignore CVE-2016-4658"
-
 .migration-paths: &migration-paths
   stage: test
   <<: *dedicated-runner
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 639f078fe1fd3..a4342bc541245 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -2,6 +2,18 @@
 documentation](doc/development/changelog.md) for instructions on adding your own
 entry.
 
+## 9.1.3 (2017-05-05)
+
+- Do not show private groups on subgroups page if user doesn't have access to.
+- Enforce project features when searching blobs and wikis.
+- Fixed branches dropdown rendering branch names as HTML.
+- Make Asciidoc & other markup go through pipeline to prevent XSS.
+- Validate URLs in markdown using URI to detect the host correctly.
+- Fix for XSS in project import view caused by Hamlit filter usage.
+- Sanitize submodule URLs before linking to them in the file tree view.
+- Refactor snippets finder & dont return internal snippets for external users.
+- Fix snippets visibility for show action - external users can not see internal snippets.
+
 ## 9.1.2 (2017-05-01)
 
 - Add index on ci_runners.contacted_at. !10876 (blackst0ne)
@@ -277,6 +289,18 @@ entry.
 - Only send chat notifications for the default branch.
 - Don't fill in the default kubernetes namespace.
 
+## 9.0.7 (2017-05-05)
+
+- Enforce project features when searching blobs and wikis.
+- Fixed branches dropdown rendering branch names as HTML.
+- Make Asciidoc & other markup go through pipeline to prevent XSS.
+- Validate URLs in markdown using URI to detect the host correctly.
+- Fix for XSS in project import view caused by Hamlit filter usage.
+- Sanitize submodule URLs before linking to them in the file tree view.
+- Refactor snippets finder & dont return internal snippets for external users.
+- Fix snippets visibility for show action - external users can not see internal snippets.
+- Do not show private groups on subgroups page if user doesn't have access to.
+
 ## 9.0.6 (2017-04-21)
 
 - Bugfix: POST /projects/:id/hooks and PUT /projects/:id/hook/:hook_id no longer ignore the the job_events param in the V4 API. !10586
@@ -621,6 +645,17 @@ entry.
 - Change development tanuki favicon colors to match logo color order.
 - API issues - support filtering by iids.
 
+## 8.17.6 (2017-05-05)
+
+- Enforce project features when searching blobs and wikis.
+- Fixed branches dropdown rendering branch names as HTML.
+- Make Asciidoc & other markup go through pipeline to prevent XSS.
+- Validate URLs in markdown using URI to detect the host correctly.
+- Fix for XSS in project import view caused by Hamlit filter usage.
+- Sanitize submodule URLs before linking to them in the file tree view.
+- Refactor snippets finder & dont return internal snippets for external users.
+- Fix snippets visibility for show action - external users can not see internal snippets.
+
 ## 8.17.5 (2017-04-05)
 
 - Don’t show source project name when user does not have access.
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index 3bedfe1061d19..bd03c6b0c94d8 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -19,11 +19,14 @@ _This notice should stay as the first item in the CONTRIBUTING.MD file._
 - [Helping others](#helping-others)
 - [I want to contribute!](#i-want-to-contribute)
 - [Workflow labels](#workflow-labels)
+  - [Type labels (~"feature proposal", ~bug, ~customer, etc.)](#type-labels-feature-proposal-bug-customer-etc)
+  - [Subject labels (~wiki, ~"container registry", ~ldap, ~api, etc.)](#subject-labels-wiki-container-registry-ldap-api-etc)
+  - [Team labels (~CI, ~Discussion, ~Edge, ~Platform, etc.)](#team-labels-ci-discussion-edge-platform-etc)
+  - [Priority labels (~Deliverable and ~Stretch)](#priority-labels-deliverable-and-stretch)
+  - [Label for community contributors (~"Accepting Merge Requests")](#label-for-community-contributors-accepting-merge-requests)
 - [Implement design & UI elements](#implement-design--ui-elements)
-- [Release retrospective and kickoff](#release-retrospective-and-kickoff)
-  - [Retrospective](#retrospective)
-  - [Kickoff](#kickoff)
 - [Issue tracker](#issue-tracker)
+  - [Issue triaging](#issue-triaging)
   - [Feature proposals](#feature-proposals)
   - [Issue tracker guidelines](#issue-tracker-guidelines)
   - [Issue weight](#issue-weight)
@@ -32,9 +35,7 @@ _This notice should stay as the first item in the CONTRIBUTING.MD file._
   - [Stewardship](#stewardship)
 - [Merge requests](#merge-requests)
   - [Merge request guidelines](#merge-request-guidelines)
-  - [Getting your merge request reviewed, approved, and merged](#getting-your-merge-request-reviewed-approved-and-merged)
   - [Contribution acceptance criteria](#contribution-acceptance-criteria)
-- [Changes for Stable Releases](#changes-for-stable-releases)
 - [Definition of done](#definition-of-done)
 - [Style guides](#style-guides)
 - [Code of conduct](#code-of-conduct)
@@ -105,34 +106,128 @@ contributing to GitLab.
 
 ## Workflow labels
 
-Labelling issues is described in the [GitLab Inc engineering workflow].
+To allow for asynchronous issue handling, we use [milestones][milestones-page]
+and [labels][labels-page]. Leads and product managers handle most of the
+scheduling into milestones. Labelling is a task for everyone.
 
-## Implement design & UI elements
+Most issues will have labels for at least one of the following:
 
-Please see the [UX Guide for GitLab].
+- Type: ~"feature proposal", ~bug, ~customer, etc.
+- Subject: ~wiki, ~"container registry", ~ldap, ~api, etc.
+- Team: ~CI, ~Discussion, ~Edge, ~Frontend, ~Platform, etc.
+- Priority: ~Deliverable, ~Stretch
+
+All labels, their meaning and priority are defined on the
+[labels page][labels-page].
+
+If you come across an issue that has none of these, and you're allowed to set
+labels, you can _always_ add the team and type, and often also the subject.
+
+[milestones-page]: https://gitlab.com/gitlab-org/gitlab-ce/milestones
+[labels-page]: https://gitlab.com/gitlab-org/gitlab-ce/labels
+
+### Type labels (~"feature proposal", ~bug, ~customer, etc.)
+
+Type labels are very important. They define what kind of issue this is. Every
+issue should have one or more.
+
+Examples of type labels are ~"feature proposal", ~bug, ~customer, ~security,
+and ~"direction".
+
+A number of type labels have a priority assigned to them, which automatically
+makes them float to the top, depending on their importance.
+
+Type labels are always lowercase, and can have any color, besides blue (which is
+already reserved for subject labels).
+
+The descriptions on the [labels page][labels-page] explain what falls under each type label.
+
+### Subject labels (~wiki, ~"container registry", ~ldap, ~api, etc.)
+
+Subject labels are labels that define what area or feature of GitLab this issue
+hits. They are not always necessary, but very convenient.
+
+If you are an expert in a particular area, it makes it easier to find issues to
+work on. You can also subscribe to those labels to receive an email each time an
+issue is labelled with a subject label corresponding to your expertise.
+
+Examples of subject labels are ~wiki, ~"container registry", ~ldap, ~api,
+~issues, ~"merge requests", ~labels, and ~"container registry".
+
+Subject labels are always all-lowercase.
+
+### Team labels (~CI, ~Discussion, ~Edge, ~Platform, etc.)
+
+Team labels specify what team is responsible for this issue.
+Assigning a team label makes sure issues get the attention of the appropriate
+people.
+
+The current team labels are ~Build, ~CI, ~Discussion, ~Documentation, ~Edge,
+~Gitaly, ~Platform, ~Prometheus, ~Release, and ~"UX".
+
+The descriptions on the [labels page][labels-page] explain what falls under the
+responsibility of each team.
+
+Within those team labels, we also have the ~backend and ~frontend labels to
+indicate if an issue needs backend work, frontend work, or both.
+
+Team labels are always capitalized so that they show up as the first label for
+any issue.
 
-## Release retrospective and kickoff
+### Priority labels (~Deliverable and ~Stretch)
 
-### Retrospective
+Priority labels help us clearly communicate expectations of the work for the
+release. There are two levels of priority labels:
 
-After each release, we have a retrospective call where we discuss what went well,
-what went wrong, and what we can improve for the next release. The
-[retrospective notes] are public and you are invited to comment on them.
-If you're interested, you can even join the
-[retrospective call][retro-kickoff-call], on the first working day after the
-22nd at 6pm CET / 9am PST.
+- ~Deliverable: Issues that are expected to be delivered in the current
+  milestone.
+- ~Stretch: Issues that are a stretch goal for delivering in the current
+  milestone. If these issues are not done in the current release, they will
+  strongly be considered for the next release.
 
-### Kickoff
+### Label for community contributors (~"Accepting Merge Requests")
 
-Before working on the next release, we have a
-kickoff call to explain what we expect to ship in the next release. The
-[kickoff notes] are public and you are invited to comment on them.
-If you're interested, you can even join the [kickoff call][retro-kickoff-call],
-on the first working day after the 7th at 6pm CET / 9am PST..
+Issues that are beneficial to our users, 'nice to haves', that we currently do
+not have the capacity for or want to give the priority to, are labeled as
+~"Accepting Merge Requests", so the community can make a contribution.
 
-[retrospective notes]: https://docs.google.com/document/d/1nEkM_7Dj4bT21GJy0Ut3By76FZqCfLBmFQNVThmW2TY/edit?usp=sharing
-[kickoff notes]: https://docs.google.com/document/d/1ElPkZ90A8ey_iOkTvUs_ByMlwKK6NAB2VOK5835wYK0/edit?usp=sharing
-[retro-kickoff-call]: https://gitlab.zoom.us/j/918821206
+Community contributors can submit merge requests for any issue they want, but
+the ~"Accepting Merge Requests" label has a special meaning. It points to
+changes that:
+
+1. We already agreed on,
+1. Are well-defined,
+1. Are likely to get accepted by a maintainer.
+
+We want to avoid a situation when a contributor picks an
+~"Accepting Merge Requests" issue and then their merge request gets closed,
+because we realize that it does not fit our vision, or we want to solve it in a
+different way.
+
+We add the ~"Accepting Merge Requests" label to:
+
+- Low priority ~bug issues (i.e. we do not add it to the bugs that we want to
+solve in the ~"Next Patch Release")
+- Small ~"feature proposal" that do not need ~UX / ~"Product work", or for which
+the ~UX / ~"Product work" is already done
+- Small ~"technical debt" issues
+
+After adding the ~"Accepting Merge Requests" label, we try to estimate the
+[weight](#issue-weight) of the issue. We use issue weight to let contributors
+know how difficult the issue is. Additionally:
+
+- We advertise [~"Accepting Merge Requests" issues with weight < 5][up-for-grabs]
+  as suitable for people that have never contributed to GitLab before on the
+  [Up For Grabs campaign](http://up-for-grabs.net)
+- We encourage people that have never contributed to any open source project to
+  look for [~"Accepting Merge Requests" issues with a weight of 1][firt-timers]
+
+[up-for-grabs]: https://gitlab.com/gitlab-org/gitlab-ce/issues?label_name=Accepting+Merge+Requests&scope=all&sort=weight_asc&state=opened
+[firt-timers]: https://gitlab.com/gitlab-org/gitlab-ce/issues?label_name%5B%5D=Accepting+Merge+Requests&scope=all&sort=upvotes_desc&state=opened&weight=1
+
+## Implement design & UI elements
+
+Please see the [UX Guide for GitLab].
 
 ## Issue tracker
 
@@ -156,6 +251,21 @@ If it happens that you know the solution to an existing bug, please first
 open the issue in order to keep track of it and then open the relevant merge
 request that potentially fixes it.
 
+### Issue triaging
+
+Our issue triage policies are [described in our handbook]. You are very welcome
+to help the GitLab team triage issues. We also organize [issue bash events] once
+every quarter.
+
+The most important thing is making sure valid issues receive feedback from the
+development team. Therefore the priority is mentioning developers that can help
+on those issues. Please select someone with relevant experience from the
+[GitLab team][team]. If there is nobody mentioned with that expertise look in
+the commit history for the affected files to find someone.
+
+[described in our handbook]: https://about.gitlab.com/handbook/engineering/issues/issue-triage-policies/
+[issue bash events]: https://gitlab.com/gitlab-org/gitlab-ce/issues/17815
+
 ### Feature proposals
 
 To create a feature proposal for CE, open an issue on the
@@ -329,13 +439,17 @@ request is as follows:
      "Description" field.
   1. If you are contributing documentation, choose `Documentation` from the
      "Choose a template" menu and fill in the template.
+  1. Mention the issue(s) your merge request solves, using the `Solves #XXX` or
+    `Closes #XXX` syntax to auto-close the issue(s) once the merge request will
+    be merged.
+1. If you're allowed to, set a relevant milestone and labels
 1. If the MR changes the UI it should include *Before* and *After* screenshots
 1. If the MR changes CSS classes please include the list of affected pages,
    `grep css-class ./app -R`
-1. Link any relevant [issues][ce-tracker] in the merge request description and
-   leave a comment on them with a link back to the MR
 1. Be prepared to answer questions and incorporate feedback even if requests
    for this arrive weeks or months after your MR submission
+  1. If a discussion has been addressed, select the "Resolve discussion" button
+    beneath it to mark it resolved.
 1. If your MR touches code that executes shell commands, reads or opens files or
    handles paths to files on disk, make sure it adheres to the
    [shell command guidelines](doc/development/shell_commands.md)
@@ -371,24 +485,6 @@ Please ensure that your merge request meets the contribution acceptance criteria
 When having your code reviewed and when reviewing merge requests please take the
 [code review guidelines](doc/development/code_review.md) into account.
 
-### Getting your merge request reviewed, approved, and merged
-
-There are a few rules to get your merge request accepted:
-
-1. Your merge request should only be **merged by a [maintainer][team]**.
-  1. If your merge request includes only backend changes [^1], it must be
-    **approved by a [backend maintainer][team]**.
-  1. If your merge request includes only frontend changes [^1], it must be
-    **approved by a [frontend maintainer][team]**.
-  1. If your merge request includes frontend and backend changes [^1], it must
-    be **approved by a [frontend and a backend maintainer][team]**.
-1. To lower the amount of merge requests maintainers need to review, you can
-  ask or assign any [reviewers][team] for a first review.
-  1. If you need some guidance (e.g. it's your first merge request), feel free
-    to ask one of the [Merge request coaches][team].
-  1. The reviewer will assign the merge request to a maintainer once the
-    reviewer is satisfied with the state of the merge request.
-
 ### Contribution acceptance criteria
 
 1. The change is as small as possible
@@ -418,8 +514,7 @@ There are a few rules to get your merge request accepted:
 1. If you need polling to support real-time features, please use
    [polling with ETag caching][polling-etag].
 1. Changes after submitting the merge request should be in separate commits
-   (no squashing). If necessary, you will be asked to squash when the review is
-   over, before merging.
+   (no squashing).
 1. It conforms to the [style guides](#style-guides) and the following:
     - If your change touches a line that does not follow the style, modify the
       entire line to follow it. This prevents linting tools from generating warnings.
@@ -430,19 +525,6 @@ There are a few rules to get your merge request accepted:
    See the instructions in that document for help if your MR fails the
    "license-finder" test with a "Dependencies that need approval" error.
 
-## Changes for Stable Releases
-
-Sometimes certain changes have to be added to an existing stable release.
-Two examples are bug fixes and performance improvements. In these cases the
-corresponding merge request should be updated to have the following:
-
-1. A milestone indicating what release the merge request should be merged into.
-1. The label "Pick into Stable"
-
-This makes it easier for release managers to keep track of what still has to be
-merged and where changes have to be merged into.
-Like all merge requests the target should be master so all bugfixes are in master.
-
 ## Definition of done
 
 If you contribute to GitLab please know that changes involve more than just
@@ -451,16 +533,16 @@ the feature you contribute through all of these steps.
 
 1. Description explaining the relevancy (see following item)
 1. Working and clean code that is commented where needed
-1. Unit and integration tests that pass on the CI server
+1. [Unit and system tests][testing] that pass on the CI server
 1. Performance/scalability implications have been considered, addressed, and tested
-1. [Documented][doc-styleguide] in the /doc directory
-1. Changelog entry added
+1. [Documented][doc-styleguide] in the `/doc` directory
+1. [Changelog entry added][changelog], if necessary
 1. Reviewed and any concerns are addressed
-1. Merged by the project lead
-1. Added to the release blog article
-1. Added to [the website](https://gitlab.com/gitlab-com/www-gitlab-com/) if relevant
+1. Merged by a project maintainer
+1. Added to the release blog article, if relevant
+1. Added to [the website](https://gitlab.com/gitlab-com/www-gitlab-com/), if relevant
 1. Community questions answered
-1. Answers to questions radiated (in docs/wiki/etc.)
+1. Answers to questions radiated (in docs/wiki/support etc.)
 
 If you add a dependency in GitLab (such as an operating system package) please
 consider updating the following and note the applicability of each in your
@@ -483,7 +565,7 @@ merge request:
     - string literal quoting style **Option A**: single quoted by default
 1.  [Rails](https://github.com/bbatsov/rails-style-guide)
 1.  [Newlines styleguide][newlines-styleguide]
-1.  [Testing](doc/development/testing.md)
+1.  [Testing][testing]
 1.  [JavaScript styleguide][js-styleguide]
 1.  [SCSS styleguide][scss-styleguide]
 1.  [Shell commands](doc/development/shell_commands.md) created by GitLab
@@ -578,6 +660,7 @@ When your code contains more than 500 changes, any major breaking changes, or an
 [license-finder-doc]: doc/development/licensing.md
 [GitLab Inc engineering workflow]: https://about.gitlab.com/handbook/engineering/workflow/#labelling-issues
 [polling-etag]: https://docs.gitlab.com/ce/development/polling.html
+[testing]: doc/development/testing.md
 
 [^1]: Please note that specs other than JavaScript specs are considered backend
       code.
diff --git a/Gemfile b/Gemfile
index 976b105ca7c10..d60419a9d24c8 100644
--- a/Gemfile
+++ b/Gemfile
@@ -266,6 +266,12 @@ gem 'sentry-raven', '~> 2.4.0'
 
 gem 'premailer-rails', '~> 1.9.0'
 
+# I18n
+gem 'ruby_parser', '~> 3.8.4', require: false
+gem 'gettext_i18n_rails', '~> 1.8.0'
+gem 'gettext_i18n_rails_js', '~> 1.2.0'
+gem 'gettext', '~> 3.2.2', require: false, group: :development
+
 # Metrics
 group :metrics do
   gem 'allocations', '~> 1.0', require: false, platform: :mri
diff --git a/Gemfile.lock b/Gemfile.lock
index 0d90374fd14b4..a18bfc103dd0c 100644
--- a/Gemfile.lock
+++ b/Gemfile.lock
@@ -222,6 +222,7 @@ GEM
     faraday_middleware-multi_json (0.0.6)
       faraday_middleware
       multi_json
+    fast_gettext (1.4.0)
     ffaker (2.4.0)
     ffi (1.9.10)
     flay (2.8.1)
@@ -275,6 +276,16 @@ GEM
     gemojione (3.0.1)
       json
     get_process_mem (0.2.0)
+    gettext (3.2.2)
+      locale (>= 2.0.5)
+      text (>= 1.3.0)
+    gettext_i18n_rails (1.8.0)
+      fast_gettext (>= 0.9.0)
+    gettext_i18n_rails_js (1.2.0)
+      gettext (>= 3.0.2)
+      gettext_i18n_rails (>= 0.7.1)
+      po_to_json (>= 1.0.0)
+      rails (>= 3.2.0)
     gherkin-ruby (0.3.2)
     gitaly (0.5.0)
       google-protobuf (~> 3.1)
@@ -450,6 +461,7 @@ GEM
     licensee (8.7.0)
       rugged (~> 0.24)
     little-plugger (1.1.4)
+    locale (2.1.2)
     logging (2.1.0)
       little-plugger (~> 1.1)
       multi_json (~> 1.10)
@@ -553,6 +565,8 @@ GEM
       ast (~> 2.2)
     path_expander (1.0.1)
     pg (0.18.4)
+    po_to_json (1.0.1)
+      json (>= 1.6.0)
     poltergeist (1.9.0)
       capybara (~> 2.1)
       cliver (~> 0.3.1)
@@ -805,6 +819,7 @@ GEM
     temple (0.7.7)
     test_after_commit (1.1.0)
       activerecord (>= 3.2)
+    text (1.3.1)
     thin (1.7.0)
       daemons (~> 1.0, >= 1.0.9)
       eventmachine (~> 1.0, >= 1.0.4)
@@ -937,6 +952,9 @@ DEPENDENCIES
   fuubar (~> 2.0.0)
   gemnasium-gitlab-service (~> 0.2)
   gemojione (~> 3.0)
+  gettext (~> 3.2.2)
+  gettext_i18n_rails (~> 1.8.0)
+  gettext_i18n_rails_js (~> 1.2.0)
   gitaly (~> 0.5.0)
   github-linguist (~> 4.7.0)
   gitlab-flowdock-git-hook (~> 1.0.1)
@@ -1030,6 +1048,7 @@ DEPENDENCIES
   rubocop-rspec (~> 1.15.0)
   ruby-fogbugz (~> 0.2.1)
   ruby-prof (~> 0.16.2)
+  ruby_parser (~> 3.8.4)
   rufus-scheduler (~> 3.1.10)
   rugged (~> 0.25.1.1)
   sanitize (~> 2.0)
diff --git a/PROCESS.md b/PROCESS.md
index fac3c22e09fc8..3b97a4e8c75fc 100644
--- a/PROCESS.md
+++ b/PROCESS.md
@@ -1,35 +1,53 @@
-# GitLab Contributing Process
+## GitLab Core Team & GitLab Inc. Contribution Process
+
+---
+
+<!-- START doctoc generated TOC please keep comment here to allow auto update -->
+<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
+**Table of Contents**  *generated with [DocToc](https://github.com/thlorenz/doctoc)*
+
+- [Purpose of describing the contributing process](#purpose-of-describing-the-contributing-process)
+- [Common actions](#common-actions)
+  - [Merge request coaching](#merge-request-coaching)
+- [Assigning issues](#assigning-issues)
+- [Be kind](#be-kind)
+- [Feature freeze on the 7th for the release on the 22nd](#feature-freeze-on-the-7th-for-the-release-on-the-22nd)
+  - [Between the 1st and the 7th](#between-the-1st-and-the-7th)
+  - [On the 7th](#on-the-7th)
+  - [After the 7th](#after-the-7th)
+- [Release retrospective and kickoff](#release-retrospective-and-kickoff)
+  - [Retrospective](#retrospective)
+  - [Kickoff](#kickoff)
+- [Copy & paste responses](#copy--paste-responses)
+  - [Improperly formatted issue](#improperly-formatted-issue)
+  - [Issue report for old version](#issue-report-for-old-version)
+  - [Support requests and configuration questions](#support-requests-and-configuration-questions)
+  - [Code format](#code-format)
+  - [Issue fixed in newer version](#issue-fixed-in-newer-version)
+  - [Improperly formatted merge request](#improperly-formatted-merge-request)
+  - [Inactivity close of an issue](#inactivity-close-of-an-issue)
+  - [Inactivity close of a merge request](#inactivity-close-of-a-merge-request)
+  - [Accepting merge requests](#accepting-merge-requests)
+  - [Only accepting merge requests with green tests](#only-accepting-merge-requests-with-green-tests)
+  - [Closing down the issue tracker on GitHub](#closing-down-the-issue-tracker-on-github)
+
+<!-- END doctoc generated TOC please keep comment here to allow auto update -->
+
+---
 
 ## Purpose of describing the contributing process
 
-Below we describe the contributing process to GitLab for two reasons. So that
-contributors know what to expect from maintainers (possible responses, friendly
-treatment, etc.). And so that maintainers know what to expect from contributors
-(use the latest version, ensure that the issue is addressed, friendly treatment,
-etc.).
+Below we describe the contributing process to GitLab for two reasons:
+
+1. Contributors know what to expect from maintainers (possible responses, friendly
+  treatment, etc.)
+1. Maintainers know what to expect from contributors (use the latest version,
+  ensure that the issue is addressed, friendly treatment, etc.).
 
 - [GitLab Inc engineers should refer to the engineering workflow document](https://about.gitlab.com/handbook/engineering/workflow/)
 
 ## Common actions
 
-### Issue triaging
-
-Our issue triage policies are [described in our handbook]. You are very welcome
-to help the GitLab team triage issues. We also organize [issue bash events] once
-every quarter.
-
-The most important thing is making sure valid issues receive feedback from the
-development team. Therefore the priority is mentioning developers that can help
-on those issues. Please select someone with relevant experience from
-[GitLab team][team]. If there is nobody mentioned with that expertise
-look in the commit history for the affected files to find someone. Avoid
-mentioning the lead developer, this is the person that is least likely to give a
-timely response. If the involvement of the lead developer is needed the other
-core team members will mention this person.
-
-[described in our handbook]: https://about.gitlab.com/handbook/engineering/issues/issue-triage-policies/
-[issue bash events]: https://gitlab.com/gitlab-org/gitlab-ce/issues/17815
-
 ### Merge request coaching
 
 Several people from the [GitLab team][team] are helping community members to get
@@ -37,12 +55,6 @@ their contributions accepted by meeting our [Definition of done][done].
 
 What you can expect from them is described at https://about.gitlab.com/jobs/merge-request-coach/.
 
-## Workflow labels
-
-Labelling issues is described in the [GitLab Inc engineering workflow].
-
-[GitLab Inc engineering workflow]: https://about.gitlab.com/handbook/engineering/workflow/#labelling-issues
-
 ## Assigning issues
 
 If an issue is complex and needs the attention of a specific person, assignment is a good option but assigning issues might discourage other people from contributing to that issue. We need all the contributions we can get so this should never be discouraged. Also, an assigned person might not have time for a few weeks, so others should feel free to takeover.
@@ -146,6 +158,29 @@ release should have the correct milestone assigned _and_ have the label
 Merge requests without a milestone and this label will
 not be merged into any stable branches.
 
+## Release retrospective and kickoff
+
+### Retrospective
+
+After each release, we have a retrospective call where we discuss what went well,
+what went wrong, and what we can improve for the next release. The
+[retrospective notes] are public and you are invited to comment on them.
+If you're interested, you can even join the
+[retrospective call][retro-kickoff-call], on the first working day after the
+22nd at 6pm CET / 9am PST.
+
+### Kickoff
+
+Before working on the next release, we have a
+kickoff call to explain what we expect to ship in the next release. The
+[kickoff notes] are public and you are invited to comment on them.
+If you're interested, you can even join the [kickoff call][retro-kickoff-call],
+on the first working day after the 7th at 6pm CET / 9am PST..
+
+[retrospective notes]: https://docs.google.com/document/d/1nEkM_7Dj4bT21GJy0Ut3By76FZqCfLBmFQNVThmW2TY/edit?usp=sharing
+[kickoff notes]: https://docs.google.com/document/d/1ElPkZ90A8ey_iOkTvUs_ByMlwKK6NAB2VOK5835wYK0/edit?usp=sharing
+[retro-kickoff-call]: https://gitlab.zoom.us/j/918821206
+
 ## Copy & paste responses
 
 ### Improperly formatted issue
diff --git a/app/assets/javascripts/autosave.js b/app/assets/javascripts/autosave.js
index 8630b18a73f96..cfab6c40b34f7 100644
--- a/app/assets/javascripts/autosave.js
+++ b/app/assets/javascripts/autosave.js
@@ -1,8 +1,11 @@
 /* eslint-disable func-names, space-before-function-paren, wrap-iife, no-param-reassign, quotes, prefer-template, no-var, one-var, no-unused-vars, one-var-declaration-per-line, no-void, consistent-return, no-empty, max-len */
+import AccessorUtilities from './lib/utils/accessor';
 
 window.Autosave = (function() {
   function Autosave(field, key) {
     this.field = field;
+    this.isLocalStorageAvailable = AccessorUtilities.isLocalStorageAccessSafe();
+
     if (key.join != null) {
       key = key.join("/");
     }
@@ -17,16 +20,12 @@ window.Autosave = (function() {
   }
 
   Autosave.prototype.restore = function() {
-    var e, text;
-    if (window.localStorage == null) {
-      return;
-    }
-    try {
-      text = window.localStorage.getItem(this.key);
-    } catch (error) {
-      e = error;
-      return;
-    }
+    var text;
+
+    if (!this.isLocalStorageAvailable) return;
+
+    text = window.localStorage.getItem(this.key);
+
     if ((text != null ? text.length : void 0) > 0) {
       this.field.val(text);
     }
@@ -35,27 +34,22 @@ window.Autosave = (function() {
 
   Autosave.prototype.save = function() {
     var text;
-    if (window.localStorage == null) {
-      return;
-    }
     text = this.field.val();
-    if ((text != null ? text.length : void 0) > 0) {
-      try {
-        return window.localStorage.setItem(this.key, text);
-      } catch (error) {}
-    } else {
-      return this.reset();
+
+    if (this.isLocalStorageAvailable && (text != null ? text.length : void 0) > 0) {
+      return window.localStorage.setItem(this.key, text);
     }
+
+    return this.reset();
   };
 
   Autosave.prototype.reset = function() {
-    if (window.localStorage == null) {
-      return;
-    }
-    try {
-      return window.localStorage.removeItem(this.key);
-    } catch (error) {}
+    if (!this.isLocalStorageAvailable) return;
+
+    return window.localStorage.removeItem(this.key);
   };
 
   return Autosave;
 })();
+
+export default window.Autosave;
diff --git a/app/assets/javascripts/behaviors/gl_emoji/unicode_support_map.js b/app/assets/javascripts/behaviors/gl_emoji/unicode_support_map.js
index aa522e20c3603..257df55e54fb4 100644
--- a/app/assets/javascripts/behaviors/gl_emoji/unicode_support_map.js
+++ b/app/assets/javascripts/behaviors/gl_emoji/unicode_support_map.js
@@ -1,3 +1,5 @@
+import AccessorUtilities from '../../lib/utils/accessor';
+
 const unicodeSupportTestMap = {
   // man, student (emojione does not have any of these yet), http://emojipedia.org/emoji-zwj-sequences/
   // occupationZwj: '\u{1F468}\u{200D}\u{1F393}',
@@ -140,16 +142,25 @@ function generateUnicodeSupportMap(testMap) {
 
 function getUnicodeSupportMap() {
   let unicodeSupportMap;
-  const userAgentFromCache = window.localStorage.getItem('gl-emoji-user-agent');
+  let userAgentFromCache;
+
+  const isLocalStorageAvailable = AccessorUtilities.isLocalStorageAccessSafe();
+
+  if (isLocalStorageAvailable) userAgentFromCache = window.localStorage.getItem('gl-emoji-user-agent');
+
   try {
     unicodeSupportMap = JSON.parse(window.localStorage.getItem('gl-emoji-unicode-support-map'));
   } catch (err) {
     // swallow
   }
+
   if (!unicodeSupportMap || userAgentFromCache !== navigator.userAgent) {
     unicodeSupportMap = generateUnicodeSupportMap(unicodeSupportTestMap);
-    window.localStorage.setItem('gl-emoji-user-agent', navigator.userAgent);
-    window.localStorage.setItem('gl-emoji-unicode-support-map', JSON.stringify(unicodeSupportMap));
+
+    if (isLocalStorageAvailable) {
+      window.localStorage.setItem('gl-emoji-user-agent', navigator.userAgent);
+      window.localStorage.setItem('gl-emoji-unicode-support-map', JSON.stringify(unicodeSupportMap));
+    }
   }
 
   return unicodeSupportMap;
diff --git a/app/assets/javascripts/behaviors/quick_submit.js b/app/assets/javascripts/behaviors/quick_submit.js
index 3d162b244135c..1f9e044808451 100644
--- a/app/assets/javascripts/behaviors/quick_submit.js
+++ b/app/assets/javascripts/behaviors/quick_submit.js
@@ -43,8 +43,8 @@ $(document).on('keydown.quick_submit', '.js-quick-submit', (e) => {
   const $submitButton = $form.find('input[type=submit], button[type=submit]');
 
   if (!$submitButton.attr('disabled')) {
+    $submitButton.trigger('click', [e]);
     $submitButton.disable();
-    $form.submit();
   }
 });
 
diff --git a/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
new file mode 100644
index 0000000000000..cdbfe36ca1cc3
--- /dev/null
+++ b/app/assets/javascripts/blob/balsamiq/balsamiq_viewer.js
@@ -0,0 +1,114 @@
+/* global Flash */
+
+import sqljs from 'sql.js';
+import { template as _template } from 'underscore';
+
+const PREVIEW_TEMPLATE = _template(`
+  <div class="panel panel-default">
+    <div class="panel-heading"><%- name %></div>
+    <div class="panel-body">
+      <img class="img-thumbnail" src="data:image/png;base64,<%- image %>"/>
+    </div>
+  </div>
+`);
+
+class BalsamiqViewer {
+  constructor(viewer) {
+    this.viewer = viewer;
+    this.endpoint = this.viewer.dataset.endpoint;
+  }
+
+  loadFile() {
+    const xhr = new XMLHttpRequest();
+
+    xhr.open('GET', this.endpoint, true);
+    xhr.responseType = 'arraybuffer';
+
+    xhr.onload = this.renderFile.bind(this);
+    xhr.onerror = BalsamiqViewer.onError;
+
+    xhr.send();
+  }
+
+  renderFile(loadEvent) {
+    const container = document.createElement('ul');
+
+    this.initDatabase(loadEvent.target.response);
+
+    const previews = this.getPreviews();
+    previews.forEach((preview) => {
+      const renderedPreview = this.renderPreview(preview);
+
+      container.appendChild(renderedPreview);
+    });
+
+    container.classList.add('list-inline');
+    container.classList.add('previews');
+
+    this.viewer.appendChild(container);
+  }
+
+  initDatabase(data) {
+    const previewBinary = new Uint8Array(data);
+
+    this.database = new sqljs.Database(previewBinary);
+  }
+
+  getPreviews() {
+    const thumbnails = this.database.exec('SELECT * FROM thumbnails');
+
+    return thumbnails[0].values.map(BalsamiqViewer.parsePreview);
+  }
+
+  getResource(resourceID) {
+    const resources = this.database.exec(`SELECT * FROM resources WHERE id = '${resourceID}'`);
+
+    return resources[0];
+  }
+
+  renderPreview(preview) {
+    const previewElement = document.createElement('li');
+
+    previewElement.classList.add('preview');
+    previewElement.innerHTML = this.renderTemplate(preview);
+
+    return previewElement;
+  }
+
+  renderTemplate(preview) {
+    const resource = this.getResource(preview.resourceID);
+    const name = BalsamiqViewer.parseTitle(resource);
+    const image = preview.image;
+
+    const template = PREVIEW_TEMPLATE({
+      name,
+      image,
+    });
+
+    return template;
+  }
+
+  static parsePreview(preview) {
+    return JSON.parse(preview[1]);
+  }
+
+  /*
+   * resource = {
+   *   columns: ['ID', 'BRANCHID', 'ATTRIBUTES', 'DATA'],
+   *   values: [['id', 'branchId', 'attributes', 'data']],
+   * }
+   *
+   * 'attributes' being a JSON string containing the `name` property.
+   */
+  static parseTitle(resource) {
+    return JSON.parse(resource.values[0][2]).name;
+  }
+
+  static onError() {
+    const flash = new Flash('Balsamiq file could not be loaded.');
+
+    return flash;
+  }
+}
+
+export default BalsamiqViewer;
diff --git a/app/assets/javascripts/blob/balsamiq_viewer.js b/app/assets/javascripts/blob/balsamiq_viewer.js
new file mode 100644
index 0000000000000..1dacf84470f3f
--- /dev/null
+++ b/app/assets/javascripts/blob/balsamiq_viewer.js
@@ -0,0 +1,6 @@
+import BalsamiqViewer from './balsamiq/balsamiq_viewer';
+
+document.addEventListener('DOMContentLoaded', () => {
+  const balsamiqViewer = new BalsamiqViewer(document.getElementById('js-balsamiq-viewer'));
+  balsamiqViewer.loadFile();
+});
diff --git a/app/assets/javascripts/blob/file_template_mediator.js b/app/assets/javascripts/blob/file_template_mediator.js
index 3062cd51ee370..a20c6ca7a21ab 100644
--- a/app/assets/javascripts/blob/file_template_mediator.js
+++ b/app/assets/javascripts/blob/file_template_mediator.js
@@ -99,7 +99,7 @@ export default class FileTemplateMediator {
     });
   }
 
-  selectTemplateType(item, el, e) {
+  selectTemplateType(item, e) {
     if (e) {
       e.preventDefault();
     }
@@ -117,6 +117,10 @@ export default class FileTemplateMediator {
     this.cacheToggleText();
   }
 
+  selectTemplateTypeOptions(options) {
+    this.selectTemplateType(options.selectedObj, options.e);
+  }
+
   selectTemplateFile(selector, query, data) {
     selector.renderLoading();
     // in case undo menu is already already there
diff --git a/app/assets/javascripts/blob/file_template_selector.js b/app/assets/javascripts/blob/file_template_selector.js
index 31dd45fac89ad..ab5b3751c4e2f 100644
--- a/app/assets/javascripts/blob/file_template_selector.js
+++ b/app/assets/javascripts/blob/file_template_selector.js
@@ -52,9 +52,17 @@ export default class FileTemplateSelector {
       .removeClass('fa-spinner fa-spin');
   }
 
-  reportSelection(query, el, e, data) {
+  reportSelection(options) {
+    const { query, e, data } = options;
     e.preventDefault();
     return this.mediator.selectTemplateFile(this, query, data);
   }
+
+  reportSelectionName(options) {
+    const opts = options;
+    opts.query = options.selectedObj.name;
+
+    this.reportSelection(opts);
+  }
 }
 
diff --git a/app/assets/javascripts/blob/target_branch_dropdown.js b/app/assets/javascripts/blob/target_branch_dropdown.js
index 216f069ef71cd..d52d69b127488 100644
--- a/app/assets/javascripts/blob/target_branch_dropdown.js
+++ b/app/assets/javascripts/blob/target_branch_dropdown.js
@@ -37,8 +37,8 @@ class TargetBranchDropDown {
         }
         return SELECT_ITEM_MSG;
       },
-      clicked(item, el, e) {
-        e.preventDefault();
+      clicked(options) {
+        options.e.preventDefault();
         self.onClick.call(self);
       },
       fieldName: self.fieldName,
diff --git a/app/assets/javascripts/blob/template_selector.js b/app/assets/javascripts/blob/template_selector.js
index d7c1c32efbd40..888883163c5b2 100644
--- a/app/assets/javascripts/blob/template_selector.js
+++ b/app/assets/javascripts/blob/template_selector.js
@@ -24,7 +24,7 @@ export default class TemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (item, el, e) => this.fetchFileTemplate(item, el, e),
+      clicked: options => this.fetchFileTemplate(options),
       text: item => item.name,
     });
   }
@@ -51,7 +51,10 @@ export default class TemplateSelector {
     return this.$dropdownContainer.removeClass('hidden');
   }
 
-  fetchFileTemplate(item, el, e) {
+  fetchFileTemplate(options) {
+    const { e } = options;
+    const item = options.selectedObj;
+
     e.preventDefault();
     return this.requestFile(item);
   }
diff --git a/app/assets/javascripts/blob/template_selectors/ci_yaml_selector.js b/app/assets/javascripts/blob/template_selectors/ci_yaml_selector.js
index 935df07677cb2..f2f81af137b4b 100644
--- a/app/assets/javascripts/blob/template_selectors/ci_yaml_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/ci_yaml_selector.js
@@ -25,7 +25,7 @@ export default class BlobCiYamlSelector extends FileTemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (query, el, e) => this.reportSelection(query.name, el, e),
+      clicked: options => this.reportSelectionName(options),
       text: item => item.name,
     });
   }
diff --git a/app/assets/javascripts/blob/template_selectors/dockerfile_selector.js b/app/assets/javascripts/blob/template_selectors/dockerfile_selector.js
index b4b4d09c315b3..3cb7b960aaa7c 100644
--- a/app/assets/javascripts/blob/template_selectors/dockerfile_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/dockerfile_selector.js
@@ -25,7 +25,7 @@ export default class DockerfileSelector extends FileTemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (query, el, e) => this.reportSelection(query.name, el, e),
+      clicked: options => this.reportSelectionName(options),
       text: item => item.name,
     });
   }
diff --git a/app/assets/javascripts/blob/template_selectors/gitignore_selector.js b/app/assets/javascripts/blob/template_selectors/gitignore_selector.js
index aefae54ae71a4..7efda8e7f50d8 100644
--- a/app/assets/javascripts/blob/template_selectors/gitignore_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/gitignore_selector.js
@@ -24,7 +24,7 @@ export default class BlobGitignoreSelector extends FileTemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (query, el, e) => this.reportSelection(query.name, el, e),
+      clicked: options => this.reportSelectionName(options),
       text: item => item.name,
     });
   }
diff --git a/app/assets/javascripts/blob/template_selectors/license_selector.js b/app/assets/javascripts/blob/template_selectors/license_selector.js
index c8abd689ab4ef..1d757332f6c9f 100644
--- a/app/assets/javascripts/blob/template_selectors/license_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/license_selector.js
@@ -24,13 +24,22 @@ export default class BlobLicenseSelector extends FileTemplateSelector {
       search: {
         fields: ['name'],
       },
-      clicked: (query, el, e) => {
+      clicked: (options) => {
+        const { e } = options;
+        const el = options.$el;
+        const query = options.selectedObj;
+
         const data = {
           project: this.$dropdown.data('project'),
           fullname: this.$dropdown.data('fullname'),
         };
 
-        this.reportSelection(query.id, el, e, data);
+        this.reportSelection({
+          query: query.id,
+          el,
+          e,
+          data,
+        });
       },
       text: item => item.name,
     });
diff --git a/app/assets/javascripts/blob/template_selectors/type_selector.js b/app/assets/javascripts/blob/template_selectors/type_selector.js
index 56f23ef05687a..a09381014a754 100644
--- a/app/assets/javascripts/blob/template_selectors/type_selector.js
+++ b/app/assets/javascripts/blob/template_selectors/type_selector.js
@@ -17,7 +17,7 @@ export default class FileTemplateTypeSelector extends FileTemplateSelector {
       filterable: false,
       selectable: true,
       toggleLabel: item => item.name,
-      clicked: (item, el, e) => this.mediator.selectTemplateType(item, el, e),
+      clicked: options => this.mediator.selectTemplateTypeOptions(options),
       text: item => item.name,
     });
   }
diff --git a/app/assets/javascripts/boards/boards_bundle.js b/app/assets/javascripts/boards/boards_bundle.js
index 7a90da2fd9e1e..9650f95fda2f6 100644
--- a/app/assets/javascripts/boards/boards_bundle.js
+++ b/app/assets/javascripts/boards/boards_bundle.js
@@ -13,7 +13,7 @@ require('./models/issue');
 require('./models/label');
 require('./models/list');
 require('./models/milestone');
-require('./models/user');
+require('./models/assignee');
 require('./stores/boards_store');
 require('./stores/modal_store');
 require('./services/board_service');
@@ -65,6 +65,7 @@ $(() => {
       bulkUpdatePath: $boardApp.dataset.bulkUpdatePath,
       detailIssue: Store.detail,
       milestoneTitle: $boardApp.dataset.boardMilestoneTitle,
+      defaultAvatar: $boardApp.dataset.defaultAvatar,
     },
     computed: {
       detailIssueVisible () {
@@ -98,7 +99,7 @@ $(() => {
       gl.boardService.all()
         .then((resp) => {
           resp.json().forEach((board) => {
-            const list = Store.addList(board);
+            const list = Store.addList(board, this.defaultAvatar);
 
             if (list.type === 'closed') {
               list.position = Infinity;
diff --git a/app/assets/javascripts/boards/components/board_new_issue.js b/app/assets/javascripts/boards/components/board_new_issue.js
index b5a8d947a9ce7..9ed262fefd3c8 100644
--- a/app/assets/javascripts/boards/components/board_new_issue.js
+++ b/app/assets/javascripts/boards/components/board_new_issue.js
@@ -26,6 +26,7 @@ export default {
         title: this.title,
         labels,
         subscribed: true,
+        assignees: [],
       });
 
       if (Store.state.currentBoard) {
diff --git a/app/assets/javascripts/boards/components/board_sidebar.js b/app/assets/javascripts/boards/components/board_sidebar.js
index f0066d4ec5d72..317cef9f22750 100644
--- a/app/assets/javascripts/boards/components/board_sidebar.js
+++ b/app/assets/javascripts/boards/components/board_sidebar.js
@@ -3,8 +3,13 @@
 /* global MilestoneSelect */
 /* global LabelsSelect */
 /* global Sidebar */
+/* global Flash */
 
 import Vue from 'vue';
+import eventHub from '../../sidebar/event_hub';
+
+import AssigneeTitle from '../../sidebar/components/assignees/assignee_title';
+import Assignees from '../../sidebar/components/assignees/assignees';
 
 require('./sidebar/remove_issue');
 
@@ -22,6 +27,7 @@ gl.issueBoards.BoardSidebar = Vue.extend({
       detail: Store.detail,
       issue: {},
       list: {},
+      loadingAssignees: false,
     };
   },
   computed: {
@@ -43,6 +49,10 @@ gl.issueBoards.BoardSidebar = Vue.extend({
 
         this.issue = this.detail.issue;
         this.list = this.detail.list;
+
+        this.$nextTick(() => {
+          this.endpoint = this.$refs.assigneeDropdown.dataset.issueUpdate;
+        });
       },
       deep: true
     },
@@ -53,12 +63,57 @@ gl.issueBoards.BoardSidebar = Vue.extend({
           $('.right-sidebar').getNiceScroll().resize();
         });
       }
-    }
+
+      this.issue = this.detail.issue;
+      this.list = this.detail.list;
+    },
+    deep: true
   },
   methods: {
     closeSidebar () {
       this.detail.issue = {};
-    }
+    },
+    assignSelf () {
+      // Notify gl dropdown that we are now assigning to current user
+      this.$refs.assigneeBlock.dispatchEvent(new Event('assignYourself'));
+
+      this.addAssignee(this.currentUser);
+      this.saveAssignees();
+    },
+    removeAssignee (a) {
+      gl.issueBoards.BoardsStore.detail.issue.removeAssignee(a);
+    },
+    addAssignee (a) {
+      gl.issueBoards.BoardsStore.detail.issue.addAssignee(a);
+    },
+    removeAllAssignees () {
+      gl.issueBoards.BoardsStore.detail.issue.removeAllAssignees();
+    },
+    saveAssignees () {
+      this.loadingAssignees = true;
+
+      gl.issueBoards.BoardsStore.detail.issue.update(this.endpoint)
+        .then(() => {
+          this.loadingAssignees = false;
+        })
+        .catch(() => {
+          this.loadingAssignees = false;
+          return new Flash('An error occurred while saving assignees');
+        });
+    },
+  },
+  created () {
+    // Get events from glDropdown
+    eventHub.$on('sidebar.removeAssignee', this.removeAssignee);
+    eventHub.$on('sidebar.addAssignee', this.addAssignee);
+    eventHub.$on('sidebar.removeAllAssignees', this.removeAllAssignees);
+    eventHub.$on('sidebar.saveAssignees', this.saveAssignees);
+  },
+  beforeDestroy() {
+    eventHub.$off('sidebar.removeAssignee', this.removeAssignee);
+    eventHub.$off('sidebar.addAssignee', this.addAssignee);
+    eventHub.$off('sidebar.removeAllAssignees', this.removeAllAssignees);
+    eventHub.$off('sidebar.saveAssignees', this.saveAssignees);
   },
   mounted () {
     new IssuableContext(this.currentUser);
@@ -70,5 +125,7 @@ gl.issueBoards.BoardSidebar = Vue.extend({
   },
   components: {
     removeBtn: gl.issueBoards.RemoveIssueBtn,
+    'assignee-title': AssigneeTitle,
+    assignees: Assignees,
   },
 });
diff --git a/app/assets/javascripts/boards/components/issue_card_inner.js b/app/assets/javascripts/boards/components/issue_card_inner.js
index fc154ee7b8b03..710207db0c741 100644
--- a/app/assets/javascripts/boards/components/issue_card_inner.js
+++ b/app/assets/javascripts/boards/components/issue_card_inner.js
@@ -31,18 +31,36 @@ gl.issueBoards.IssueCardInner = Vue.extend({
       default: false,
     },
   },
+  data() {
+    return {
+      limitBeforeCounter: 3,
+      maxRender: 4,
+      maxCounter: 99,
+    };
+  },
   computed: {
-    cardUrl() {
-      return `${this.issueLinkBase}/${this.issue.id}`;
+    numberOverLimit() {
+      return this.issue.assignees.length - this.limitBeforeCounter;
     },
-    assigneeUrl() {
-      return `${this.rootPath}${this.issue.assignee.username}`;
+    assigneeCounterTooltip() {
+      return `${this.assigneeCounterLabel} more`;
+    },
+    assigneeCounterLabel() {
+      if (this.numberOverLimit > this.maxCounter) {
+        return `${this.maxCounter}+`;
+      }
+
+      return `+${this.numberOverLimit}`;
     },
-    assigneeUrlTitle() {
-      return `Assigned to ${this.issue.assignee.name}`;
+    shouldRenderCounter() {
+      if (this.issue.assignees.length <= this.maxRender) {
+        return false;
+      }
+
+      return this.issue.assignees.length > this.numberOverLimit;
     },
-    avatarUrlTitle() {
-      return `Avatar for ${this.issue.assignee.name}`;
+    cardUrl() {
+      return `${this.issueLinkBase}/${this.issue.id}`;
     },
     issueId() {
       return `#${this.issue.id}`;
@@ -52,6 +70,28 @@ gl.issueBoards.IssueCardInner = Vue.extend({
     },
   },
   methods: {
+    isIndexLessThanlimit(index) {
+      return index < this.limitBeforeCounter;
+    },
+    shouldRenderAssignee(index) {
+      // Eg. maxRender is 4,
+      // Render up to all 4 assignees if there are only 4 assigness
+      // Otherwise render up to the limitBeforeCounter
+      if (this.issue.assignees.length <= this.maxRender) {
+        return index < this.maxRender;
+      }
+
+      return index < this.limitBeforeCounter;
+    },
+    assigneeUrl(assignee) {
+      return `${this.rootPath}${assignee.username}`;
+    },
+    assigneeUrlTitle(assignee) {
+      return `Assigned to ${assignee.name}`;
+    },
+    avatarUrlTitle(assignee) {
+      return `Avatar for ${assignee.name}`;
+    },
     showLabel(label) {
       if (!this.list) return true;
 
@@ -105,25 +145,39 @@ gl.issueBoards.IssueCardInner = Vue.extend({
             {{ issueId }}
           </span>
         </h4>
-        <a
-          class="card-assignee has-tooltip js-no-trigger"
-          :href="assigneeUrl"
-          :title="assigneeUrlTitle"
-          v-if="issue.assignee"
-          data-container="body"
-        >
-          <img
-            class="avatar avatar-inline s20 js-no-trigger"
-            :src="issue.assignee.avatar"
-            width="20"
-            height="20"
-            :alt="avatarUrlTitle"
-          />
-        </a>
+        <div class="card-assignee">
+          <a
+            class="has-tooltip js-no-trigger"
+            :href="assigneeUrl(assignee)"
+            :title="assigneeUrlTitle(assignee)"
+            v-for="(assignee, index) in issue.assignees"
+            v-if="shouldRenderAssignee(index)"
+            data-container="body"
+            data-placement="bottom"
+          >
+            <img
+              class="avatar avatar-inline s20"
+              :src="assignee.avatar"
+              width="20"
+              height="20"
+              :alt="avatarUrlTitle(assignee)"
+            />
+          </a>
+          <span
+            class="avatar-counter has-tooltip"
+            :title="assigneeCounterTooltip"
+            v-if="shouldRenderCounter"
+          >
+           {{ assigneeCounterLabel }}
+          </span>
+        </div>
       </div>
-      <div class="card-footer" v-if="showLabelFooter">
+      <div
+        class="card-footer"
+        v-if="showLabelFooter"
+      >
         <button
-          class="label color-label has-tooltip js-no-trigger"
+          class="label color-label has-tooltip"
           v-for="label in issue.labels"
           type="button"
           v-if="showLabel(label)"
diff --git a/app/assets/javascripts/boards/components/new_list_dropdown.js b/app/assets/javascripts/boards/components/new_list_dropdown.js
index 7e3bb79af1d4a..f29b6caa1acec 100644
--- a/app/assets/javascripts/boards/components/new_list_dropdown.js
+++ b/app/assets/javascripts/boards/components/new_list_dropdown.js
@@ -52,7 +52,9 @@ gl.issueBoards.newListDropdownInit = () => {
       filterable: true,
       selectable: true,
       multiSelect: true,
-      clicked (label, $el, e) {
+      clicked (options) {
+        const { e } = options;
+        const label = options.selectedObj;
         e.preventDefault();
 
         if (!Store.findList('title', label.title)) {
diff --git a/app/assets/javascripts/boards/models/assignee.js b/app/assets/javascripts/boards/models/assignee.js
new file mode 100644
index 0000000000000..05dd449e4fd8d
--- /dev/null
+++ b/app/assets/javascripts/boards/models/assignee.js
@@ -0,0 +1,12 @@
+/* eslint-disable no-unused-vars */
+
+class ListAssignee {
+  constructor(user, defaultAvatar) {
+    this.id = user.id;
+    this.name = user.name;
+    this.username = user.username;
+    this.avatar = user.avatar_url || defaultAvatar;
+  }
+}
+
+window.ListAssignee = ListAssignee;
diff --git a/app/assets/javascripts/boards/models/issue.js b/app/assets/javascripts/boards/models/issue.js
index 40ff31d0160bc..c2f32bed9f6b8 100644
--- a/app/assets/javascripts/boards/models/issue.js
+++ b/app/assets/javascripts/boards/models/issue.js
@@ -1,12 +1,12 @@
 /* eslint-disable no-unused-vars, space-before-function-paren, arrow-body-style, arrow-parens, comma-dangle, max-len */
 /* global ListLabel */
 /* global ListMilestone */
-/* global ListUser */
+/* global ListAssignee */
 
 import Vue from 'vue';
 
 class ListIssue {
-  constructor (obj) {
+  constructor (obj, defaultAvatar) {
     this.globalId = obj.id;
     this.id = obj.iid;
     this.title = obj.title;
@@ -14,15 +14,11 @@ class ListIssue {
     this.dueDate = obj.due_date;
     this.subscribed = obj.subscribed;
     this.labels = [];
+    this.assignees = [];
     this.selected = false;
-    this.assignee = false;
     this.position = obj.relative_position || Infinity;
     this.milestone_id = obj.milestone_id;
 
-    if (obj.assignee) {
-      this.assignee = new ListUser(obj.assignee);
-    }
-
     if (obj.milestone) {
       this.milestone = new ListMilestone(obj.milestone);
     }
@@ -30,6 +26,8 @@ class ListIssue {
     obj.labels.forEach((label) => {
       this.labels.push(new ListLabel(label));
     });
+
+    this.assignees = obj.assignees.map(a => new ListAssignee(a, defaultAvatar));
   }
 
   addLabel (label) {
@@ -52,6 +50,26 @@ class ListIssue {
     labels.forEach(this.removeLabel.bind(this));
   }
 
+  addAssignee (assignee) {
+    if (!this.findAssignee(assignee)) {
+      this.assignees.push(new ListAssignee(assignee));
+    }
+  }
+
+  findAssignee (findAssignee) {
+    return this.assignees.filter(assignee => assignee.id === findAssignee.id)[0];
+  }
+
+  removeAssignee (removeAssignee) {
+    if (removeAssignee) {
+      this.assignees = this.assignees.filter(assignee => assignee.id !== removeAssignee.id);
+    }
+  }
+
+  removeAllAssignees () {
+    this.assignees = [];
+  }
+
   getLists () {
     return gl.issueBoards.BoardsStore.state.lists.filter(list => list.findIssue(this.id));
   }
@@ -61,7 +79,7 @@ class ListIssue {
       issue: {
         milestone_id: this.milestone ? this.milestone.id : null,
         due_date: this.dueDate,
-        assignee_id: this.assignee ? this.assignee.id : null,
+        assignee_ids: this.assignees.length > 0 ? this.assignees.map((u) => u.id) : [0],
         label_ids: this.labels.map((label) => label.id)
       }
     };
diff --git a/app/assets/javascripts/boards/models/list.js b/app/assets/javascripts/boards/models/list.js
index cac5f8291e0f1..64fc4f5e25233 100644
--- a/app/assets/javascripts/boards/models/list.js
+++ b/app/assets/javascripts/boards/models/list.js
@@ -6,7 +6,7 @@ import queryData from '../utils/query_data';
 const PER_PAGE = 20;
 
 class List {
-  constructor (obj) {
+  constructor (obj, defaultAvatar) {
     this.id = obj.id;
     this._uid = this.guid();
     this.position = obj.position;
@@ -18,6 +18,7 @@ class List {
     this.loadingMore = false;
     this.issues = [];
     this.issuesSize = 0;
+    this.defaultAvatar = defaultAvatar;
 
     if (obj.label) {
       this.label = new ListLabel(obj.label);
@@ -107,7 +108,7 @@ class List {
 
   createIssues (data) {
     data.forEach((issueObj) => {
-      this.addIssue(new ListIssue(issueObj));
+      this.addIssue(new ListIssue(issueObj, this.defaultAvatar));
     });
   }
 
diff --git a/app/assets/javascripts/boards/models/user.js b/app/assets/javascripts/boards/models/user.js
deleted file mode 100644
index 8e9de4d4cbb69..0000000000000
--- a/app/assets/javascripts/boards/models/user.js
+++ /dev/null
@@ -1,12 +0,0 @@
-/* eslint-disable no-unused-vars */
-
-class ListUser {
-  constructor(user) {
-    this.id = user.id;
-    this.name = user.name;
-    this.username = user.username;
-    this.avatar = user.avatar_url;
-  }
-}
-
-window.ListUser = ListUser;
diff --git a/app/assets/javascripts/boards/stores/boards_store.js b/app/assets/javascripts/boards/stores/boards_store.js
index 5a449bd911722..afa0c4f3245b3 100644
--- a/app/assets/javascripts/boards/stores/boards_store.js
+++ b/app/assets/javascripts/boards/stores/boards_store.js
@@ -28,8 +28,8 @@ gl.issueBoards.BoardsStore = {
     this.state.currentPage = '';
     this.state.reload = false;
   },
-  addList (listObj) {
-    const list = new List(listObj);
+  addList (listObj, defaultAvatar) {
+    const list = new List(listObj, defaultAvatar);
     this.state.lists.push(list);
 
     return list;
diff --git a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
index abe48572347b0..8d3d34f836f5e 100644
--- a/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/limit_warning_component.js
@@ -9,9 +9,9 @@ export default {
     <span v-if="count === 50" class="events-info pull-right">
       <i class="fa fa-warning has-tooltip"
           aria-hidden="true"
-          title="Limited to showing 50 events at most"
+          :title="n__('Limited to showing %d event at most', 'Limited to showing %d events at most', 50)"
           data-placement="top"></i>
-      Showing 50 events
+      {{ n__('Showing %d event', 'Showing %d events', 50) }}
     </span>
   `,
 };
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
index 80bd2df6f42ea..0d9ad197abf18 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_code_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageCodeComponent = Vue.extend({
             <a :href="mergeRequest.url" class="issue-link">!{{ mergeRequest.iid }}</a>
             &middot;
             <span>
-              Opened
+              {{ __('OpenedNDaysAgo|Opened') }}
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              by
+              {{ __('ByAuthor|by') }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
           </div>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
index 20a43798fbedc..ad285874643fd 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_issue_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageIssueComponent = Vue.extend({
             <a :href="issue.url" class="issue-link">#{{ issue.iid }}</a>
             &middot;
             <span>
-              Opened
+              {{ __('OpenedNDaysAgo|Opened') }}
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-              by
+              {{ __('ByAuthor|by') }}
               <a :href="issue.author.webUrl" class="issue-author-link">
                 {{ issue.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
index f33cac3da8248..222084deee94c 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_plan_component.js
@@ -31,10 +31,10 @@ global.cycleAnalytics.StagePlanComponent = Vue.extend({
               </a>
             </h5>
             <span>
-              First
+              {{ __('FirstPushedBy|First') }}
               <span class="commit-icon">${iconCommit}</span>
               <a :href="commit.commitUrl" class="commit-hash-link monospace">{{ commit.shortSha }}</a>
-              pushed by
+              {{ __('FirstPushedBy|pushed by') }}
               <a :href="commit.author.webUrl" class="commit-author-link">
                 {{ commit.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
index 657f538537433..a14ebc3ece942 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_production_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageProductionComponent = Vue.extend({
             <a :href="issue.url" class="issue-link">#{{ issue.iid }}</a>
             &middot;
             <span>
-              Opened
+              {{ __('OpenedNDaysAgo|Opened') }}
               <a :href="issue.url" class="issue-date">{{ issue.createdAt }}</a>
             </span>
             <span>
-            by
+            {{ __('ByAuthor|by') }}
             <a :href="issue.author.webUrl" class="issue-author-link">
               {{ issue.author.name }}
             </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
index 8a801300647aa..1a5bf9bc0b511 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_review_component.js
@@ -28,11 +28,11 @@ global.cycleAnalytics.StageReviewComponent = Vue.extend({
             <a :href="mergeRequest.url" class="issue-link">!{{ mergeRequest.iid }}</a>
             &middot;
             <span>
-              Opened
+              {{ __('OpenedNDaysAgo|Opened') }}
               <a :href="mergeRequest.url" class="issue-date">{{ mergeRequest.createdAt }}</a>
             </span>
             <span>
-              by
+              {{ __('ByAuthor|by') }}
               <a :href="mergeRequest.author.webUrl" class="issue-author-link">{{ mergeRequest.author.name }}</a>
             </span>
             <template v-if="mergeRequest.state === 'closed'">
diff --git a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
index 4a28637958848..b1e9362434f1f 100644
--- a/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/stage_staging_component.js
@@ -32,7 +32,7 @@ global.cycleAnalytics.StageStagingComponent = Vue.extend({
             </h5>
             <span>
               <a :href="build.url" class="build-date">{{ build.date }}</a>
-              by
+              {{ __('ByAuthor|by') }}
               <a :href="build.author.webUrl" class="issue-author-link">
                 {{ build.author.name }}
               </a>
diff --git a/app/assets/javascripts/cycle_analytics/components/total_time_component.js b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
index 77edcb7627323..d5e6167b2a82f 100644
--- a/app/assets/javascripts/cycle_analytics/components/total_time_component.js
+++ b/app/assets/javascripts/cycle_analytics/components/total_time_component.js
@@ -12,10 +12,10 @@ global.cycleAnalytics.TotalTimeComponent = Vue.extend({
   template: `
     <span class="total-time">
       <template v-if="Object.keys(time).length">
-        <template v-if="time.days">{{ time.days }} <span>{{ time.days === 1 ? 'day' : 'days' }}</span></template>
-        <template v-if="time.hours">{{ time.hours }} <span>hr</span></template>
-        <template v-if="time.mins && !time.days">{{ time.mins }} <span>mins</span></template>
-        <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>s</span></template>
+        <template v-if="time.days">{{ time.days }} <span>{{ n__('day', 'days', time.days) }}</span></template>
+        <template v-if="time.hours">{{ time.hours }} <span>{{ n__('Time|hr', 'Time|hrs', time.hours) }}</span></template>
+        <template v-if="time.mins && !time.days">{{ time.mins }} <span>{{ n__('Time|min', 'Time|mins', time.mins) }}</span></template>
+        <template v-if="time.seconds && Object.keys(time).length === 1 || time.seconds === 0">{{ time.seconds }} <span>{{ s__('Time|s') }}</span></template>
       </template>
       <template v-else>
         --
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_bundle.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_bundle.js
index 48cab437e02e6..c8e53cb554eb2 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_bundle.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_bundle.js
@@ -2,6 +2,7 @@
 
 import Vue from 'vue';
 import Cookies from 'js-cookie';
+import Translate from '../vue_shared/translate';
 import LimitWarningComponent from './components/limit_warning_component';
 
 require('./components/stage_code_component');
@@ -16,6 +17,8 @@ require('./cycle_analytics_service');
 require('./cycle_analytics_store');
 require('./default_event_objects');
 
+Vue.use(Translate);
+
 $(() => {
   const OVERVIEW_DIALOG_COOKIE = 'cycle_analytics_help_dismissed';
   const cycleAnalyticsEl = document.querySelector('#cycle-analytics');
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
index 681d6eef56577..6504d7db2f2ae 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_service.js
@@ -30,7 +30,7 @@ class CycleAnalyticsService {
       startDate,
     } = options;
 
-    return $.get(`${this.requestPath}/events/${stage.title.toLowerCase()}.json`, {
+    return $.get(`${this.requestPath}/events/${stage.name}.json`, {
       cycle_analytics: {
         start_date: startDate,
       },
diff --git a/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js b/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
index 6536a8fd7fa28..50bd394e90e0a 100644
--- a/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
+++ b/app/assets/javascripts/cycle_analytics/cycle_analytics_store.js
@@ -1,4 +1,5 @@
 /* eslint-disable no-param-reassign */
+import { __ } from '../locale';
 
 require('../lib/utils/text_utility');
 const DEFAULT_EVENT_OBJECTS = require('./default_event_objects');
@@ -7,13 +8,13 @@ const global = window.gl || (window.gl = {});
 global.cycleAnalytics = global.cycleAnalytics || {};
 
 const EMPTY_STAGE_TEXTS = {
-  issue: 'The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.',
-  plan: 'The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.',
-  code: 'The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.',
-  test: 'The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.',
-  review: 'The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.',
-  staging: 'The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.',
-  production: 'The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.',
+  issue: __('The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.'),
+  plan: __('The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.'),
+  code: __('The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.'),
+  test: __('The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.'),
+  review: __('The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.'),
+  staging: __('The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.'),
+  production: __('The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.'),
 };
 
 global.cycleAnalytics.CycleAnalyticsStore = {
@@ -38,7 +39,7 @@ global.cycleAnalytics.CycleAnalyticsStore = {
     });
 
     newData.stages.forEach((item) => {
-      const stageSlug = gl.text.dasherize(item.title.toLowerCase());
+      const stageSlug = gl.text.dasherize(item.name.toLowerCase());
       item.active = false;
       item.isUserAllowed = data.permissions[stageSlug];
       item.emptyStageText = EMPTY_STAGE_TEXTS[stageSlug];
diff --git a/app/assets/javascripts/deploy_keys/components/action_btn.vue b/app/assets/javascripts/deploy_keys/components/action_btn.vue
new file mode 100644
index 0000000000000..3ff3a9d977e77
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/components/action_btn.vue
@@ -0,0 +1,54 @@
+<script>
+  import eventHub from '../eventhub';
+
+  export default {
+    data() {
+      return {
+        isLoading: false,
+      };
+    },
+    props: {
+      deployKey: {
+        type: Object,
+        required: true,
+      },
+      type: {
+        type: String,
+        required: true,
+      },
+      btnCssClass: {
+        type: String,
+        required: false,
+        default: 'btn-default',
+      },
+    },
+    methods: {
+      doAction() {
+        this.isLoading = true;
+
+        eventHub.$emit(`${this.type}.key`, this.deployKey);
+      },
+    },
+    computed: {
+      text() {
+        return `${this.type.charAt(0).toUpperCase()}${this.type.slice(1)}`;
+      },
+    },
+  };
+</script>
+
+<template>
+  <button
+    class="btn btn-sm prepend-left-10"
+    :class="[{ disabled: isLoading }, btnCssClass]"
+    :disabled="isLoading"
+    @click="doAction">
+    {{ text }}
+    <i
+      v-if="isLoading"
+      class="fa fa-spinner fa-spin"
+      aria-hidden="true"
+      aria-label="Loading">
+    </i>
+  </button>
+</template>
diff --git a/app/assets/javascripts/deploy_keys/components/app.vue b/app/assets/javascripts/deploy_keys/components/app.vue
new file mode 100644
index 0000000000000..7315a9e11cb02
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/components/app.vue
@@ -0,0 +1,102 @@
+<script>
+  /* global Flash */
+  import eventHub from '../eventhub';
+  import DeployKeysService from '../service';
+  import DeployKeysStore from '../store';
+  import keysPanel from './keys_panel.vue';
+
+  export default {
+    data() {
+      return {
+        isLoading: false,
+        store: new DeployKeysStore(),
+      };
+    },
+    props: {
+      endpoint: {
+        type: String,
+        required: true,
+      },
+    },
+    computed: {
+      hasKeys() {
+        return Object.keys(this.keys).length;
+      },
+      keys() {
+        return this.store.keys;
+      },
+    },
+    components: {
+      keysPanel,
+    },
+    methods: {
+      fetchKeys() {
+        this.isLoading = true;
+
+        this.service.getKeys()
+          .then((data) => {
+            this.isLoading = false;
+            this.store.keys = data;
+          })
+          .catch(() => new Flash('Error getting deploy keys'));
+      },
+      enableKey(deployKey) {
+        this.service.enableKey(deployKey.id)
+          .then(() => this.fetchKeys())
+          .catch(() => new Flash('Error enabling deploy key'));
+      },
+      disableKey(deployKey) {
+        // eslint-disable-next-line no-alert
+        if (confirm('You are going to remove this deploy key. Are you sure?')) {
+          this.service.disableKey(deployKey.id)
+            .then(() => this.fetchKeys())
+            .catch(() => new Flash('Error removing deploy key'));
+        }
+      },
+    },
+    created() {
+      this.service = new DeployKeysService(this.endpoint);
+
+      eventHub.$on('enable.key', this.enableKey);
+      eventHub.$on('remove.key', this.disableKey);
+      eventHub.$on('disable.key', this.disableKey);
+    },
+    mounted() {
+      this.fetchKeys();
+    },
+    beforeDestroy() {
+      eventHub.$off('enable.key', this.enableKey);
+      eventHub.$off('remove.key', this.disableKey);
+      eventHub.$off('disable.key', this.disableKey);
+    },
+  };
+</script>
+
+<template>
+  <div class="col-lg-9 col-lg-offset-3 append-bottom-default deploy-keys">
+    <div
+      class="text-center"
+      v-if="isLoading && !hasKeys">
+      <i
+        class="fa fa-spinner fa-spin fa-2x"
+        aria-hidden="true"
+        aria-label="Loading deploy keys">
+      </i>
+    </div>
+    <div v-else-if="hasKeys">
+      <keys-panel
+        title="Enabled deploy keys for this project"
+        :keys="keys.enabled_keys"
+        :store="store" />
+      <keys-panel
+        title="Deploy keys from projects you have access to"
+        :keys="keys.available_project_keys"
+        :store="store" />
+      <keys-panel
+        v-if="keys.public_keys.length"
+        title="Public deploy keys available to any project"
+        :keys="keys.public_keys"
+        :store="store" />
+    </div>
+  </div>
+</template>
diff --git a/app/assets/javascripts/deploy_keys/components/key.vue b/app/assets/javascripts/deploy_keys/components/key.vue
new file mode 100644
index 0000000000000..0a06a481b9686
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/components/key.vue
@@ -0,0 +1,80 @@
+<script>
+  import actionBtn from './action_btn.vue';
+
+  export default {
+    props: {
+      deployKey: {
+        type: Object,
+        required: true,
+      },
+      store: {
+        type: Object,
+        required: true,
+      },
+    },
+    components: {
+      actionBtn,
+    },
+    computed: {
+      timeagoDate() {
+        return gl.utils.getTimeago().format(this.deployKey.created_at);
+      },
+    },
+    methods: {
+      isEnabled(id) {
+        return this.store.findEnabledKey(id) !== undefined;
+      },
+    },
+  };
+</script>
+
+<template>
+  <div>
+    <div class="pull-left append-right-10 hidden-xs">
+      <i
+        aria-hidden="true"
+        class="fa fa-key key-icon">
+      </i>
+    </div>
+    <div class="deploy-key-content key-list-item-info">
+      <strong class="title">
+        {{ deployKey.title }}
+      </strong>
+      <div class="description">
+        {{ deployKey.fingerprint }}
+      </div>
+      <div
+        v-if="deployKey.can_push"
+        class="write-access-allowed">
+        Write access allowed
+      </div>
+    </div>
+    <div class="deploy-key-content prepend-left-default deploy-key-projects">
+      <a
+        v-for="project in deployKey.projects"
+        class="label deploy-project-label"
+        :href="project.full_path">
+        {{ project.full_name }}
+      </a>
+    </div>
+    <div class="deploy-key-content">
+      <span class="key-created-at">
+        created {{ timeagoDate }}
+      </span>
+      <action-btn
+        v-if="!isEnabled(deployKey.id)"
+        :deploy-key="deployKey"
+        type="enable"/>
+      <action-btn
+        v-else-if="deployKey.destroyed_when_orphaned && deployKey.almost_orphaned"
+        :deploy-key="deployKey"
+        btn-css-class="btn-warning"
+        type="remove" />
+      <action-btn
+        v-else
+        :deploy-key="deployKey"
+        btn-css-class="btn-warning"
+        type="disable" />
+    </div>
+  </div>
+</template>
diff --git a/app/assets/javascripts/deploy_keys/components/keys_panel.vue b/app/assets/javascripts/deploy_keys/components/keys_panel.vue
new file mode 100644
index 0000000000000..eccc470578b6e
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/components/keys_panel.vue
@@ -0,0 +1,52 @@
+<script>
+  import key from './key.vue';
+
+  export default {
+    props: {
+      title: {
+        type: String,
+        required: true,
+      },
+      keys: {
+        type: Array,
+        required: true,
+      },
+      showHelpBox: {
+        type: Boolean,
+        required: false,
+        default: true,
+      },
+      store: {
+        type: Object,
+        required: true,
+      },
+    },
+    components: {
+      key,
+    },
+  };
+</script>
+
+<template>
+  <div class="deploy-keys-panel">
+    <h5>
+      {{ title }}
+      ({{ keys.length }})
+    </h5>
+    <ul class="well-list"
+      v-if="keys.length">
+      <li
+        v-for="deployKey in keys"
+        :key="deployKey.id">
+        <key
+          :deploy-key="deployKey"
+          :store="store" />
+      </li>
+    </ul>
+    <div
+      class="settings-message text-center"
+      v-else-if="showHelpBox">
+      No deploy keys found. Create one with the form above.
+    </div>
+  </div>
+</template>
diff --git a/app/assets/javascripts/deploy_keys/eventhub.js b/app/assets/javascripts/deploy_keys/eventhub.js
new file mode 100644
index 0000000000000..0948c2e53524a
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/eventhub.js
@@ -0,0 +1,3 @@
+import Vue from 'vue';
+
+export default new Vue();
diff --git a/app/assets/javascripts/deploy_keys/index.js b/app/assets/javascripts/deploy_keys/index.js
new file mode 100644
index 0000000000000..a5f232f950a61
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/index.js
@@ -0,0 +1,21 @@
+import Vue from 'vue';
+import deployKeysApp from './components/app.vue';
+
+document.addEventListener('DOMContentLoaded', () => new Vue({
+  el: document.getElementById('js-deploy-keys'),
+  data() {
+    return {
+      endpoint: this.$options.el.dataset.endpoint,
+    };
+  },
+  components: {
+    deployKeysApp,
+  },
+  render(createElement) {
+    return createElement('deploy-keys-app', {
+      props: {
+        endpoint: this.endpoint,
+      },
+    });
+  },
+}));
diff --git a/app/assets/javascripts/deploy_keys/service/index.js b/app/assets/javascripts/deploy_keys/service/index.js
new file mode 100644
index 0000000000000..fe6dbaa949809
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/service/index.js
@@ -0,0 +1,34 @@
+import Vue from 'vue';
+import VueResource from 'vue-resource';
+
+Vue.use(VueResource);
+
+export default class DeployKeysService {
+  constructor(endpoint) {
+    this.endpoint = endpoint;
+
+    this.resource = Vue.resource(`${this.endpoint}{/id}`, {}, {
+      enable: {
+        method: 'PUT',
+        url: `${this.endpoint}{/id}/enable`,
+      },
+      disable: {
+        method: 'PUT',
+        url: `${this.endpoint}{/id}/disable`,
+      },
+    });
+  }
+
+  getKeys() {
+    return this.resource.get()
+      .then(response => response.json());
+  }
+
+  enableKey(id) {
+    return this.resource.enable({ id }, {});
+  }
+
+  disableKey(id) {
+    return this.resource.disable({ id }, {});
+  }
+}
diff --git a/app/assets/javascripts/deploy_keys/store/index.js b/app/assets/javascripts/deploy_keys/store/index.js
new file mode 100644
index 0000000000000..6210361af26e8
--- /dev/null
+++ b/app/assets/javascripts/deploy_keys/store/index.js
@@ -0,0 +1,9 @@
+export default class DeployKeysStore {
+  constructor() {
+    this.keys = {};
+  }
+
+  findEnabledKey(id) {
+    return this.keys.enabled_keys.find(key => key.id === id);
+  }
+}
diff --git a/app/assets/javascripts/dispatcher.js b/app/assets/javascripts/dispatcher.js
index c48116d45991a..3844666d25501 100644
--- a/app/assets/javascripts/dispatcher.js
+++ b/app/assets/javascripts/dispatcher.js
@@ -55,6 +55,7 @@ import ShortcutsWiki from './shortcuts_wiki';
 import BlobViewer from './blob/viewer/index';
 import GeoNodes from './geo_nodes';
 import ServiceDeskRoot from './projects/settings_service_desk/service_desk_root';
+import AutoWidthDropdownSelect from './issuable/auto_width_dropdown_select';
 
 const ShortcutsBlob = require('./shortcuts_blob');
 
@@ -204,6 +205,7 @@ const ShortcutsBlob = require('./shortcuts_blob');
           new LabelsSelect();
           new MilestoneSelect();
           new gl.IssuableTemplateSelectors();
+          new AutoWidthDropdownSelect($('.js-target-branch-select')).init();
           break;
         case 'projects:tags:new':
           new ZenMode();
@@ -263,6 +265,7 @@ const ShortcutsBlob = require('./shortcuts_blob');
           }
           break;
         case 'projects:pipelines:builds':
+        case 'projects:pipelines:failures':
         case 'projects:pipelines:show':
           const { controllerAction } = document.querySelector('.js-pipeline-container').dataset;
           const pipelineStatusUrl = `${document.querySelector('.js-pipeline-tab-link a').getAttribute('href')}/status.json`;
@@ -357,6 +360,9 @@ const ShortcutsBlob = require('./shortcuts_blob');
         case 'projects:artifacts:browse':
           new BuildArtifacts();
           break;
+        case 'projects:artifacts:file':
+          new BlobViewer();
+          break;
         case 'help:index':
           gl.VersionCheckImage.bindErrorEvent($('img.js-version-status-badge'));
           break;
diff --git a/app/assets/javascripts/droplab/constants.js b/app/assets/javascripts/droplab/constants.js
index 8883ed9aa142a..868d47e91b3e4 100644
--- a/app/assets/javascripts/droplab/constants.js
+++ b/app/assets/javascripts/droplab/constants.js
@@ -3,11 +3,14 @@ const DATA_DROPDOWN = 'data-dropdown';
 const SELECTED_CLASS = 'droplab-item-selected';
 const ACTIVE_CLASS = 'droplab-item-active';
 const IGNORE_CLASS = 'droplab-item-ignore';
+// Matches `{{anything}}` and `{{ everything }}`.
+const TEMPLATE_REGEX = /\{\{(.+?)\}\}/g;
 
 export {
   DATA_TRIGGER,
   DATA_DROPDOWN,
   SELECTED_CLASS,
   ACTIVE_CLASS,
+  TEMPLATE_REGEX,
   IGNORE_CLASS,
 };
diff --git a/app/assets/javascripts/droplab/drop_down.js b/app/assets/javascripts/droplab/drop_down.js
index 1fb4d63923c52..de3927d683ccd 100644
--- a/app/assets/javascripts/droplab/drop_down.js
+++ b/app/assets/javascripts/droplab/drop_down.js
@@ -94,7 +94,7 @@ Object.assign(DropDown.prototype, {
   },
 
   renderChildren: function(data) {
-    var html = utils.t(this.templateString, data);
+    var html = utils.template(this.templateString, data);
     var template = document.createElement('div');
 
     template.innerHTML = html;
diff --git a/app/assets/javascripts/droplab/utils.js b/app/assets/javascripts/droplab/utils.js
index c149a33a1e9ed..4da7344604eda 100644
--- a/app/assets/javascripts/droplab/utils.js
+++ b/app/assets/javascripts/droplab/utils.js
@@ -1,19 +1,19 @@
 /* eslint-disable */
 
-import { DATA_TRIGGER, DATA_DROPDOWN } from './constants';
+import { template as _template } from 'underscore';
+import { DATA_TRIGGER, DATA_DROPDOWN, TEMPLATE_REGEX } from './constants';
 
 const utils = {
   toCamelCase(attr) {
     return this.camelize(attr.split('-').slice(1).join(' '));
   },
 
-  t(s, d) {
-    for (const p in d) {
-      if (Object.prototype.hasOwnProperty.call(d, p)) {
-        s = s.replace(new RegExp(`{{${p}}}`, 'g'), d[p]);
-      }
-    }
-    return s;
+  template(templateString, data) {
+    const template = _template(templateString, {
+      escape: TEMPLATE_REGEX,
+    });
+
+    return template(data);
   },
 
   camelize(str) {
diff --git a/app/assets/javascripts/filtered_search/components/recent_searches_dropdown_content.js b/app/assets/javascripts/filtered_search/components/recent_searches_dropdown_content.js
index 9126422b33547..15052dbd362f5 100644
--- a/app/assets/javascripts/filtered_search/components/recent_searches_dropdown_content.js
+++ b/app/assets/javascripts/filtered_search/components/recent_searches_dropdown_content.js
@@ -8,6 +8,11 @@ export default {
       type: Array,
       required: true,
     },
+    isLocalStorageAvailable: {
+      type: Boolean,
+      required: false,
+      default: true,
+    },
   },
 
   computed: {
@@ -47,7 +52,12 @@ export default {
 
   template: `
     <div>
-      <ul v-if="hasItems">
+      <div
+        v-if="!isLocalStorageAvailable"
+        class="dropdown-info-note">
+        This feature requires local storage to be enabled
+      </div>
+      <ul v-else-if="hasItems">
         <li
           v-for="(item, index) in processedItems"
           :key="index">
diff --git a/app/assets/javascripts/filtered_search/dropdown_hint.js b/app/assets/javascripts/filtered_search/dropdown_hint.js
index 3e7a892756cdd..5e9434fd48f4d 100644
--- a/app/assets/javascripts/filtered_search/dropdown_hint.js
+++ b/app/assets/javascripts/filtered_search/dropdown_hint.js
@@ -62,7 +62,7 @@ class DropdownHint extends gl.FilteredSearchDropdown {
           Object.assign({
             icon: `fa-${icon}`,
             hint,
-            tag: `&lt;${tag}&gt;`,
+            tag: `<${tag}>`,
           }, type && { type }),
         );
       }
diff --git a/app/assets/javascripts/filtered_search/filtered_search_manager.js b/app/assets/javascripts/filtered_search/filtered_search_manager.js
index 58ff18acd4c9f..22fe2c7f721b3 100644
--- a/app/assets/javascripts/filtered_search/filtered_search_manager.js
+++ b/app/assets/javascripts/filtered_search/filtered_search_manager.js
@@ -1,5 +1,3 @@
-/* global Flash */
-
 import FilteredSearchContainer from './container';
 import RecentSearchesRoot from './recent_searches_root';
 import RecentSearchesStore from './stores/recent_searches_store';
@@ -19,7 +17,9 @@ class FilteredSearchManager {
       this.filteredSearchTokenKeys = gl.FilteredSearchTokenKeysWithWeights;
     }
 
-    this.recentSearchesStore = new RecentSearchesStore();
+    this.recentSearchesStore = new RecentSearchesStore({
+      isLocalStorageAvailable: RecentSearchesService.isAvailable(),
+    });
     let recentSearchesKey = 'issue-recent-searches';
     if (page === 'merge_requests') {
       recentSearchesKey = 'merge-request-recent-searches';
@@ -28,9 +28,10 @@ class FilteredSearchManager {
 
     // Fetch recent searches from localStorage
     this.fetchingRecentSearchesPromise = this.recentSearchesService.fetch()
-      .catch(() => {
+      .catch((error) => {
+        if (error.name === 'RecentSearchesServiceError') return undefined;
         // eslint-disable-next-line no-new
-        new Flash('An error occured while parsing recent searches');
+        new window.Flash('An error occured while parsing recent searches');
         // Gracefully fail to empty array
         return [];
       })
diff --git a/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js b/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
index 453ecccc6fc8d..f3003b86493ea 100644
--- a/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
+++ b/app/assets/javascripts/filtered_search/filtered_search_visual_tokens.js
@@ -1,3 +1,5 @@
+import AjaxCache from '~/lib/utils/ajax_cache';
+import '~/flash'; /* global Flash */
 import FilteredSearchContainer from './container';
 
 class FilteredSearchVisualTokens {
@@ -48,6 +50,40 @@ class FilteredSearchVisualTokens {
     `;
   }
 
+  static updateLabelTokenColor(tokenValueContainer, tokenValue) {
+    const filteredSearchInput = FilteredSearchContainer.container.querySelector('.filtered-search');
+    const baseEndpoint = filteredSearchInput.dataset.baseEndpoint;
+    const labelsEndpoint = `${baseEndpoint}/labels.json`;
+
+    return AjaxCache.retrieve(labelsEndpoint)
+    .then((labels) => {
+      const matchingLabel = (labels || []).find(label => `~${gl.DropdownUtils.getEscapedText(label.title)}` === tokenValue);
+
+      if (!matchingLabel) {
+        return;
+      }
+
+      const tokenValueStyle = tokenValueContainer.style;
+      tokenValueStyle.backgroundColor = matchingLabel.color;
+      tokenValueStyle.color = matchingLabel.text_color;
+
+      if (matchingLabel.text_color === '#FFFFFF') {
+        const removeToken = tokenValueContainer.querySelector('.remove-token');
+        removeToken.classList.add('inverted');
+      }
+    })
+    .catch(() => new Flash('An error occurred while fetching label colors.'));
+  }
+
+  static renderVisualTokenValue(parentElement, tokenName, tokenValue) {
+    const tokenValueContainer = parentElement.querySelector('.value-container');
+    tokenValueContainer.querySelector('.value').innerText = tokenValue;
+
+    if (tokenName.toLowerCase() === 'label') {
+      FilteredSearchVisualTokens.updateLabelTokenColor(tokenValueContainer, tokenValue);
+    }
+  }
+
   static addVisualTokenElement(name, value, isSearchTerm) {
     const li = document.createElement('li');
     li.classList.add('js-visual-token');
@@ -55,7 +91,7 @@ class FilteredSearchVisualTokens {
 
     if (value) {
       li.innerHTML = FilteredSearchVisualTokens.createVisualTokenElementHTML();
-      li.querySelector('.value').innerText = value;
+      FilteredSearchVisualTokens.renderVisualTokenValue(li, name, value);
     } else {
       li.innerHTML = '<div class="name"></div>';
     }
@@ -74,7 +110,7 @@ class FilteredSearchVisualTokens {
       const name = FilteredSearchVisualTokens.getLastTokenPartial();
       lastVisualToken.innerHTML = FilteredSearchVisualTokens.createVisualTokenElementHTML();
       lastVisualToken.querySelector('.name').innerText = name;
-      lastVisualToken.querySelector('.value').innerText = value;
+      FilteredSearchVisualTokens.renderVisualTokenValue(lastVisualToken, name, value);
     }
   }
 
@@ -183,6 +219,9 @@ class FilteredSearchVisualTokens {
 
   static moveInputToTheRight() {
     const input = FilteredSearchContainer.container.querySelector('.filtered-search');
+
+    if (!input) return;
+
     const inputLi = input.parentElement;
     const tokenContainer = FilteredSearchContainer.container.querySelector('.tokens-container');
 
diff --git a/app/assets/javascripts/filtered_search/recent_searches_root.js b/app/assets/javascripts/filtered_search/recent_searches_root.js
index 4e38409e12a5a..b2e6f63aacf34 100644
--- a/app/assets/javascripts/filtered_search/recent_searches_root.js
+++ b/app/assets/javascripts/filtered_search/recent_searches_root.js
@@ -29,12 +29,15 @@ class RecentSearchesRoot {
   }
 
   render() {
+    const state = this.store.state;
     this.vm = new Vue({
       el: this.wrapperElement,
-      data: this.store.state,
+      data() { return state; },
       template: `
         <recent-searches-dropdown-content
-          :items="recentSearches" />
+          :items="recentSearches"
+          :is-local-storage-available="isLocalStorageAvailable"
+          />
       `,
       components: {
         'recent-searches-dropdown-content': RecentSearchesDropdownContent,
diff --git a/app/assets/javascripts/filtered_search/services/recent_searches_service.js b/app/assets/javascripts/filtered_search/services/recent_searches_service.js
index 3e402d5aed006..a056dea928dcc 100644
--- a/app/assets/javascripts/filtered_search/services/recent_searches_service.js
+++ b/app/assets/javascripts/filtered_search/services/recent_searches_service.js
@@ -1,9 +1,17 @@
+import RecentSearchesServiceError from './recent_searches_service_error';
+import AccessorUtilities from '../../lib/utils/accessor';
+
 class RecentSearchesService {
   constructor(localStorageKey = 'issuable-recent-searches') {
     this.localStorageKey = localStorageKey;
   }
 
   fetch() {
+    if (!RecentSearchesService.isAvailable()) {
+      const error = new RecentSearchesServiceError();
+      return Promise.reject(error);
+    }
+
     const input = window.localStorage.getItem(this.localStorageKey);
 
     let searches = [];
@@ -19,8 +27,14 @@ class RecentSearchesService {
   }
 
   save(searches = []) {
+    if (!RecentSearchesService.isAvailable()) return;
+
     window.localStorage.setItem(this.localStorageKey, JSON.stringify(searches));
   }
+
+  static isAvailable() {
+    return AccessorUtilities.isLocalStorageAccessSafe();
+  }
 }
 
 export default RecentSearchesService;
diff --git a/app/assets/javascripts/filtered_search/services/recent_searches_service_error.js b/app/assets/javascripts/filtered_search/services/recent_searches_service_error.js
new file mode 100644
index 0000000000000..5917b223d63fd
--- /dev/null
+++ b/app/assets/javascripts/filtered_search/services/recent_searches_service_error.js
@@ -0,0 +1,11 @@
+class RecentSearchesServiceError {
+  constructor(message) {
+    this.name = 'RecentSearchesServiceError';
+    this.message = message || 'Recent Searches Service is unavailable';
+  }
+}
+
+// Can't use `extends` for builtin prototypes and get true inheritance yet
+RecentSearchesServiceError.prototype = Error.prototype;
+
+export default RecentSearchesServiceError;
diff --git a/app/assets/javascripts/gfm_auto_complete.js b/app/assets/javascripts/gfm_auto_complete.js
index 687a462a0d41a..f1b99023c723c 100644
--- a/app/assets/javascripts/gfm_auto_complete.js
+++ b/app/assets/javascripts/gfm_auto_complete.js
@@ -101,9 +101,17 @@ window.gl.GfmAutoComplete = {
       }
     }
   },
-  setup: function(input) {
+  setup: function(input, enableMap = {
+    emojis: true,
+    members: true,
+    issues: true,
+    milestones: true,
+    mergeRequests: true,
+    labels: true
+  }) {
     // Add GFM auto-completion to all input fields, that accept GFM input.
     this.input = input || $('.js-gfm-input');
+    this.enableMap = enableMap;
     this.setupLifecycle();
   },
   setupLifecycle() {
@@ -115,7 +123,84 @@ window.gl.GfmAutoComplete = {
       $input.on('inserted-commands.atwho', $input.trigger.bind($input, 'keyup'));
     });
   },
+
   setupAtWho: function($input) {
+    if (this.enableMap.emojis) this.setupEmoji($input);
+    if (this.enableMap.members) this.setupMembers($input);
+    if (this.enableMap.issues) this.setupIssues($input);
+    if (this.enableMap.milestones) this.setupMilestones($input);
+    if (this.enableMap.mergeRequests) this.setupMergeRequests($input);
+    if (this.enableMap.labels) this.setupLabels($input);
+
+    // We don't instantiate the slash commands autocomplete for note and issue/MR edit forms
+    $input.filter('[data-supports-slash-commands="true"]').atwho({
+      at: '/',
+      alias: 'commands',
+      searchKey: 'search',
+      skipSpecialCharacterTest: true,
+      data: this.defaultLoadingData,
+      displayTpl: function(value) {
+        if (this.isLoading(value)) return this.Loading.template;
+        var tpl = '<li>/${name}';
+        if (value.aliases.length > 0) {
+          tpl += ' <small>(or /<%- aliases.join(", /") %>)</small>';
+        }
+        if (value.params.length > 0) {
+          tpl += ' <small><%- params.join(" ") %></small>';
+        }
+        if (value.description !== '') {
+          tpl += '<small class="description"><i><%- description %></i></small>';
+        }
+        tpl += '</li>';
+        return _.template(tpl)(value);
+      }.bind(this),
+      insertTpl: function(value) {
+        var tpl = "/${name} ";
+        var reference_prefix = null;
+        if (value.params.length > 0) {
+          reference_prefix = value.params[0][0];
+          if (/^[@%~]/.test(reference_prefix)) {
+            tpl += '<%- reference_prefix %>';
+          }
+        }
+        return _.template(tpl)({ reference_prefix: reference_prefix });
+      },
+      suffix: '',
+      callbacks: {
+        sorter: this.DefaultOptions.sorter,
+        filter: this.DefaultOptions.filter,
+        beforeInsert: this.DefaultOptions.beforeInsert,
+        beforeSave: function(commands) {
+          if (gl.GfmAutoComplete.isLoading(commands)) return commands;
+          return $.map(commands, function(c) {
+            var search = c.name;
+            if (c.aliases.length > 0) {
+              search = search + " " + c.aliases.join(" ");
+            }
+            return {
+              name: c.name,
+              aliases: c.aliases,
+              params: c.params,
+              description: c.description,
+              search: search
+            };
+          });
+        },
+        matcher: function(flag, subtext, should_startWithSpace, acceptSpaceBar) {
+          var regexp = /(?:^|\n)\/([A-Za-z_]*)$/gi;
+          var match = regexp.exec(subtext);
+          if (match) {
+            return match[1];
+          } else {
+            return null;
+          }
+        }
+      }
+    });
+    return;
+  },
+
+  setupEmoji($input) {
     // Emoji
     $input.atwho({
       at: ':',
@@ -139,6 +224,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupMembers($input) {
     // Team Members
     $input.atwho({
       at: '@',
@@ -180,6 +268,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupIssues($input) {
     $input.atwho({
       at: '#',
       alias: 'issues',
@@ -208,6 +299,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupMilestones($input) {
     $input.atwho({
       at: '%',
       alias: 'milestones',
@@ -236,6 +330,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupMergeRequests($input) {
     $input.atwho({
       at: '!',
       alias: 'mergerequests',
@@ -264,6 +361,9 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
+  },
+
+  setupLabels($input) {
     $input.atwho({
       at: '~',
       alias: 'labels',
@@ -298,73 +398,8 @@ window.gl.GfmAutoComplete = {
         }
       }
     });
-    // We don't instantiate the slash commands autocomplete for note and issue/MR edit forms
-    $input.filter('[data-supports-slash-commands="true"]').atwho({
-      at: '/',
-      alias: 'commands',
-      searchKey: 'search',
-      skipSpecialCharacterTest: true,
-      data: this.defaultLoadingData,
-      displayTpl: function(value) {
-        if (this.isLoading(value)) return this.Loading.template;
-        var tpl = '<li>/${name}';
-        if (value.aliases.length > 0) {
-          tpl += ' <small>(or /<%- aliases.join(", /") %>)</small>';
-        }
-        if (value.params.length > 0) {
-          tpl += ' <small><%- params.join(" ") %></small>';
-        }
-        if (value.description !== '') {
-          tpl += '<small class="description"><i><%- description %></i></small>';
-        }
-        tpl += '</li>';
-        return _.template(tpl)(value);
-      }.bind(this),
-      insertTpl: function(value) {
-        var tpl = "/${name} ";
-        var reference_prefix = null;
-        if (value.params.length > 0) {
-          reference_prefix = value.params[0][0];
-          if (/^[@%~]/.test(reference_prefix)) {
-            tpl += '<%- reference_prefix %>';
-          }
-        }
-        return _.template(tpl)({ reference_prefix: reference_prefix });
-      },
-      suffix: '',
-      callbacks: {
-        sorter: this.DefaultOptions.sorter,
-        filter: this.DefaultOptions.filter,
-        beforeInsert: this.DefaultOptions.beforeInsert,
-        beforeSave: function(commands) {
-          if (gl.GfmAutoComplete.isLoading(commands)) return commands;
-          return $.map(commands, function(c) {
-            var search = c.name;
-            if (c.aliases.length > 0) {
-              search = search + " " + c.aliases.join(" ");
-            }
-            return {
-              name: c.name,
-              aliases: c.aliases,
-              params: c.params,
-              description: c.description,
-              search: search
-            };
-          });
-        },
-        matcher: function(flag, subtext, should_startWithSpace, acceptSpaceBar) {
-          var regexp = /(?:^|\n)\/([A-Za-z_]*)$/gi;
-          var match = regexp.exec(subtext);
-          if (match) {
-            return match[1];
-          } else {
-            return null;
-          }
-        }
-      }
-    });
-    return;
   },
+
   fetchData: function($input, at) {
     if (this.isLoadingData[at]) return;
     this.isLoadingData[at] = true;
diff --git a/app/assets/javascripts/gl_dropdown.js b/app/assets/javascripts/gl_dropdown.js
index 491c34869c2ff..72f147242b2a4 100644
--- a/app/assets/javascripts/gl_dropdown.js
+++ b/app/assets/javascripts/gl_dropdown.js
@@ -255,7 +255,8 @@ GitLabDropdown = (function() {
               }
             };
           // Remote data
-          })(this)
+          })(this),
+          instance: this,
         });
       }
     }
@@ -269,6 +270,7 @@ GitLabDropdown = (function() {
         remote: this.options.filterRemote,
         query: this.options.data,
         keys: searchFields,
+        instance: this,
         elements: (function(_this) {
           return function() {
             selector = '.dropdown-content li:not(' + NON_SELECTABLE_CLASSES + ')';
@@ -343,21 +345,26 @@ GitLabDropdown = (function() {
       }
       this.dropdown.on("click", selector, function(e) {
         var $el, selected, selectedObj, isMarking;
-        $el = $(this);
+        $el = $(e.currentTarget);
         selected = self.rowClicked($el);
         selectedObj = selected ? selected[0] : null;
         isMarking = selected ? selected[1] : null;
-        if (self.options.clicked) {
-          self.options.clicked(selectedObj, $el, e, isMarking);
+        if (this.options.clicked) {
+          this.options.clicked.call(this, {
+            selectedObj,
+            $el,
+            e,
+            isMarking,
+          });
         }
 
         // Update label right after all modifications in dropdown has been done
-        if (self.options.toggleLabel) {
-          self.updateLabel(selectedObj, $el, self);
+        if (this.options.toggleLabel) {
+          this.updateLabel(selectedObj, $el, this);
         }
 
         $el.trigger('blur');
-      });
+      }.bind(this));
     }
   }
 
@@ -439,15 +446,34 @@ GitLabDropdown = (function() {
     }
   };
 
+  GitLabDropdown.prototype.filteredFullData = function() {
+    return this.fullData.filter(r => typeof r === 'object'
+      && !Object.prototype.hasOwnProperty.call(r, 'beforeDivider')
+      && !Object.prototype.hasOwnProperty.call(r, 'header')
+    );
+  };
+
   GitLabDropdown.prototype.opened = function(e) {
     var contentHtml;
     this.resetRows();
     this.addArrowKeyEvent();
 
+    const dropdownToggle = this.dropdown.find('.dropdown-menu-toggle');
+    const hasFilterBulkUpdate = dropdownToggle.hasClass('js-filter-bulk-update');
+    const hasMultiSelect = dropdownToggle.hasClass('js-multiselect');
+
     // Makes indeterminate items effective
-    if (this.fullData && this.dropdown.find('.dropdown-menu-toggle').hasClass('js-filter-bulk-update')) {
+    if (this.fullData && hasFilterBulkUpdate) {
       this.parseData(this.fullData);
     }
+
+    // Process the data to make sure rendered data
+    // matches the correct layout
+    if (this.fullData && hasMultiSelect && this.options.processData) {
+      const inputValue = this.filterInput.val();
+      this.options.processData.call(this.options, inputValue, this.filteredFullData(), this.parseData.bind(this));
+    }
+
     contentHtml = $('.dropdown-content', this.dropdown).html();
     if (this.remote && contentHtml === "") {
       this.remote.execute();
@@ -724,6 +750,11 @@ GitLabDropdown = (function() {
     if (this.options.inputId != null) {
       $input.attr('id', this.options.inputId);
     }
+
+    if (this.options.inputMeta) {
+      $input.attr('data-meta', selectedObject[this.options.inputMeta]);
+    }
+
     return this.dropdown.before($input);
   };
 
@@ -844,7 +875,14 @@ GitLabDropdown = (function() {
     if (instance == null) {
       instance = null;
     }
-    return $(this.el).find(".dropdown-toggle-text").text(this.options.toggleLabel(selected, el, instance));
+
+    let toggleText = this.options.toggleLabel(selected, el, instance);
+    if (this.options.updateLabel) {
+      // Option to override the dropdown label text
+      toggleText = this.options.updateLabel;
+    }
+
+    return $(this.el).find(".dropdown-toggle-text").text(toggleText);
   };
 
   GitLabDropdown.prototype.clearField = function(field, isInput) {
diff --git a/app/assets/javascripts/issuable/auto_width_dropdown_select.js b/app/assets/javascripts/issuable/auto_width_dropdown_select.js
new file mode 100644
index 0000000000000..2203a56315e86
--- /dev/null
+++ b/app/assets/javascripts/issuable/auto_width_dropdown_select.js
@@ -0,0 +1,38 @@
+let instanceCount = 0;
+
+class AutoWidthDropdownSelect {
+  constructor(selectElement) {
+    this.$selectElement = $(selectElement);
+    this.dropdownClass = `js-auto-width-select-dropdown-${instanceCount}`;
+    instanceCount += 1;
+  }
+
+  init() {
+    const dropdownClass = this.dropdownClass;
+    this.$selectElement.select2({
+      dropdownCssClass: dropdownClass,
+      dropdownCss() {
+        let resultantWidth = 'auto';
+        const $dropdown = $(`.${dropdownClass}`);
+
+        // We have to look at the parent because
+        // `offsetParent` on a `display: none;` is `null`
+        const offsetParentWidth = $(this).parent().offsetParent().width();
+        // Reset any width to let it naturally flow
+        $dropdown.css('width', 'auto');
+        if ($dropdown.outerWidth(false) > offsetParentWidth) {
+          resultantWidth = offsetParentWidth;
+        }
+
+        return {
+          width: resultantWidth,
+          maxWidth: offsetParentWidth,
+        };
+      },
+    });
+
+    return this;
+  }
+}
+
+export default AutoWidthDropdownSelect;
diff --git a/app/assets/javascripts/issuable/issuable_bundle.js b/app/assets/javascripts/issuable/issuable_bundle.js
deleted file mode 100644
index e927cc0077c5b..0000000000000
--- a/app/assets/javascripts/issuable/issuable_bundle.js
+++ /dev/null
@@ -1 +0,0 @@
-require('./time_tracking/time_tracking_bundle');
diff --git a/app/assets/javascripts/issuable/time_tracking/components/collapsed_state.js b/app/assets/javascripts/issuable/time_tracking/components/collapsed_state.js
deleted file mode 100644
index 2fdf7a3cdb2f4..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/collapsed_state.js
+++ /dev/null
@@ -1,60 +0,0 @@
-import Vue from 'vue';
-import stopwatchSvg from 'icons/_icon_stopwatch.svg';
-
-require('../../../lib/utils/pretty_time');
-
-(() => {
-  Vue.component('time-tracking-collapsed-state', {
-    name: 'time-tracking-collapsed-state',
-    props: {
-      showComparisonState: {
-        type: Boolean,
-        required: true,
-      },
-      showSpentOnlyState: {
-        type: Boolean,
-        required: true,
-      },
-      showEstimateOnlyState: {
-        type: Boolean,
-        required: true,
-      },
-      showNoTimeTrackingState: {
-        type: Boolean,
-        required: true,
-      },
-      timeSpentHumanReadable: {
-        type: String,
-        required: false,
-      },
-      timeEstimateHumanReadable: {
-        type: String,
-        required: false,
-      },
-    },
-    methods: {
-      abbreviateTime(timeStr) {
-        return gl.utils.prettyTime.abbreviateTime(timeStr);
-      },
-    },
-    template: `
-      <div class='sidebar-collapsed-icon'>
-        ${stopwatchSvg}
-        <div class='time-tracking-collapsed-summary'>
-          <div class='compare' v-if='showComparisonState'>
-            <span>{{ abbreviateTime(timeSpentHumanReadable) }} / {{ abbreviateTime(timeEstimateHumanReadable) }}</span>
-          </div>
-          <div class='estimate-only' v-if='showEstimateOnlyState'>
-            <span class='bold'>-- / {{ abbreviateTime(timeEstimateHumanReadable) }}</span>
-          </div>
-          <div class='spend-only' v-if='showSpentOnlyState'>
-            <span class='bold'>{{ abbreviateTime(timeSpentHumanReadable) }} / --</span>
-          </div>
-          <div class='no-tracking' v-if='showNoTimeTrackingState'>
-            <span class='no-value'>None</span>
-          </div>
-        </div>
-      </div>
-      `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/comparison_pane.js b/app/assets/javascripts/issuable/time_tracking/components/comparison_pane.js
deleted file mode 100644
index 21914ad794620..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/comparison_pane.js
+++ /dev/null
@@ -1,82 +0,0 @@
-import Vue from 'vue';
-
-require('../../../lib/utils/pretty_time');
-
-(() => {
-  const prettyTime = gl.utils.prettyTime;
-
-  Vue.component('time-tracking-comparison-pane', {
-    name: 'time-tracking-comparison-pane',
-    props: {
-      timeSpent: {
-        type: Number,
-        required: true,
-      },
-      timeEstimate: {
-        type: Number,
-        required: true,
-      },
-      timeSpentHumanReadable: {
-        type: String,
-        required: true,
-      },
-      timeEstimateHumanReadable: {
-        type: String,
-        required: true,
-      },
-    },
-    computed: {
-      parsedRemaining() {
-        const diffSeconds = this.timeEstimate - this.timeSpent;
-        return prettyTime.parseSeconds(diffSeconds);
-      },
-      timeRemainingHumanReadable() {
-        return prettyTime.stringifyTime(this.parsedRemaining);
-      },
-      timeRemainingTooltip() {
-        const prefix = this.timeRemainingMinutes < 0 ? 'Over by' : 'Time remaining:';
-        return `${prefix} ${this.timeRemainingHumanReadable}`;
-      },
-      /* Diff values for comparison meter */
-      timeRemainingMinutes() {
-        return this.timeEstimate - this.timeSpent;
-      },
-      timeRemainingPercent() {
-        return `${Math.floor((this.timeSpent / this.timeEstimate) * 100)}%`;
-      },
-      timeRemainingStatusClass() {
-        return this.timeEstimate >= this.timeSpent ? 'within_estimate' : 'over_estimate';
-      },
-      /* Parsed time values */
-      parsedEstimate() {
-        return prettyTime.parseSeconds(this.timeEstimate);
-      },
-      parsedSpent() {
-        return prettyTime.parseSeconds(this.timeSpent);
-      },
-    },
-    template: `
-      <div class='time-tracking-comparison-pane'>
-        <div class='compare-meter' data-toggle='tooltip' data-placement='top' role='timeRemainingDisplay'
-          :aria-valuenow='timeRemainingTooltip'
-          :title='timeRemainingTooltip'
-          :data-original-title='timeRemainingTooltip'
-          :class='timeRemainingStatusClass'>
-          <div class='meter-container' role='timeSpentPercent' :aria-valuenow='timeRemainingPercent'>
-            <div :style='{ width: timeRemainingPercent }' class='meter-fill'></div>
-          </div>
-          <div class='compare-display-container'>
-            <div class='compare-display pull-left'>
-              <span class='compare-label'>Spent</span>
-              <span class='compare-value spent'>{{ timeSpentHumanReadable }}</span>
-            </div>
-            <div class='compare-display estimated pull-right'>
-              <span class='compare-label'>Est</span>
-              <span class='compare-value'>{{ timeEstimateHumanReadable }}</span>
-            </div>
-          </div>
-        </div>
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/estimate_only_pane.js b/app/assets/javascripts/issuable/time_tracking/components/estimate_only_pane.js
deleted file mode 100644
index deb84e5e43a95..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/estimate_only_pane.js
+++ /dev/null
@@ -1,19 +0,0 @@
-import Vue from 'vue';
-
-(() => {
-  Vue.component('time-tracking-estimate-only-pane', {
-    name: 'time-tracking-estimate-only-pane',
-    props: {
-      timeEstimateHumanReadable: {
-        type: String,
-        required: true,
-      },
-    },
-    template: `
-      <div class='time-tracking-estimate-only-pane'>
-        <span class='bold'>Estimated:</span>
-        {{ timeEstimateHumanReadable }}
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/help_state.js b/app/assets/javascripts/issuable/time_tracking/components/help_state.js
deleted file mode 100644
index 25e11b9d1fce8..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/help_state.js
+++ /dev/null
@@ -1,30 +0,0 @@
-import Vue from 'vue';
-
-(() => {
-  Vue.component('time-tracking-help-state', {
-    name: 'time-tracking-help-state',
-    props: {
-      docsUrl: {
-        type: String,
-        required: true,
-      },
-    },
-    template: `
-      <div class='time-tracking-help-state'>
-        <div class='time-tracking-info'>
-          <h4>Track time with slash commands</h4>
-          <p>Slash commands can be used in the issues description and comment boxes.</p>
-          <p>
-            <code>/estimate</code>
-            will update the estimated time with the latest command.
-          </p>
-          <p>
-            <code>/spend</code>
-            will update the sum of the time spent.
-          </p>
-          <a class='btn btn-default learn-more-button' :href='docsUrl'>Learn more</a>
-        </div>
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/no_tracking_pane.js b/app/assets/javascripts/issuable/time_tracking/components/no_tracking_pane.js
deleted file mode 100644
index b081adf5e643b..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/no_tracking_pane.js
+++ /dev/null
@@ -1,12 +0,0 @@
-import Vue from 'vue';
-
-(() => {
-  Vue.component('time-tracking-no-tracking-pane', {
-    name: 'time-tracking-no-tracking-pane',
-    template: `
-      <div class='time-tracking-no-tracking-pane'>
-        <span class='no-value'>No estimate or time spent</span>
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/spent_only_pane.js b/app/assets/javascripts/issuable/time_tracking/components/spent_only_pane.js
deleted file mode 100644
index 52a9f5dce23a9..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/spent_only_pane.js
+++ /dev/null
@@ -1,19 +0,0 @@
-import Vue from 'vue';
-
-(() => {
-  Vue.component('time-tracking-spent-only-pane', {
-    name: 'time-tracking-spent-only-pane',
-    props: {
-      timeSpentHumanReadable: {
-        type: String,
-        required: true,
-      },
-    },
-    template: `
-      <div class='time-tracking-spend-only-pane'>
-        <span class='bold'>Spent:</span>
-        {{ timeSpentHumanReadable }}
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/components/time_tracker.js b/app/assets/javascripts/issuable/time_tracking/components/time_tracker.js
deleted file mode 100644
index 6e4f11c65b77a..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/components/time_tracker.js
+++ /dev/null
@@ -1,135 +0,0 @@
-import Vue from 'vue';
-
-require('./help_state');
-require('./collapsed_state');
-require('./spent_only_pane');
-require('./no_tracking_pane');
-require('./estimate_only_pane');
-require('./comparison_pane');
-
-(() => {
-  Vue.component('issuable-time-tracker', {
-    name: 'issuable-time-tracker',
-    props: {
-      time_estimate: {
-        type: Number,
-        required: true,
-        default: 0,
-      },
-      time_spent: {
-        type: Number,
-        required: true,
-        default: 0,
-      },
-      human_time_estimate: {
-        type: String,
-        required: false,
-      },
-      human_time_spent: {
-        type: String,
-        required: false,
-      },
-      docsUrl: {
-        type: String,
-        required: true,
-      },
-    },
-    data() {
-      return {
-        showHelp: false,
-      };
-    },
-    computed: {
-      timeSpent() {
-        return this.time_spent;
-      },
-      timeEstimate() {
-        return this.time_estimate;
-      },
-      timeEstimateHumanReadable() {
-        return this.human_time_estimate;
-      },
-      timeSpentHumanReadable() {
-        return this.human_time_spent;
-      },
-      hasTimeSpent() {
-        return !!this.timeSpent;
-      },
-      hasTimeEstimate() {
-        return !!this.timeEstimate;
-      },
-      showComparisonState() {
-        return this.hasTimeEstimate && this.hasTimeSpent;
-      },
-      showEstimateOnlyState() {
-        return this.hasTimeEstimate && !this.hasTimeSpent;
-      },
-      showSpentOnlyState() {
-        return this.hasTimeSpent && !this.hasTimeEstimate;
-      },
-      showNoTimeTrackingState() {
-        return !this.hasTimeEstimate && !this.hasTimeSpent;
-      },
-      showHelpState() {
-        return !!this.showHelp;
-      },
-    },
-    methods: {
-      toggleHelpState(show) {
-        this.showHelp = show;
-      },
-    },
-    template: `
-      <div class='time_tracker time-tracking-component-wrap' v-cloak>
-        <time-tracking-collapsed-state
-          :show-comparison-state='showComparisonState'
-          :show-no-time-tracking-state='showNoTimeTrackingState'
-          :show-help-state='showHelpState'
-          :show-spent-only-state='showSpentOnlyState'
-          :show-estimate-only-state='showEstimateOnlyState'
-          :time-spent-human-readable='timeSpentHumanReadable'
-          :time-estimate-human-readable='timeEstimateHumanReadable'>
-        </time-tracking-collapsed-state>
-        <div class='title hide-collapsed'>
-          Time tracking
-          <div class='help-button pull-right'
-            v-if='!showHelpState'
-            @click='toggleHelpState(true)'>
-            <i class='fa fa-question-circle' aria-hidden='true'></i>
-          </div>
-          <div class='close-help-button pull-right'
-            v-if='showHelpState'
-            @click='toggleHelpState(false)'>
-            <i class='fa fa-close' aria-hidden='true'></i>
-          </div>
-        </div>
-        <div class='time-tracking-content hide-collapsed'>
-          <time-tracking-estimate-only-pane
-            v-if='showEstimateOnlyState'
-            :time-estimate-human-readable='timeEstimateHumanReadable'>
-          </time-tracking-estimate-only-pane>
-          <time-tracking-spent-only-pane
-            v-if='showSpentOnlyState'
-            :time-spent-human-readable='timeSpentHumanReadable'>
-          </time-tracking-spent-only-pane>
-          <time-tracking-no-tracking-pane
-            v-if='showNoTimeTrackingState'>
-          </time-tracking-no-tracking-pane>
-          <time-tracking-comparison-pane
-            v-if='showComparisonState'
-            :time-estimate='timeEstimate'
-            :time-spent='timeSpent'
-            :time-spent-human-readable='timeSpentHumanReadable'
-            :time-estimate-human-readable='timeEstimateHumanReadable'>
-          </time-tracking-comparison-pane>
-          <transition name='help-state-toggle'>
-            <time-tracking-help-state
-              v-if='showHelpState'
-              :docs-url='docsUrl'>
-            </time-tracking-help-state>
-          </transition>
-        </div>
-      </div>
-    `,
-  });
-})();
diff --git a/app/assets/javascripts/issuable/time_tracking/time_tracking_bundle.js b/app/assets/javascripts/issuable/time_tracking/time_tracking_bundle.js
deleted file mode 100644
index 1689a69e1ed94..0000000000000
--- a/app/assets/javascripts/issuable/time_tracking/time_tracking_bundle.js
+++ /dev/null
@@ -1,66 +0,0 @@
-import Vue from 'vue';
-import VueResource from 'vue-resource';
-
-require('./components/time_tracker');
-require('../../smart_interval');
-require('../../subbable_resource');
-
-Vue.use(VueResource);
-
-(() => {
-  /* This Vue instance represents what will become the parent instance for the
-    * sidebar. It will be responsible for managing `issuable` state and propagating
-    * changes to sidebar components. We will want to create a separate service to
-    * interface with the server at that point.
-   */
-
-  class IssuableTimeTracking {
-    constructor(issuableJSON) {
-      const parsedIssuable = JSON.parse(issuableJSON);
-      return this.initComponent(parsedIssuable);
-    }
-
-    initComponent(parsedIssuable) {
-      this.parentInstance = new Vue({
-        el: '#issuable-time-tracker',
-        data: {
-          issuable: parsedIssuable,
-        },
-        methods: {
-          fetchIssuable() {
-            return gl.IssuableResource.get.call(gl.IssuableResource, {
-              type: 'GET',
-              url: gl.IssuableResource.endpoint,
-            });
-          },
-          updateState(data) {
-            this.issuable = data;
-          },
-          subscribeToUpdates() {
-            gl.IssuableResource.subscribe(data => this.updateState(data));
-          },
-          listenForSlashCommands() {
-            $(document).on('ajax:success', '.gfm-form', (e, data) => {
-              const subscribedCommands = ['spend_time', 'time_estimate'];
-              const changedCommands = data.commands_changes
-                ? Object.keys(data.commands_changes)
-                : [];
-              if (changedCommands && _.intersection(subscribedCommands, changedCommands).length) {
-                this.fetchIssuable();
-              }
-            });
-          },
-        },
-        created() {
-          this.fetchIssuable();
-        },
-        mounted() {
-          this.subscribeToUpdates();
-          this.listenForSlashCommands();
-        },
-      });
-    }
-  }
-
-  gl.IssuableTimeTracking = IssuableTimeTracking;
-})(window.gl || (window.gl = {}));
diff --git a/app/assets/javascripts/issue_status_select.js b/app/assets/javascripts/issue_status_select.js
index b2cfd3ef2a3cb..56cb536dcde7d 100644
--- a/app/assets/javascripts/issue_status_select.js
+++ b/app/assets/javascripts/issue_status_select.js
@@ -19,8 +19,8 @@
               return label;
             };
           })(this),
-          clicked: function(item, $el, e) {
-            return e.preventDefault();
+          clicked: function(options) {
+            return options.e.preventDefault();
           },
           id: function(obj, el) {
             return $(el).data("id");
diff --git a/app/assets/javascripts/issues_bulk_assignment.js b/app/assets/javascripts/issues_bulk_assignment.js
index e0ebd36a65ce8..fee3429e2b846 100644
--- a/app/assets/javascripts/issues_bulk_assignment.js
+++ b/app/assets/javascripts/issues_bulk_assignment.js
@@ -88,7 +88,10 @@
       const formData = {
         update: {
           state_event: this.form.find('input[name="update[state_event]"]').val(),
+          // For Merge Requests
           assignee_id: this.form.find('input[name="update[assignee_id]"]').val(),
+          // For Issues
+          assignee_ids: [this.form.find('input[name="update[assignee_ids][]"]').val()],
           milestone_id: this.form.find('input[name="update[milestone_id]"]').val(),
           issuable_ids: this.form.find('input[name="update[issuable_ids]"]').val(),
           subscription_event: this.form.find('input[name="update[subscription_event]"]').val(),
diff --git a/app/assets/javascripts/labels_select.js b/app/assets/javascripts/labels_select.js
index 9a60f5464df95..ac5ce84e31b1f 100644
--- a/app/assets/javascripts/labels_select.js
+++ b/app/assets/javascripts/labels_select.js
@@ -330,7 +330,10 @@
           },
           multiSelect: $dropdown.hasClass('js-multiselect'),
           vue: $dropdown.hasClass('js-issue-board-sidebar'),
-          clicked: function(label, $el, e, isMarking) {
+          clicked: function(options) {
+            const { $el, e, isMarking } = options;
+            const label = options.selectedObj;
+
             var isIssueIndex, isMRIndex, page, boardsModel;
             var fadeOutLoader = () => {
               $loading.fadeOut();
@@ -352,7 +355,7 @@
 
             if ($dropdown.hasClass('js-filter-bulk-update')) {
               _this.enableBulkLabelDropdown();
-              _this.setDropdownData($dropdown, isMarking, this.id(label));
+              _this.setDropdownData($dropdown, isMarking, label.id);
               return;
             }
 
diff --git a/app/assets/javascripts/lib/utils/accessor.js b/app/assets/javascripts/lib/utils/accessor.js
new file mode 100644
index 0000000000000..1d18992af6325
--- /dev/null
+++ b/app/assets/javascripts/lib/utils/accessor.js
@@ -0,0 +1,47 @@
+function isPropertyAccessSafe(base, property) {
+  let safe;
+
+  try {
+    safe = !!base[property];
+  } catch (error) {
+    safe = false;
+  }
+
+  return safe;
+}
+
+function isFunctionCallSafe(base, functionName, ...args) {
+  let safe = true;
+
+  try {
+    base[functionName](...args);
+  } catch (error) {
+    safe = false;
+  }
+
+  return safe;
+}
+
+function isLocalStorageAccessSafe() {
+  let safe;
+
+  const TEST_KEY = 'isLocalStorageAccessSafe';
+  const TEST_VALUE = 'true';
+
+  safe = isPropertyAccessSafe(window, 'localStorage');
+  if (!safe) return safe;
+
+  safe = isFunctionCallSafe(window.localStorage, 'setItem', TEST_KEY, TEST_VALUE);
+
+  if (safe) window.localStorage.removeItem(TEST_KEY);
+
+  return safe;
+}
+
+const AccessorUtilities = {
+  isPropertyAccessSafe,
+  isFunctionCallSafe,
+  isLocalStorageAccessSafe,
+};
+
+export default AccessorUtilities;
diff --git a/app/assets/javascripts/lib/utils/ajax_cache.js b/app/assets/javascripts/lib/utils/ajax_cache.js
new file mode 100644
index 0000000000000..d99eefb5089b2
--- /dev/null
+++ b/app/assets/javascripts/lib/utils/ajax_cache.js
@@ -0,0 +1,32 @@
+const AjaxCache = {
+  internalStorage: { },
+  get(endpoint) {
+    return this.internalStorage[endpoint];
+  },
+  hasData(endpoint) {
+    return Object.prototype.hasOwnProperty.call(this.internalStorage, endpoint);
+  },
+  purge(endpoint) {
+    delete this.internalStorage[endpoint];
+  },
+  retrieve(endpoint) {
+    if (AjaxCache.hasData(endpoint)) {
+      return Promise.resolve(AjaxCache.get(endpoint));
+    }
+
+    return new Promise((resolve, reject) => {
+      $.ajax(endpoint) // eslint-disable-line promise/catch-or-return
+      .then(data => resolve(data),
+        (jqXHR, textStatus, errorThrown) => {
+          const error = new Error(`${endpoint}: ${errorThrown}`);
+          error.textStatus = textStatus;
+          reject(error);
+        },
+      );
+    })
+    .then((data) => { this.internalStorage[endpoint] = data; })
+    .then(() => AjaxCache.get(endpoint));
+  },
+};
+
+export default AjaxCache;
diff --git a/app/assets/javascripts/lib/utils/common_utils.js b/app/assets/javascripts/lib/utils/common_utils.js
index 8058672eaa90e..2f682fbd2fbf8 100644
--- a/app/assets/javascripts/lib/utils/common_utils.js
+++ b/app/assets/javascripts/lib/utils/common_utils.js
@@ -35,6 +35,14 @@
       });
     };
 
+    w.gl.utils.ajaxPost = function(url, data) {
+      return $.ajax({
+        type: 'POST',
+        url: url,
+        data: data,
+      });
+    };
+
     w.gl.utils.extractLast = function(term) {
       return this.split(term).pop();
     };
diff --git a/app/assets/javascripts/locale/de/app.js b/app/assets/javascripts/locale/de/app.js
new file mode 100644
index 0000000000000..e96090da80e41
--- /dev/null
+++ b/app/assets/javascripts/locale/de/app.js
@@ -0,0 +1 @@
+var locales = locales || {}; locales['de'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:37-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"German","Language":"de","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"de","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Deployed Jobs":[""],"Related Issues":[""],"Related Jobs":[""],"Related Merge Requests":[""],"Related Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/en/app.js b/app/assets/javascripts/locale/en/app.js
new file mode 100644
index 0000000000000..ade9b667b3c0f
--- /dev/null
+++ b/app/assets/javascripts/locale/en/app.js
@@ -0,0 +1 @@
+var locales = locales || {}; locales['en'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-04-12 22:36-0500","Last-Translator":"FULL NAME <EMAIL@ADDRESS>","Language-Team":"English","Language":"en","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","lang":"en","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":[""],"Commit":["",""],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":[""],"CycleAnalyticsStage|Code":[""],"CycleAnalyticsStage|Issue":[""],"CycleAnalyticsStage|Plan":[""],"CycleAnalyticsStage|Production":[""],"CycleAnalyticsStage|Review":[""],"CycleAnalyticsStage|Staging":[""],"CycleAnalyticsStage|Test":[""],"Deploy":["",""],"FirstPushedBy|First":[""],"FirstPushedBy|pushed by":[""],"From issue creation until deploy to production":[""],"From merge request merge until deploy to production":[""],"Introducing Cycle Analytics":[""],"Last %d day":["",""],"Limited to showing %d event at most":["",""],"Median":[""],"New Issue":["",""],"Not available":[""],"Not enough data":[""],"OpenedNDaysAgo|Opened":[""],"Pipeline Health":[""],"ProjectLifecycle|Stage":[""],"Read more":[""],"Related Commits":[""],"Related Deployed Jobs":[""],"Related Issues":[""],"Related Jobs":[""],"Related Merge Requests":[""],"Related Merged Requests":[""],"Showing %d event":["",""],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":[""],"The collection of events added to the data gathered for that stage.":[""],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":[""],"The phase of the development lifecycle.":[""],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":[""],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":[""],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":[""],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":[""],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":[""],"The time taken by each data entry gathered by that stage.":[""],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":[""],"Time before an issue gets scheduled":[""],"Time before an issue starts implementation":[""],"Time between merge request creation and merge/close":[""],"Time until first merge request":[""],"Time|hr":["",""],"Time|min":["",""],"Time|s":[""],"Total Time":[""],"Total test time for all commits/merges":[""],"Want to see the data? Please ask an administrator for access.":[""],"We don't have enough data to show this stage.":[""],"You need permission.":[""],"day":["",""]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/es/app.js b/app/assets/javascripts/locale/es/app.js
new file mode 100644
index 0000000000000..3dafa21f2359a
--- /dev/null
+++ b/app/assets/javascripts/locale/es/app.js
@@ -0,0 +1 @@
+var locales = locales || {}; locales['es'] = {"domain":"app","locale_data":{"app":{"":{"Project-Id-Version":"gitlab 1.0.0","Report-Msgid-Bugs-To":"","PO-Revision-Date":"2017-05-04 19:24-0500","Language-Team":"Spanish","Language":"es","MIME-Version":"1.0","Content-Type":"text/plain; charset=UTF-8","Content-Transfer-Encoding":"8bit","Plural-Forms":"nplurals=2; plural=n != 1;","Last-Translator":"","X-Generator":"Poedit 2.0.1","lang":"es","domain":"app","plural_forms":"nplurals=2; plural=n != 1;"},"ByAuthor|by":["por"],"Commit":["Cambio","Cambios"],"Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.":["Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."],"CycleAnalyticsStage|Code":["Código"],"CycleAnalyticsStage|Issue":["Incidencia"],"CycleAnalyticsStage|Plan":["Planificación"],"CycleAnalyticsStage|Production":["Producción"],"CycleAnalyticsStage|Review":["Revisión"],"CycleAnalyticsStage|Staging":["Puesta en escena"],"CycleAnalyticsStage|Test":["Pruebas"],"Deploy":["Despliegue","Despliegues"],"FirstPushedBy|First":["Primer"],"FirstPushedBy|pushed by":["enviado por"],"From issue creation until deploy to production":["Desde la creación de la incidencia hasta el despliegue a producción"],"From merge request merge until deploy to production":["Desde la integración de la solicitud de fusión hasta el despliegue a producción"],"Introducing Cycle Analytics":["Introducción a Cycle Analytics"],"Last %d day":["Último %d día","Últimos %d días"],"Limited to showing %d event at most":["Limitado a mostrar máximo %d evento","Limitado a mostrar máximo %d eventos"],"Median":["Mediana"],"New Issue":["Nueva incidencia","Nuevas incidencias"],"Not available":["No disponible"],"Not enough data":["No hay suficientes datos"],"OpenedNDaysAgo|Opened":["Abierto"],"Pipeline Health":["Estado del Pipeline"],"ProjectLifecycle|Stage":["Etapa"],"Read more":["Leer más"],"Related Commits":["Cambios Relacionados"],"Related Deployed Jobs":["Trabajos Desplegados Relacionados"],"Related Issues":["Incidencias Relacionadas"],"Related Jobs":["Trabajos Relacionados"],"Related Merge Requests":["Solicitudes de fusión Relacionadas"],"Related Merged Requests":["Solicitudes de fusión Relacionadas"],"Showing %d event":["Mostrando %d evento","Mostrando %d eventos"],"The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request.":["La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."],"The collection of events added to the data gathered for that stage.":["La colección de eventos agregados a los datos recopilados para esa etapa."],"The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage.":["La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."],"The phase of the development lifecycle.":["La etapa del ciclo de vida de desarrollo."],"The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit.":["La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe el primer cambio."],"The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle.":["La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."],"The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request.":["La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."],"The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time.":["La etapa de puesta en escena muestra el tiempo entre la fusión y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."],"The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running.":["La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."],"The time taken by each data entry gathered by that stage.":["El tiempo utilizado por cada entrada de datos obtenido por esa etapa."],"The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.":["El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."],"Time before an issue gets scheduled":["Tiempo antes de que una incidencia sea programada"],"Time before an issue starts implementation":["Tiempo antes de que empieze la implementación de una incidencia"],"Time between merge request creation and merge/close":["Tiempo entre la creación de la solicitud de fusión y la integración o cierre de ésta"],"Time until first merge request":["Tiempo hasta la primera solicitud de fusión"],"Time|hr":["hr","hrs"],"Time|min":["min","mins"],"Time|s":["s"],"Total Time":["Tiempo Total"],"Total test time for all commits/merges":["Tiempo total de pruebas para todos los cambios o integraciones"],"Want to see the data? Please ask an administrator for access.":["¿Quieres ver los datos? Por favor pide acceso al administrador."],"We don't have enough data to show this stage.":["No hay suficientes datos para mostrar en esta etapa."],"You need permission.":["Necesitas permisos."],"day":["día","días"]}}};
\ No newline at end of file
diff --git a/app/assets/javascripts/locale/index.js b/app/assets/javascripts/locale/index.js
new file mode 100644
index 0000000000000..7ba676d6d20aa
--- /dev/null
+++ b/app/assets/javascripts/locale/index.js
@@ -0,0 +1,70 @@
+import Jed from 'jed';
+
+/**
+  This is required to require all the translation folders in the current directory
+  this saves us having to do this manually & keep up to date with new languages
+**/
+function requireAll(requireContext) { return requireContext.keys().map(requireContext); }
+
+const allLocales = requireAll(require.context('./', true, /^(?!.*(?:index.js$)).*\.js$/));
+const locales = allLocales.reduce((d, obj) => {
+  const data = d;
+  const localeKey = Object.keys(obj)[0];
+
+  data[localeKey] = obj[localeKey];
+
+  return data;
+}, {});
+
+let lang = document.querySelector('html').getAttribute('lang') || 'en';
+lang = lang.replace(/-/g, '_');
+
+const locale = new Jed(locales[lang]);
+
+/**
+  Translates `text`
+
+  @param text The text to be translated
+  @returns {String} The translated text
+**/
+const gettext = locale.gettext.bind(locale);
+
+/**
+  Translate the text with a number
+  if the number is more than 1 it will use the `pluralText` translation.
+  This method allows for contexts, see below re. contexts
+
+  @param text Singular text to translate (eg. '%d day')
+  @param pluralText Plural text to translate (eg. '%d days')
+  @param count Number to decide which translation to use (eg. 2)
+  @returns {String} Translated text with the number replaced (eg. '2 days')
+**/
+const ngettext = (text, pluralText, count) => {
+  const translated = locale.ngettext(text, pluralText, count).replace(/%d/g, count).split('|');
+
+  return translated[translated.length - 1];
+};
+
+/**
+  Translate context based text
+  Either pass in the context translation like `Context|Text to translate`
+  or allow for dynamic text by doing passing in the context first & then the text to translate
+
+  @param keyOrContext Can be either the key to translate including the context
+                      (eg. 'Context|Text') or just the context for the translation
+                      (eg. 'Context')
+  @param key Is the dynamic variable you want to be translated
+  @returns {String} Translated context based text
+**/
+const pgettext = (keyOrContext, key) => {
+  const normalizedKey = key ? `${keyOrContext}|${key}` : keyOrContext;
+  const translated = gettext(normalizedKey).split('|');
+
+  return translated[translated.length - 1];
+};
+
+export { lang };
+export { gettext as __ };
+export { ngettext as n__ };
+export { pgettext as s__ };
+export default locale;
diff --git a/app/assets/javascripts/main.js b/app/assets/javascripts/main.js
index 5f96d63802399..8e8123f72a932 100644
--- a/app/assets/javascripts/main.js
+++ b/app/assets/javascripts/main.js
@@ -158,7 +158,6 @@ import './single_file_diff';
 import './smart_interval';
 import './snippets_list';
 import './star';
-import './subbable_resource';
 import './subscription';
 import './subscription_select';
 import './syntax_highlight';
diff --git a/app/assets/javascripts/members.js b/app/assets/javascripts/members.js
index f7d809dd5f4f8..7dec27b2c6cfe 100644
--- a/app/assets/javascripts/members.js
+++ b/app/assets/javascripts/members.js
@@ -43,7 +43,9 @@
 
             return $el.text();
           },
-          clicked: (selected, $link) => {
+          clicked: (options) => {
+            const $link = options.$el;
+
             if (!$link.data('revert')) {
               this.formSubmit(null, $link);
             } else {
diff --git a/app/assets/javascripts/merge_request_widget.js b/app/assets/javascripts/merge_request_widget.js
index 0742ca79fd50b..df80c1458c44c 100644
--- a/app/assets/javascripts/merge_request_widget.js
+++ b/app/assets/javascripts/merge_request_widget.js
@@ -313,7 +313,7 @@ import MiniPipelineGraph from './mini_pipeline_graph_dropdown';
 
     MergeRequestWidget.prototype.updateCommitUrls = function(id) {
       const commitsUrl = this.opts.commits_path;
-      $('.js-commit-link').text(`#${id}`).attr('href', [commitsUrl, id].join('/'));
+      $('.js-commit-link').text(id).attr('href', [commitsUrl, id].join('/'));
     };
 
     MergeRequestWidget.prototype.initMiniPipelineGraph = function() {
diff --git a/app/assets/javascripts/milestone_select.js b/app/assets/javascripts/milestone_select.js
index 29ce1df2dee61..d43cd84bbd682 100644
--- a/app/assets/javascripts/milestone_select.js
+++ b/app/assets/javascripts/milestone_select.js
@@ -142,7 +142,10 @@
 
             return true;
           },
-          clicked: function(selected, $el, e) {
+          clicked: function(options) {
+            const { $el, e } = options;
+            let selected = options.selectedObj;
+
             var data, isIssueIndex, isMRIndex, page, boardsStore;
             if (!selected) return;
             page = $('body').data('page');
diff --git a/app/assets/javascripts/namespace_select.js b/app/assets/javascripts/namespace_select.js
index b98e6121967a7..36bc1257cefd2 100644
--- a/app/assets/javascripts/namespace_select.js
+++ b/app/assets/javascripts/namespace_select.js
@@ -58,7 +58,8 @@
       });
     }
 
-    NamespaceSelect.prototype.onSelectItem = function(item, el, e) {
+    NamespaceSelect.prototype.onSelectItem = function(options) {
+      const { e } = options;
       return e.preventDefault();
     };
 
diff --git a/app/assets/javascripts/new_branch_form.js b/app/assets/javascripts/new_branch_form.js
index 5828f460a235a..67046d52a653c 100644
--- a/app/assets/javascripts/new_branch_form.js
+++ b/app/assets/javascripts/new_branch_form.js
@@ -34,6 +34,7 @@
         filterByText: true,
         remote: false,
         fieldName: $branchSelect.data('field-name'),
+        filterInput: 'input[type="search"]',
         selectable: true,
         isSelectable: function(branch, $el) {
           return !$el.hasClass('is-active');
@@ -50,6 +51,21 @@
           }
         }
       });
+
+      const $dropdownContainer = $branchSelect.closest('.dropdown');
+      const $fieldInput = $(`input[name="${$branchSelect.data('field-name')}"]`, $dropdownContainer);
+      const $filterInput = $('input[type="search"]', $dropdownContainer);
+
+      $filterInput.on('keyup', (e) => {
+        const keyCode = e.keyCode || e.which;
+        if (keyCode !== 13) return;
+
+        const text = $filterInput.val();
+        $fieldInput.val(text);
+        $('.dropdown-toggle-text', $branchSelect).text(text);
+
+        $dropdownContainer.removeClass('open');
+      });
     };
 
     NewBranchForm.prototype.setupRestrictions = function() {
diff --git a/app/assets/javascripts/notes.js b/app/assets/javascripts/notes.js
index 87f03a40eba3f..55391ebc089a2 100644
--- a/app/assets/javascripts/notes.js
+++ b/app/assets/javascripts/notes.js
@@ -26,12 +26,13 @@ const normalizeNewlines = function(str) {
 
   this.Notes = (function() {
     const MAX_VISIBLE_COMMIT_LIST_COUNT = 3;
+    const REGEX_SLASH_COMMANDS = /\/\w+/g;
 
     Notes.interval = null;
 
     function Notes(notes_url, note_ids, last_fetched_at, view) {
       this.updateTargetButtons = bind(this.updateTargetButtons, this);
-      this.updateCloseButton = bind(this.updateCloseButton, this);
+      this.updateComment = bind(this.updateComment, this);
       this.visibilityChange = bind(this.visibilityChange, this);
       this.cancelDiscussionForm = bind(this.cancelDiscussionForm, this);
       this.addDiffNote = bind(this.addDiffNote, this);
@@ -47,6 +48,7 @@ const normalizeNewlines = function(str) {
       this.refresh = bind(this.refresh, this);
       this.keydownNoteText = bind(this.keydownNoteText, this);
       this.toggleCommitList = bind(this.toggleCommitList, this);
+      this.postComment = bind(this.postComment, this);
 
       this.notes_url = notes_url;
       this.note_ids = note_ids;
@@ -82,28 +84,19 @@ const normalizeNewlines = function(str) {
     };
 
     Notes.prototype.addBinding = function() {
-      // add note to UI after creation
-      $(document).on("ajax:success", ".js-main-target-form", this.addNote);
-      $(document).on("ajax:success", ".js-discussion-note-form", this.addDiscussionNote);
-      // catch note ajax errors
-      $(document).on("ajax:error", ".js-main-target-form", this.addNoteError);
-      // change note in UI after update
-      $(document).on("ajax:success", "form.edit-note", this.updateNote);
       // Edit note link
       $(document).on("click", ".js-note-edit", this.showEditForm.bind(this));
       $(document).on("click", ".note-edit-cancel", this.cancelEdit);
       // Reopen and close actions for Issue/MR combined with note form submit
-      $(document).on("click", ".js-comment-button", this.updateCloseButton);
+      $(document).on("click", ".js-comment-submit-button", this.postComment);
+      $(document).on("click", ".js-comment-save-button", this.updateComment);
       $(document).on("keyup input", ".js-note-text", this.updateTargetButtons);
       // resolve a discussion
-      $(document).on('click', '.js-comment-resolve-button', this.resolveDiscussion);
+      $(document).on('click', '.js-comment-resolve-button', this.postComment);
       // remove a note (in general)
       $(document).on("click", ".js-note-delete", this.removeNote);
       // delete note attachment
       $(document).on("click", ".js-note-attachment-delete", this.removeAttachment);
-      // reset main target form after submit
-      $(document).on("ajax:complete", ".js-main-target-form", this.reenableTargetFormSubmitButton);
-      $(document).on("ajax:success", ".js-main-target-form", this.resetMainTargetForm);
       // reset main target form when clicking discard
       $(document).on("click", ".js-note-discard", this.resetMainTargetForm);
       // update the file name when an attachment is selected
@@ -120,20 +113,20 @@ const normalizeNewlines = function(str) {
       $(document).on("visibilitychange", this.visibilityChange);
       // when issue status changes, we need to refresh data
       $(document).on("issuable:change", this.refresh);
+      // ajax:events that happen on Form when actions like Reopen, Close are performed on Issues and MRs.
+      $(document).on("ajax:success", ".js-main-target-form", this.addNote);
+      $(document).on("ajax:success", ".js-discussion-note-form", this.addDiscussionNote);
+      $(document).on("ajax:success", ".js-main-target-form", this.resetMainTargetForm);
+      $(document).on("ajax:complete", ".js-main-target-form", this.reenableTargetFormSubmitButton);
       // when a key is clicked on the notes
       return $(document).on("keydown", ".js-note-text", this.keydownNoteText);
     };
 
     Notes.prototype.cleanBinding = function() {
-      $(document).off("ajax:success", ".js-main-target-form");
-      $(document).off("ajax:success", ".js-discussion-note-form");
-      $(document).off("ajax:success", "form.edit-note");
       $(document).off("click", ".js-note-edit");
       $(document).off("click", ".note-edit-cancel");
       $(document).off("click", ".js-note-delete");
       $(document).off("click", ".js-note-attachment-delete");
-      $(document).off("ajax:complete", ".js-main-target-form");
-      $(document).off("ajax:success", ".js-main-target-form");
       $(document).off("click", ".js-discussion-reply-button");
       $(document).off("click", ".js-add-diff-note-button");
       $(document).off("visibilitychange");
@@ -144,6 +137,9 @@ const normalizeNewlines = function(str) {
       $(document).off("keydown", ".js-note-text");
       $(document).off('click', '.js-comment-resolve-button');
       $(document).off("click", '.system-note-commit-list-toggler');
+      $(document).off("ajax:success", ".js-main-target-form");
+      $(document).off("ajax:success", ".js-discussion-note-form");
+      $(document).off("ajax:complete", ".js-main-target-form");
     };
 
     Notes.initCommentTypeToggle = function (form) {
@@ -276,12 +272,8 @@ const normalizeNewlines = function(str) {
       return this.initRefresh();
     };
 
-    Notes.prototype.handleCreateChanges = function(noteEntity) {
+    Notes.prototype.handleSlashCommands = function(noteEntity) {
       var votesBlock;
-      if (typeof noteEntity === 'undefined') {
-        return;
-      }
-
       if (noteEntity.commands_changes) {
         if ('merge' in noteEntity.commands_changes) {
           $.get(mrRefreshWidgetUrl);
@@ -556,24 +548,29 @@ const normalizeNewlines = function(str) {
     Adds new note to list.
      */
 
-    Notes.prototype.addNote = function(xhr, note, status) {
-      this.handleCreateChanges(note);
+    Notes.prototype.addNote = function($form, note) {
       return this.renderNote(note);
     };
 
-    Notes.prototype.addNoteError = function(xhr, note, status) {
-      return new Flash('Your comment could not be submitted! Please check your network connection and try again.', 'alert', this.parentTimeline);
+    Notes.prototype.addNoteError = ($form) => {
+      let formParentTimeline;
+      if ($form.hasClass('js-main-target-form')) {
+        formParentTimeline = $form.parents('.timeline');
+      } else if ($form.hasClass('js-discussion-note-form')) {
+        formParentTimeline = $form.closest('.discussion-notes').find('.notes');
+      }
+      return new Flash('Your comment could not be submitted! Please check your network connection and try again.', 'alert', formParentTimeline);
     };
 
+    Notes.prototype.updateNoteError = $parentTimeline => new Flash('Your comment could not be updated! Please check your network connection and try again.');
+
     /*
     Called in response to the new note form being submitted
 
     Adds new note to list.
      */
 
-    Notes.prototype.addDiscussionNote = function(xhr, note, status) {
-      var $form = $(xhr.target);
-
+    Notes.prototype.addDiscussionNote = function($form, note, isNewDiffComment) {
       if ($form.attr('data-resolve-all') != null) {
         var projectPath = $form.data('project-path');
         var discussionId = $form.data('discussion-id');
@@ -586,7 +583,9 @@ const normalizeNewlines = function(str) {
 
       this.renderNote(note, $form);
       // cleanup after successfully creating a diff/discussion note
-      this.removeDiscussionNoteForm($form);
+      if (isNewDiffComment) {
+        this.removeDiscussionNoteForm($form);
+      }
     };
 
     /*
@@ -596,17 +595,18 @@ const normalizeNewlines = function(str) {
      */
 
     Notes.prototype.updateNote = function(_xhr, noteEntity, _status) {
-      var $html, $note_li;
+      var $noteEntityEl, $note_li;
       // Convert returned HTML to a jQuery object so we can modify it further
-      $html = $(noteEntity.html);
+      $noteEntityEl = $(noteEntity.html);
+      $noteEntityEl.addClass('fade-in-full');
       this.revertNoteEditForm();
-      gl.utils.localTimeAgo($('.js-timeago', $html));
-      $html.renderGFM();
-      $html.find('.js-task-list-container').taskList('enable');
+      gl.utils.localTimeAgo($('.js-timeago', $noteEntityEl));
+      $noteEntityEl.renderGFM();
+      $noteEntityEl.find('.js-task-list-container').taskList('enable');
       // Find the note's `li` element by ID and replace it with the updated HTML
       $note_li = $('.note-row-' + noteEntity.id);
 
-      $note_li.replaceWith($html);
+      $note_li.replaceWith($noteEntityEl);
 
       if (typeof gl.diffNotesCompileComponents !== 'undefined') {
         gl.diffNotesCompileComponents();
@@ -698,7 +698,7 @@ const normalizeNewlines = function(str) {
       var $editForm = $(selector);
 
       $editForm.insertBefore('.notes-form');
-      $editForm.find('.js-comment-button').enable();
+      $editForm.find('.js-comment-save-button').enable();
       $editForm.find('.js-finish-edit-warning').hide();
     };
 
@@ -982,14 +982,6 @@ const normalizeNewlines = function(str) {
       return this.refresh();
     };
 
-    Notes.prototype.updateCloseButton = function(e) {
-      var closebtn, form, textarea;
-      textarea = $(e.target);
-      form = textarea.parents('form');
-      closebtn = form.find('.js-note-target-close');
-      return closebtn.text(closebtn.data('original-text'));
-    };
-
     Notes.prototype.updateTargetButtons = function(e) {
       var closebtn, closetext, discardbtn, form, reopenbtn, reopentext, textarea;
       textarea = $(e.target);
@@ -1078,17 +1070,6 @@ const normalizeNewlines = function(str) {
       return this.notesCountBadge.text(parseInt(this.notesCountBadge.text(), 10) + updateCount);
     };
 
-    Notes.prototype.resolveDiscussion = function() {
-      var $this = $(this);
-      var discussionId = $this.attr('data-discussion-id');
-
-      $this
-        .closest('form')
-        .attr('data-discussion-id', discussionId)
-        .attr('data-resolve-all', 'true')
-        .attr('data-project-path', $this.attr('data-project-path'));
-    };
-
     Notes.prototype.toggleCommitList = function(e) {
       const $element = $(e.currentTarget);
       const $closestSystemCommitList = $element.siblings('.system-note-commit-list');
@@ -1137,7 +1118,7 @@ const normalizeNewlines = function(str) {
     Notes.animateAppendNote = function(noteHtml, $notesList) {
       const $note = $(noteHtml);
 
-      $note.addClass('fade-in').renderGFM();
+      $note.addClass('fade-in-full').renderGFM();
       $notesList.append($note);
       return $note;
     };
@@ -1150,6 +1131,254 @@ const normalizeNewlines = function(str) {
       return $updatedNote;
     };
 
+    /**
+     * Get data from Form attributes to use for saving/submitting comment.
+     */
+    Notes.prototype.getFormData = function($form) {
+      return {
+        formData: $form.serialize(),
+        formContent: $form.find('.js-note-text').val(),
+        formAction: $form.attr('action'),
+      };
+    };
+
+    /**
+     * Identify if comment has any slash commands
+     */
+    Notes.prototype.hasSlashCommands = function(formContent) {
+      return REGEX_SLASH_COMMANDS.test(formContent);
+    };
+
+    /**
+     * Remove slash commands and leave comment with pure message
+     */
+    Notes.prototype.stripSlashCommands = function(formContent) {
+      return formContent.replace(REGEX_SLASH_COMMANDS, '').trim();
+    };
+
+    /**
+     * Create placeholder note DOM element populated with comment body
+     * that we will show while comment is being posted.
+     * Once comment is _actually_ posted on server, we will have final element
+     * in response that we will show in place of this temporary element.
+     */
+    Notes.prototype.createPlaceholderNote = function({ formContent, uniqueId, isDiscussionNote, currentUsername, currentUserFullname }) {
+      const discussionClass = isDiscussionNote ? 'discussion' : '';
+      const $tempNote = $(
+        `<li id="${uniqueId}" class="note being-posted fade-in-half timeline-entry">
+           <div class="timeline-entry-inner">
+              <div class="timeline-icon">
+                 <a href="/${currentUsername}"><span class="dummy-avatar"></span></a>
+              </div>
+              <div class="timeline-content ${discussionClass}">
+                 <div class="note-header">
+                    <div class="note-header-info">
+                       <a href="/${currentUsername}">
+                         <span class="hidden-xs">${currentUserFullname}</span>
+                         <span class="note-headline-light">@${currentUsername}</span>
+                       </a>
+                       <span class="note-headline-light">
+                          <i class="fa fa-spinner fa-spin" aria-label="Comment is being posted" aria-hidden="true"></i>
+                       </span>
+                    </div>
+                 </div>
+                 <div class="note-body">
+                   <div class="note-text">
+                     <p>${formContent}</p>
+                   </div>
+                 </div>
+              </div>
+           </div>
+        </li>`
+      );
+
+      return $tempNote;
+    };
+
+    /**
+     * This method does following tasks step-by-step whenever a new comment
+     * is submitted by user (both main thread comments as well as discussion comments).
+     *
+     * 1) Get Form metadata
+     * 2) Identify comment type; a) Main thread b) Discussion thread c) Discussion resolve
+     * 3) Build temporary placeholder element (using `createPlaceholderNote`)
+     * 4) Show placeholder note on UI
+     * 5) Perform network request to submit the note using `gl.utils.ajaxPost`
+     *    a) If request is successfully completed
+     *        1. Remove placeholder element
+     *        2. Show submitted Note element
+     *        3. Perform post-submit errands
+     *           a. Mark discussion as resolved if comment submission was for resolve.
+     *           b. Reset comment form to original state.
+     *    b) If request failed
+     *        1. Remove placeholder element
+     *        2. Show error Flash message about failure
+     */
+    Notes.prototype.postComment = function(e) {
+      e.preventDefault();
+
+      // Get Form metadata
+      const $submitBtn = $(e.target);
+      let $form = $submitBtn.parents('form');
+      const $closeBtn = $form.find('.js-note-target-close');
+      const isDiscussionNote = $submitBtn.parent().find('li.droplab-item-selected').attr('id') === 'discussion';
+      const isMainForm = $form.hasClass('js-main-target-form');
+      const isDiscussionForm = $form.hasClass('js-discussion-note-form');
+      const isDiscussionResolve = $submitBtn.hasClass('js-comment-resolve-button');
+      const { formData, formContent, formAction } = this.getFormData($form);
+      const uniqueId = _.uniqueId('tempNote_');
+      let $notesContainer;
+      let tempFormContent;
+
+      // Get reference to notes container based on type of comment
+      if (isDiscussionForm) {
+        $notesContainer = $form.parent('.discussion-notes').find('.notes');
+      } else if (isMainForm) {
+        $notesContainer = $('ul.main-notes-list');
+      }
+
+      // If comment is to resolve discussion, disable submit buttons while
+      // comment posting is finished.
+      if (isDiscussionResolve) {
+        $submitBtn.disable();
+        $form.find('.js-comment-submit-button').disable();
+      }
+
+      tempFormContent = formContent;
+      if (this.hasSlashCommands(formContent)) {
+        tempFormContent = this.stripSlashCommands(formContent);
+      }
+
+      if (tempFormContent) {
+        // Show placeholder note
+        $notesContainer.append(this.createPlaceholderNote({
+          formContent: tempFormContent,
+          uniqueId,
+          isDiscussionNote,
+          currentUsername: gon.current_username,
+          currentUserFullname: gon.current_user_fullname,
+        }));
+      }
+
+      // Clear the form textarea
+      if ($notesContainer.length) {
+        if (isMainForm) {
+          this.resetMainTargetForm(e);
+        } else if (isDiscussionForm) {
+          this.removeDiscussionNoteForm($form);
+        }
+      }
+
+      /* eslint-disable promise/catch-or-return */
+      // Make request to submit comment on server
+      gl.utils.ajaxPost(formAction, formData)
+        .then((note) => {
+          // Submission successful! remove placeholder
+          $notesContainer.find(`#${uniqueId}`).remove();
+
+          // Check if this was discussion comment
+          if (isDiscussionForm) {
+            // Remove flash-container
+            $notesContainer.find('.flash-container').remove();
+
+            // If comment intends to resolve discussion, do the same.
+            if (isDiscussionResolve) {
+              $form
+                .attr('data-discussion-id', $submitBtn.data('discussion-id'))
+                .attr('data-resolve-all', 'true')
+                .attr('data-project-path', $submitBtn.data('project-path'));
+            }
+
+            // Show final note element on UI
+            this.addDiscussionNote($form, note, $notesContainer.length === 0);
+
+            // append flash-container to the Notes list
+            if ($notesContainer.length) {
+              $notesContainer.append('<div class="flash-container" style="display: none;"></div>');
+            }
+          } else if (isMainForm) { // Check if this was main thread comment
+            // Show final note element on UI and perform form and action buttons cleanup
+            this.addNote($form, note);
+            this.reenableTargetFormSubmitButton(e);
+          }
+
+          if (note.commands_changes) {
+            this.handleSlashCommands(note);
+          }
+
+          $form.trigger('ajax:success', [note]);
+        }).fail(() => {
+          // Submission failed, remove placeholder note and show Flash error message
+          $notesContainer.find(`#${uniqueId}`).remove();
+
+          // Show form again on UI on failure
+          if (isDiscussionForm && $notesContainer.length) {
+            const replyButton = $notesContainer.parent().find('.js-discussion-reply-button');
+            $.proxy(this.replyToDiscussionNote, replyButton[0], { target: replyButton[0] }).call();
+            $form = $notesContainer.parent().find('form');
+          }
+
+          $form.find('.js-note-text').val(formContent);
+          this.reenableTargetFormSubmitButton(e);
+          this.addNoteError($form);
+        });
+
+      return $closeBtn.text($closeBtn.data('original-text'));
+    };
+
+    /**
+     * This method does following tasks step-by-step whenever an existing comment
+     * is updated by user (both main thread comments as well as discussion comments).
+     *
+     * 1) Get Form metadata
+     * 2) Update note element with new content
+     * 3) Perform network request to submit the updated note using `gl.utils.ajaxPost`
+     *    a) If request is successfully completed
+     *        1. Show submitted Note element
+     *    b) If request failed
+     *        1. Revert Note element to original content
+     *        2. Show error Flash message about failure
+     */
+    Notes.prototype.updateComment = function(e) {
+      e.preventDefault();
+
+      // Get Form metadata
+      const $submitBtn = $(e.target);
+      const $form = $submitBtn.parents('form');
+      const $closeBtn = $form.find('.js-note-target-close');
+      const $editingNote = $form.parents('.note.is-editing');
+      const $noteBody = $editingNote.find('.js-task-list-container');
+      const $noteBodyText = $noteBody.find('.note-text');
+      const { formData, formContent, formAction } = this.getFormData($form);
+
+      // Cache original comment content
+      const cachedNoteBodyText = $noteBodyText.html();
+
+      // Show updated comment content temporarily
+      $noteBodyText.html(formContent);
+      $editingNote.removeClass('is-editing fade-in-full').addClass('being-posted fade-in-half');
+      $editingNote.find('.note-headline-meta a').html('<i class="fa fa-spinner fa-spin" aria-label="Comment is being updated" aria-hidden="true"></i>');
+
+      /* eslint-disable promise/catch-or-return */
+      // Make request to update comment on server
+      gl.utils.ajaxPost(formAction, formData)
+        .then((note) => {
+          // Submission successful! render final note element
+          this.updateNote(null, note, null);
+        })
+        .fail(() => {
+          // Submission failed, revert back to original note
+          $noteBodyText.html(cachedNoteBodyText);
+          $editingNote.removeClass('being-posted fade-in');
+          $editingNote.find('.fa.fa-spinner').remove();
+
+          // Show Flash message about failure
+          this.updateNoteError();
+        });
+
+      return $closeBtn.text($closeBtn.data('original-text'));
+    };
+
     return Notes;
   })();
 }).call(window);
diff --git a/app/assets/javascripts/pipelines/services/pipelines_service.js b/app/assets/javascripts/pipelines/services/pipelines_service.js
index 255cd513490e4..b21f84b454529 100644
--- a/app/assets/javascripts/pipelines/services/pipelines_service.js
+++ b/app/assets/javascripts/pipelines/services/pipelines_service.js
@@ -40,6 +40,6 @@ export default class PipelinesService {
    * @return {Promise}
    */
   postAction(endpoint) {
-    return Vue.http.post(endpoint, {}, { emulateJSON: true });
+    return Vue.http.post(`${endpoint}.json`);
   }
 }
diff --git a/app/assets/javascripts/project.js b/app/assets/javascripts/project.js
index 2a5934c36213a..23e807a28bbf6 100644
--- a/app/assets/javascripts/project.js
+++ b/app/assets/javascripts/project.js
@@ -119,7 +119,8 @@ import Cookies from 'js-cookie';
           toggleLabel: function(obj, $el) {
             return $el.text().trim();
           },
-          clicked: function(selected, $el, e) {
+          clicked: function(options) {
+            const { e } = options;
             e.preventDefault();
             if ($('input[name="ref"]').length) {
               var $form = $dropdown.closest('form');
diff --git a/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js b/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
index 59d5f00f05093..5d54c220aac43 100644
--- a/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
+++ b/app/assets/javascripts/protected_branches/protected_branch_access_dropdown.js
@@ -53,7 +53,10 @@
             onHide();
           }
         },
-        clicked(item, $el, e) {
+        clicked(opts) {
+          const { $el, e } = opts;
+          const item = opts.selectedObj;
+
           e.preventDefault();
 
           if ($el.is('.is-active')) {
diff --git a/app/assets/javascripts/protected_branches/protected_branch_dropdown.js b/app/assets/javascripts/protected_branches/protected_branch_dropdown.js
index 1d4bb8a13d677..bc6110fcd4e4f 100644
--- a/app/assets/javascripts/protected_branches/protected_branch_dropdown.js
+++ b/app/assets/javascripts/protected_branches/protected_branch_dropdown.js
@@ -35,7 +35,8 @@ class ProtectedBranchDropdown {
         return _.escape(protectedBranch.id);
       },
       onFilter: this.toggleCreateNewButton.bind(this),
-      clicked: (item, $el, e) => {
+      clicked: (options) => {
+        const { $el, e } = options;
         e.preventDefault();
         this.onSelect();
       }
diff --git a/app/assets/javascripts/protected_tags/protected_tag_access_dropdown.js b/app/assets/javascripts/protected_tags/protected_tag_access_dropdown.js
index fff83f3af3bb3..d4c9a91a74a42 100644
--- a/app/assets/javascripts/protected_tags/protected_tag_access_dropdown.js
+++ b/app/assets/javascripts/protected_tags/protected_tag_access_dropdown.js
@@ -17,8 +17,8 @@ export default class ProtectedTagAccessDropdown {
         }
         return 'Select';
       },
-      clicked(item, $el, e) {
-        e.preventDefault();
+      clicked(options) {
+        options.e.preventDefault();
         onSelect();
       },
     });
diff --git a/app/assets/javascripts/protected_tags/protected_tag_dropdown.js b/app/assets/javascripts/protected_tags/protected_tag_dropdown.js
index 5ff4e4432622b..068e9698e1d02 100644
--- a/app/assets/javascripts/protected_tags/protected_tag_dropdown.js
+++ b/app/assets/javascripts/protected_tags/protected_tag_dropdown.js
@@ -39,8 +39,8 @@ export default class ProtectedTagDropdown {
         return _.escape(protectedTag.id);
       },
       onFilter: this.toggleCreateNewButton.bind(this),
-      clicked: (item, $el, e) => {
-        e.preventDefault();
+      clicked: (options) => {
+        options.e.preventDefault();
         this.onSelect();
       },
     });
diff --git a/app/assets/javascripts/raven/index.js b/app/assets/javascripts/raven/index.js
new file mode 100644
index 0000000000000..5325e495815f7
--- /dev/null
+++ b/app/assets/javascripts/raven/index.js
@@ -0,0 +1,16 @@
+import RavenConfig from './raven_config';
+
+const index = function index() {
+  RavenConfig.init({
+    sentryDsn: gon.sentry_dsn,
+    currentUserId: gon.current_user_id,
+    whitelistUrls: [gon.gitlab_url],
+    isProduction: process.env.NODE_ENV,
+  });
+
+  return RavenConfig;
+};
+
+index();
+
+export default index;
diff --git a/app/assets/javascripts/raven/raven_config.js b/app/assets/javascripts/raven/raven_config.js
new file mode 100644
index 0000000000000..c7fe1cacf49ea
--- /dev/null
+++ b/app/assets/javascripts/raven/raven_config.js
@@ -0,0 +1,100 @@
+import Raven from 'raven-js';
+
+const IGNORE_ERRORS = [
+  // Random plugins/extensions
+  'top.GLOBALS',
+  // See: http://blog.errorception.com/2012/03/tale-of-unfindable-js-error. html
+  'originalCreateNotification',
+  'canvas.contentDocument',
+  'MyApp_RemoveAllHighlights',
+  'http://tt.epicplay.com',
+  'Can\'t find variable: ZiteReader',
+  'jigsaw is not defined',
+  'ComboSearch is not defined',
+  'http://loading.retry.widdit.com/',
+  'atomicFindClose',
+  // Facebook borked
+  'fb_xd_fragment',
+  // ISP "optimizing" proxy - `Cache-Control: no-transform` seems to
+  // reduce this. (thanks @acdha)
+  // See http://stackoverflow.com/questions/4113268
+  'bmi_SafeAddOnload',
+  'EBCallBackMessageReceived',
+  // See http://toolbar.conduit.com/Developer/HtmlAndGadget/Methods/JSInjection.aspx
+  'conduitPage',
+];
+
+const IGNORE_URLS = [
+  // Facebook flakiness
+  /graph\.facebook\.com/i,
+  // Facebook blocked
+  /connect\.facebook\.net\/en_US\/all\.js/i,
+  // Woopra flakiness
+  /eatdifferent\.com\.woopra-ns\.com/i,
+  /static\.woopra\.com\/js\/woopra\.js/i,
+  // Chrome extensions
+  /extensions\//i,
+  /^chrome:\/\//i,
+  // Other plugins
+  /127\.0\.0\.1:4001\/isrunning/i,  // Cacaoweb
+  /webappstoolbarba\.texthelp\.com\//i,
+  /metrics\.itunes\.apple\.com\.edgesuite\.net\//i,
+];
+
+const SAMPLE_RATE = 95;
+
+const RavenConfig = {
+  IGNORE_ERRORS,
+  IGNORE_URLS,
+  SAMPLE_RATE,
+  init(options = {}) {
+    this.options = options;
+
+    this.configure();
+    this.bindRavenErrors();
+    if (this.options.currentUserId) this.setUser();
+  },
+
+  configure() {
+    Raven.config(this.options.sentryDsn, {
+      whitelistUrls: this.options.whitelistUrls,
+      environment: this.options.isProduction ? 'production' : 'development',
+      ignoreErrors: this.IGNORE_ERRORS,
+      ignoreUrls: this.IGNORE_URLS,
+      shouldSendCallback: this.shouldSendSample.bind(this),
+    }).install();
+  },
+
+  setUser() {
+    Raven.setUserContext({
+      id: this.options.currentUserId,
+    });
+  },
+
+  bindRavenErrors() {
+    window.$(document).on('ajaxError.raven', this.handleRavenErrors);
+  },
+
+  handleRavenErrors(event, req, config, err) {
+    const error = err || req.statusText;
+    const responseText = req.responseText || 'Unknown response text';
+
+    Raven.captureMessage(error, {
+      extra: {
+        type: config.type,
+        url: config.url,
+        data: config.data,
+        status: req.status,
+        response: responseText,
+        error,
+        event,
+      },
+    });
+  },
+
+  shouldSendSample() {
+    return Math.random() * 100 <= this.SAMPLE_RATE;
+  },
+};
+
+export default RavenConfig;
diff --git a/app/assets/javascripts/sidebar/components/assignees/assignee_title.js b/app/assets/javascripts/sidebar/components/assignees/assignee_title.js
new file mode 100644
index 0000000000000..a9ad3708514b7
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/assignees/assignee_title.js
@@ -0,0 +1,41 @@
+export default {
+  name: 'AssigneeTitle',
+  props: {
+    loading: {
+      type: Boolean,
+      required: false,
+      default: false,
+    },
+    numberOfAssignees: {
+      type: Number,
+      required: true,
+    },
+    editable: {
+      type: Boolean,
+      required: true,
+    },
+  },
+  computed: {
+    assigneeTitle() {
+      const assignees = this.numberOfAssignees;
+      return assignees > 1 ? `${assignees} Assignees` : 'Assignee';
+    },
+  },
+  template: `
+    <div class="title hide-collapsed">
+      {{assigneeTitle}}
+      <i
+        v-if="loading"
+        aria-hidden="true"
+        class="fa fa-spinner fa-spin block-loading"
+      />
+      <a
+        v-if="editable"
+        class="edit-link pull-right"
+        href="#"
+      >
+        Edit
+      </a>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/assignees/assignees.js b/app/assets/javascripts/sidebar/components/assignees/assignees.js
new file mode 100644
index 0000000000000..7e5feac622c7a
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/assignees/assignees.js
@@ -0,0 +1,224 @@
+export default {
+  name: 'Assignees',
+  data() {
+    return {
+      defaultRenderCount: 5,
+      defaultMaxCounter: 99,
+      showLess: true,
+    };
+  },
+  props: {
+    rootPath: {
+      type: String,
+      required: true,
+    },
+    users: {
+      type: Array,
+      required: true,
+    },
+    editable: {
+      type: Boolean,
+      required: true,
+    },
+  },
+  computed: {
+    firstUser() {
+      return this.users[0];
+    },
+    hasMoreThanTwoAssignees() {
+      return this.users.length > 2;
+    },
+    hasMoreThanOneAssignee() {
+      return this.users.length > 1;
+    },
+    hasAssignees() {
+      return this.users.length > 0;
+    },
+    hasNoUsers() {
+      return !this.users.length;
+    },
+    hasOneUser() {
+      return this.users.length === 1;
+    },
+    renderShowMoreSection() {
+      return this.users.length > this.defaultRenderCount;
+    },
+    numberOfHiddenAssignees() {
+      return this.users.length - this.defaultRenderCount;
+    },
+    isHiddenAssignees() {
+      return this.numberOfHiddenAssignees > 0;
+    },
+    hiddenAssigneesLabel() {
+      return `+ ${this.numberOfHiddenAssignees} more`;
+    },
+    collapsedTooltipTitle() {
+      const maxRender = Math.min(this.defaultRenderCount, this.users.length);
+      const renderUsers = this.users.slice(0, maxRender);
+      const names = renderUsers.map(u => u.name);
+
+      if (this.users.length > maxRender) {
+        names.push(`+ ${this.users.length - maxRender} more`);
+      }
+
+      return names.join(', ');
+    },
+    sidebarAvatarCounter() {
+      let counter = `+${this.users.length - 1}`;
+
+      if (this.users.length > this.defaultMaxCounter) {
+        counter = `${this.defaultMaxCounter}+`;
+      }
+
+      return counter;
+    },
+  },
+  methods: {
+    assignSelf() {
+      this.$emit('assign-self');
+    },
+    toggleShowLess() {
+      this.showLess = !this.showLess;
+    },
+    renderAssignee(index) {
+      return !this.showLess || (index < this.defaultRenderCount && this.showLess);
+    },
+    avatarUrl(user) {
+      return user.avatar || user.avatar_url;
+    },
+    assigneeUrl(user) {
+      return `${this.rootPath}${user.username}`;
+    },
+    assigneeAlt(user) {
+      return `${user.name}'s avatar`;
+    },
+    assigneeUsername(user) {
+      return `@${user.username}`;
+    },
+    shouldRenderCollapsedAssignee(index) {
+      const firstTwo = this.users.length <= 2 && index <= 2;
+
+      return index === 0 || firstTwo;
+    },
+  },
+  template: `
+    <div>
+      <div
+        class="sidebar-collapsed-icon sidebar-collapsed-user"
+        :class="{ 'multiple-users': hasMoreThanOneAssignee, 'has-tooltip': hasAssignees }"
+        data-container="body"
+        data-placement="left"
+        :title="collapsedTooltipTitle"
+      >
+        <i
+          v-if="hasNoUsers"
+          aria-label="No Assignee"
+          class="fa fa-user"
+        />
+        <button
+          type="button"
+          class="btn-link"
+          v-for="(user, index) in users"
+          v-if="shouldRenderCollapsedAssignee(index)"
+        >
+          <img
+            width="24"
+            class="avatar avatar-inline s24"
+            :alt="assigneeAlt(user)"
+            :src="avatarUrl(user)"
+          />
+          <span class="author">
+            {{ user.name }}
+          </span>
+        </button>
+        <button
+          v-if="hasMoreThanTwoAssignees"
+          class="btn-link"
+          type="button"
+        >
+          <span
+            class="avatar-counter sidebar-avatar-counter"
+          >
+            {{ sidebarAvatarCounter }}
+          </span>
+        </button>
+      </div>
+      <div class="value hide-collapsed">
+        <template v-if="hasNoUsers">
+          <span class="assign-yourself no-value">
+            No assignee
+            <template v-if="editable">
+             -
+              <button
+                type="button"
+                class="btn-link"
+                @click="assignSelf"
+              >
+                assign yourself
+              </button>
+            </template>
+          </span>
+        </template>
+        <template v-else-if="hasOneUser">
+          <a
+            class="author_link bold"
+            :href="assigneeUrl(firstUser)"
+          >
+            <img
+              width="32"
+              class="avatar avatar-inline s32"
+              :alt="assigneeAlt(firstUser)"
+              :src="avatarUrl(firstUser)"
+            />
+            <span class="author">
+              {{ firstUser.name }}
+            </span>
+            <span class="username">
+              {{ assigneeUsername(firstUser) }}
+            </span>
+          </a>
+        </template>
+        <template v-else>
+          <div class="user-list">
+            <div
+              class="user-item"
+              v-for="(user, index) in users"
+              v-if="renderAssignee(index)"
+            >
+              <a
+                class="user-link has-tooltip"
+                data-placement="bottom"
+                :href="assigneeUrl(user)"
+                :data-title="user.name"
+              >
+                <img
+                  width="32"
+                  class="avatar avatar-inline s32"
+                  :alt="assigneeAlt(user)"
+                  :src="avatarUrl(user)"
+                />
+              </a>
+            </div>
+          </div>
+          <div
+            v-if="renderShowMoreSection"
+            class="user-list-more"
+          >
+            <button
+              type="button"
+              class="btn-link"
+              @click="toggleShowLess"
+            >
+              <template v-if="showLess">
+                {{ hiddenAssigneesLabel }}
+              </template>
+              <template v-else>
+                - show less
+              </template>
+            </button>
+          </div>
+        </template>
+      </div>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/assignees/sidebar_assignees.js b/app/assets/javascripts/sidebar/components/assignees/sidebar_assignees.js
new file mode 100644
index 0000000000000..4ee7a9c619cee
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/assignees/sidebar_assignees.js
@@ -0,0 +1,82 @@
+/* global Flash */
+
+import AssigneeTitle from './assignee_title';
+import Assignees from './assignees';
+
+import Store from '../../stores/sidebar_store';
+import Mediator from '../../sidebar_mediator';
+
+import eventHub from '../../event_hub';
+
+export default {
+  name: 'SidebarAssignees',
+  data() {
+    return {
+      mediator: new Mediator(),
+      store: new Store(),
+      loading: false,
+      field: '',
+    };
+  },
+  components: {
+    'assignee-title': AssigneeTitle,
+    assignees: Assignees,
+  },
+  methods: {
+    assignSelf() {
+      // Notify gl dropdown that we are now assigning to current user
+      this.$el.parentElement.dispatchEvent(new Event('assignYourself'));
+
+      this.mediator.assignYourself();
+      this.saveAssignees();
+    },
+    saveAssignees() {
+      this.loading = true;
+
+      this.mediator.saveAssignees(this.field)
+        .then(() => {
+          this.loading = false;
+        })
+        .catch(() => {
+          this.loading = false;
+          return new Flash('Error occurred when saving assignees');
+        });
+    },
+  },
+  created() {
+    this.removeAssignee = this.store.removeAssignee.bind(this.store);
+    this.addAssignee = this.store.addAssignee.bind(this.store);
+    this.removeAllAssignees = this.store.removeAllAssignees.bind(this.store);
+
+    // Get events from glDropdown
+    eventHub.$on('sidebar.removeAssignee', this.removeAssignee);
+    eventHub.$on('sidebar.addAssignee', this.addAssignee);
+    eventHub.$on('sidebar.removeAllAssignees', this.removeAllAssignees);
+    eventHub.$on('sidebar.saveAssignees', this.saveAssignees);
+  },
+  beforeDestroy() {
+    eventHub.$off('sidebar.removeAssignee', this.removeAssignee);
+    eventHub.$off('sidebar.addAssignee', this.addAssignee);
+    eventHub.$off('sidebar.removeAllAssignees', this.removeAllAssignees);
+    eventHub.$off('sidebar.saveAssignees', this.saveAssignees);
+  },
+  beforeMount() {
+    this.field = this.$el.dataset.field;
+  },
+  template: `
+    <div>
+      <assignee-title
+        :number-of-assignees="store.assignees.length"
+        :loading="loading"
+        :editable="store.editable"
+      />
+      <assignees
+        class="value"
+        :root-path="store.rootPath"
+        :users="store.assignees"
+        :editable="store.editable"
+        @assign-self="assignSelf"
+      />
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/collapsed_state.js b/app/assets/javascripts/sidebar/components/time_tracking/collapsed_state.js
new file mode 100644
index 0000000000000..0da265053bd0e
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/collapsed_state.js
@@ -0,0 +1,97 @@
+import stopwatchSvg from 'icons/_icon_stopwatch.svg';
+
+import '../../../lib/utils/pretty_time';
+
+export default {
+  name: 'time-tracking-collapsed-state',
+  props: {
+    showComparisonState: {
+      type: Boolean,
+      required: true,
+    },
+    showSpentOnlyState: {
+      type: Boolean,
+      required: true,
+    },
+    showEstimateOnlyState: {
+      type: Boolean,
+      required: true,
+    },
+    showNoTimeTrackingState: {
+      type: Boolean,
+      required: true,
+    },
+    timeSpentHumanReadable: {
+      type: String,
+      required: false,
+      default: '',
+    },
+    timeEstimateHumanReadable: {
+      type: String,
+      required: false,
+      default: '',
+    },
+  },
+  computed: {
+    timeSpent() {
+      return this.abbreviateTime(this.timeSpentHumanReadable);
+    },
+    timeEstimate() {
+      return this.abbreviateTime(this.timeEstimateHumanReadable);
+    },
+    divClass() {
+      if (this.showComparisonState) {
+        return 'compare';
+      } else if (this.showEstimateOnlyState) {
+        return 'estimate-only';
+      } else if (this.showSpentOnlyState) {
+        return 'spend-only';
+      } else if (this.showNoTimeTrackingState) {
+        return 'no-tracking';
+      }
+
+      return '';
+    },
+    spanClass() {
+      if (this.showComparisonState) {
+        return '';
+      } else if (this.showEstimateOnlyState || this.showSpentOnlyState) {
+        return 'bold';
+      } else if (this.showNoTimeTrackingState) {
+        return 'no-value';
+      }
+
+      return '';
+    },
+    text() {
+      if (this.showComparisonState) {
+        return `${this.timeSpent} / ${this.timeEstimate}`;
+      } else if (this.showEstimateOnlyState) {
+        return `-- / ${this.timeEstimate}`;
+      } else if (this.showSpentOnlyState) {
+        return `${this.timeSpent} / --`;
+      } else if (this.showNoTimeTrackingState) {
+        return 'None';
+      }
+
+      return '';
+    },
+  },
+  methods: {
+    abbreviateTime(timeStr) {
+      return gl.utils.prettyTime.abbreviateTime(timeStr);
+    },
+  },
+  template: `
+    <div class="sidebar-collapsed-icon">
+      ${stopwatchSvg}
+      <div class="time-tracking-collapsed-summary">
+        <div :class="divClass">
+          <span :class="spanClass">
+            {{ text }}
+          </span>
+        </div>
+      </div>
+    </div>
+    `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/comparison_pane.js b/app/assets/javascripts/sidebar/components/time_tracking/comparison_pane.js
new file mode 100644
index 0000000000000..40f5c89c5bbcf
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/comparison_pane.js
@@ -0,0 +1,98 @@
+import '../../../lib/utils/pretty_time';
+
+const prettyTime = gl.utils.prettyTime;
+
+export default {
+  name: 'time-tracking-comparison-pane',
+  props: {
+    timeSpent: {
+      type: Number,
+      required: true,
+    },
+    timeEstimate: {
+      type: Number,
+      required: true,
+    },
+    timeSpentHumanReadable: {
+      type: String,
+      required: true,
+    },
+    timeEstimateHumanReadable: {
+      type: String,
+      required: true,
+    },
+  },
+  computed: {
+    parsedRemaining() {
+      const diffSeconds = this.timeEstimate - this.timeSpent;
+      return prettyTime.parseSeconds(diffSeconds);
+    },
+    timeRemainingHumanReadable() {
+      return prettyTime.stringifyTime(this.parsedRemaining);
+    },
+    timeRemainingTooltip() {
+      const prefix = this.timeRemainingMinutes < 0 ? 'Over by' : 'Time remaining:';
+      return `${prefix} ${this.timeRemainingHumanReadable}`;
+    },
+    /* Diff values for comparison meter */
+    timeRemainingMinutes() {
+      return this.timeEstimate - this.timeSpent;
+    },
+    timeRemainingPercent() {
+      return `${Math.floor((this.timeSpent / this.timeEstimate) * 100)}%`;
+    },
+    timeRemainingStatusClass() {
+      return this.timeEstimate >= this.timeSpent ? 'within_estimate' : 'over_estimate';
+    },
+    /* Parsed time values */
+    parsedEstimate() {
+      return prettyTime.parseSeconds(this.timeEstimate);
+    },
+    parsedSpent() {
+      return prettyTime.parseSeconds(this.timeSpent);
+    },
+  },
+  template: `
+    <div class="time-tracking-comparison-pane">
+      <div
+        class="compare-meter"
+        data-toggle="tooltip"
+        data-placement="top"
+        role="timeRemainingDisplay"
+        :aria-valuenow="timeRemainingTooltip"
+        :title="timeRemainingTooltip"
+        :data-original-title="timeRemainingTooltip"
+        :class="timeRemainingStatusClass"
+      >
+        <div
+          class="meter-container"
+          role="timeSpentPercent"
+          :aria-valuenow="timeRemainingPercent"
+        >
+          <div
+            :style="{ width: timeRemainingPercent }"
+            class="meter-fill"
+          />
+        </div>
+        <div class="compare-display-container">
+          <div class="compare-display pull-left">
+            <span class="compare-label">
+              Spent
+            </span>
+            <span class="compare-value spent">
+              {{ timeSpentHumanReadable }}
+            </span>
+          </div>
+          <div class="compare-display estimated pull-right">
+            <span class="compare-label">
+              Est
+            </span>
+            <span class="compare-value">
+              {{ timeEstimateHumanReadable }}
+            </span>
+          </div>
+        </div>
+      </div>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/estimate_only_pane.js b/app/assets/javascripts/sidebar/components/time_tracking/estimate_only_pane.js
new file mode 100644
index 0000000000000..ad1b9179db017
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/estimate_only_pane.js
@@ -0,0 +1,17 @@
+export default {
+  name: 'time-tracking-estimate-only-pane',
+  props: {
+    timeEstimateHumanReadable: {
+      type: String,
+      required: true,
+    },
+  },
+  template: `
+    <div class="time-tracking-estimate-only-pane">
+      <span class="bold">
+        Estimated:
+      </span>
+      {{ timeEstimateHumanReadable }}
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/help_state.js b/app/assets/javascripts/sidebar/components/time_tracking/help_state.js
new file mode 100644
index 0000000000000..b2a77462fe0d4
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/help_state.js
@@ -0,0 +1,44 @@
+export default {
+  name: 'time-tracking-help-state',
+  props: {
+    rootPath: {
+      type: String,
+      required: true,
+    },
+  },
+  computed: {
+    href() {
+      return `${this.rootPath}help/workflow/time_tracking.md`;
+    },
+  },
+  template: `
+    <div class="time-tracking-help-state">
+      <div class="time-tracking-info">
+        <h4>
+          Track time with slash commands
+        </h4>
+        <p>
+          Slash commands can be used in the issues description and comment boxes.
+        </p>
+        <p>
+          <code>
+            /estimate
+          </code>
+          will update the estimated time with the latest command.
+        </p>
+        <p>
+          <code>
+            /spend
+          </code>
+          will update the sum of the time spent.
+        </p>
+        <a
+          class="btn btn-default learn-more-button"
+          :href="href"
+        >
+          Learn more
+        </a>
+      </div>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/no_tracking_pane.js b/app/assets/javascripts/sidebar/components/time_tracking/no_tracking_pane.js
new file mode 100644
index 0000000000000..d1dd1dcdd277d
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/no_tracking_pane.js
@@ -0,0 +1,10 @@
+export default {
+  name: 'time-tracking-no-tracking-pane',
+  template: `
+    <div class="time-tracking-no-tracking-pane">
+      <span class="no-value">
+        No estimate or time spent
+      </span>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js b/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
new file mode 100644
index 0000000000000..244b67b3ad926
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/sidebar_time_tracking.js
@@ -0,0 +1,51 @@
+import '~/smart_interval';
+
+import timeTracker from './time_tracker';
+
+import Store from '../../stores/sidebar_store';
+import Mediator from '../../sidebar_mediator';
+
+export default {
+  data() {
+    return {
+      mediator: new Mediator(),
+      store: new Store(),
+    };
+  },
+  components: {
+    'issuable-time-tracker': timeTracker,
+  },
+  methods: {
+    listenForSlashCommands() {
+      $(document).on('ajax:success', '.gfm-form', this.slashCommandListened);
+    },
+    slashCommandListened(e, data) {
+      const subscribedCommands = ['spend_time', 'time_estimate'];
+      let changedCommands;
+      if (data !== undefined) {
+        changedCommands = data.commands_changes
+          ? Object.keys(data.commands_changes)
+          : [];
+      } else {
+        changedCommands = [];
+      }
+      if (changedCommands && _.intersection(subscribedCommands, changedCommands).length) {
+        this.mediator.fetch();
+      }
+    },
+  },
+  mounted() {
+    this.listenForSlashCommands();
+  },
+  template: `
+    <div class="block">
+      <issuable-time-tracker
+        :time_estimate="store.timeEstimate"
+        :time_spent="store.totalTimeSpent"
+        :human_time_estimate="store.humanTimeEstimate"
+        :human_time_spent="store.humanTotalTimeSpent"
+        :rootPath="store.rootPath"
+      />
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/spent_only_pane.js b/app/assets/javascripts/sidebar/components/time_tracking/spent_only_pane.js
new file mode 100644
index 0000000000000..bf9875626475e
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/spent_only_pane.js
@@ -0,0 +1,15 @@
+export default {
+  name: 'time-tracking-spent-only-pane',
+  props: {
+    timeSpentHumanReadable: {
+      type: String,
+      required: true,
+    },
+  },
+  template: `
+    <div class="time-tracking-spend-only-pane">
+      <span class="bold">Spent:</span>
+      {{ timeSpentHumanReadable }}
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/components/time_tracking/time_tracker.js b/app/assets/javascripts/sidebar/components/time_tracking/time_tracker.js
new file mode 100644
index 0000000000000..ed0d71a4f797d
--- /dev/null
+++ b/app/assets/javascripts/sidebar/components/time_tracking/time_tracker.js
@@ -0,0 +1,163 @@
+import timeTrackingHelpState from './help_state';
+import timeTrackingCollapsedState from './collapsed_state';
+import timeTrackingSpentOnlyPane from './spent_only_pane';
+import timeTrackingNoTrackingPane from './no_tracking_pane';
+import timeTrackingEstimateOnlyPane from './estimate_only_pane';
+import timeTrackingComparisonPane from './comparison_pane';
+
+import eventHub from '../../event_hub';
+
+export default {
+  name: 'issuable-time-tracker',
+  props: {
+    time_estimate: {
+      type: Number,
+      required: true,
+    },
+    time_spent: {
+      type: Number,
+      required: true,
+    },
+    human_time_estimate: {
+      type: String,
+      required: false,
+      default: '',
+    },
+    human_time_spent: {
+      type: String,
+      required: false,
+      default: '',
+    },
+    rootPath: {
+      type: String,
+      required: true,
+    },
+  },
+  data() {
+    return {
+      showHelp: false,
+    };
+  },
+  components: {
+    'time-tracking-collapsed-state': timeTrackingCollapsedState,
+    'time-tracking-estimate-only-pane': timeTrackingEstimateOnlyPane,
+    'time-tracking-spent-only-pane': timeTrackingSpentOnlyPane,
+    'time-tracking-no-tracking-pane': timeTrackingNoTrackingPane,
+    'time-tracking-comparison-pane': timeTrackingComparisonPane,
+    'time-tracking-help-state': timeTrackingHelpState,
+  },
+  computed: {
+    timeSpent() {
+      return this.time_spent;
+    },
+    timeEstimate() {
+      return this.time_estimate;
+    },
+    timeEstimateHumanReadable() {
+      return this.human_time_estimate;
+    },
+    timeSpentHumanReadable() {
+      return this.human_time_spent;
+    },
+    hasTimeSpent() {
+      return !!this.timeSpent;
+    },
+    hasTimeEstimate() {
+      return !!this.timeEstimate;
+    },
+    showComparisonState() {
+      return this.hasTimeEstimate && this.hasTimeSpent;
+    },
+    showEstimateOnlyState() {
+      return this.hasTimeEstimate && !this.hasTimeSpent;
+    },
+    showSpentOnlyState() {
+      return this.hasTimeSpent && !this.hasTimeEstimate;
+    },
+    showNoTimeTrackingState() {
+      return !this.hasTimeEstimate && !this.hasTimeSpent;
+    },
+    showHelpState() {
+      return !!this.showHelp;
+    },
+  },
+  methods: {
+    toggleHelpState(show) {
+      this.showHelp = show;
+    },
+    update(data) {
+      this.time_estimate = data.time_estimate;
+      this.time_spent = data.time_spent;
+      this.human_time_estimate = data.human_time_estimate;
+      this.human_time_spent = data.human_time_spent;
+    },
+  },
+  created() {
+    eventHub.$on('timeTracker:updateData', this.update);
+  },
+  template: `
+    <div
+      class="time_tracker time-tracking-component-wrap"
+      v-cloak
+    >
+      <time-tracking-collapsed-state
+        :show-comparison-state="showComparisonState"
+        :show-no-time-tracking-state="showNoTimeTrackingState"
+        :show-help-state="showHelpState"
+        :show-spent-only-state="showSpentOnlyState"
+        :show-estimate-only-state="showEstimateOnlyState"
+        :time-spent-human-readable="timeSpentHumanReadable"
+        :time-estimate-human-readable="timeEstimateHumanReadable"
+      />
+      <div class="title hide-collapsed">
+        Time tracking
+        <div
+          class="help-button pull-right"
+          v-if="!showHelpState"
+          @click="toggleHelpState(true)"
+        >
+            <i
+              class="fa fa-question-circle"
+              aria-hidden="true"
+            />
+        </div>
+        <div
+          class="close-help-button pull-right"
+          v-if="showHelpState"
+          @click="toggleHelpState(false)"
+        >
+          <i
+            class="fa fa-close"
+            aria-hidden="true"
+          />
+        </div>
+      </div>
+      <div class="time-tracking-content hide-collapsed">
+        <time-tracking-estimate-only-pane
+          v-if="showEstimateOnlyState"
+          :time-estimate-human-readable="timeEstimateHumanReadable"
+        />
+        <time-tracking-spent-only-pane
+          v-if="showSpentOnlyState"
+          :time-spent-human-readable="timeSpentHumanReadable"
+        />
+        <time-tracking-no-tracking-pane
+          v-if="showNoTimeTrackingState"
+        />
+        <time-tracking-comparison-pane
+          v-if="showComparisonState"
+          :time-estimate="timeEstimate"
+          :time-spent="timeSpent"
+          :time-spent-human-readable="timeSpentHumanReadable"
+          :time-estimate-human-readable="timeEstimateHumanReadable"
+        />
+        <transition name="help-state-toggle">
+          <time-tracking-help-state
+            v-if="showHelpState"
+            :rootPath="rootPath"
+          />
+        </transition>
+      </div>
+    </div>
+  `,
+};
diff --git a/app/assets/javascripts/sidebar/event_hub.js b/app/assets/javascripts/sidebar/event_hub.js
new file mode 100644
index 0000000000000..0948c2e53524a
--- /dev/null
+++ b/app/assets/javascripts/sidebar/event_hub.js
@@ -0,0 +1,3 @@
+import Vue from 'vue';
+
+export default new Vue();
diff --git a/app/assets/javascripts/sidebar/services/sidebar_service.js b/app/assets/javascripts/sidebar/services/sidebar_service.js
new file mode 100644
index 0000000000000..5a82d01dc41e7
--- /dev/null
+++ b/app/assets/javascripts/sidebar/services/sidebar_service.js
@@ -0,0 +1,28 @@
+import Vue from 'vue';
+import VueResource from 'vue-resource';
+
+Vue.use(VueResource);
+
+export default class SidebarService {
+  constructor(endpoint) {
+    if (!SidebarService.singleton) {
+      this.endpoint = endpoint;
+
+      SidebarService.singleton = this;
+    }
+
+    return SidebarService.singleton;
+  }
+
+  get() {
+    return Vue.http.get(this.endpoint);
+  }
+
+  update(key, data) {
+    return Vue.http.put(this.endpoint, {
+      [key]: data,
+    }, {
+      emulateJSON: true,
+    });
+  }
+}
diff --git a/app/assets/javascripts/sidebar/sidebar_bundle.js b/app/assets/javascripts/sidebar/sidebar_bundle.js
new file mode 100644
index 0000000000000..2b02af87d8ac4
--- /dev/null
+++ b/app/assets/javascripts/sidebar/sidebar_bundle.js
@@ -0,0 +1,24 @@
+import Vue from 'vue';
+import sidebarTimeTracking from './components/time_tracking/sidebar_time_tracking';
+import sidebarAssignees from './components/assignees/sidebar_assignees';
+
+import Mediator from './sidebar_mediator';
+
+function domContentLoaded() {
+  const mediator = new Mediator(gl.sidebarOptions);
+  mediator.fetch();
+
+  const sidebarAssigneesEl = document.querySelector('#js-vue-sidebar-assignees');
+
+  // Only create the sidebarAssignees vue app if it is found in the DOM
+  // We currently do not use sidebarAssignees for the MR page
+  if (sidebarAssigneesEl) {
+    new Vue(sidebarAssignees).$mount(sidebarAssigneesEl);
+  }
+
+  new Vue(sidebarTimeTracking).$mount('#issuable-time-tracker');
+}
+
+document.addEventListener('DOMContentLoaded', domContentLoaded);
+
+export default domContentLoaded;
diff --git a/app/assets/javascripts/sidebar/sidebar_mediator.js b/app/assets/javascripts/sidebar/sidebar_mediator.js
new file mode 100644
index 0000000000000..5ccfb4ee9c1d3
--- /dev/null
+++ b/app/assets/javascripts/sidebar/sidebar_mediator.js
@@ -0,0 +1,38 @@
+/* global Flash */
+
+import Service from './services/sidebar_service';
+import Store from './stores/sidebar_store';
+
+export default class SidebarMediator {
+  constructor(options) {
+    if (!SidebarMediator.singleton) {
+      this.store = new Store(options);
+      this.service = new Service(options.endpoint);
+      SidebarMediator.singleton = this;
+    }
+
+    return SidebarMediator.singleton;
+  }
+
+  assignYourself() {
+    this.store.addAssignee(this.store.currentUser);
+  }
+
+  saveAssignees(field) {
+    const selected = this.store.assignees.map(u => u.id);
+
+    // If there are no ids, that means we have to unassign (which is id = 0)
+    // And it only accepts an array, hence [0]
+    return this.service.update(field, selected.length === 0 ? [0] : selected);
+  }
+
+  fetch() {
+    this.service.get()
+      .then((response) => {
+        const data = response.json();
+        this.store.setAssigneeData(data);
+        this.store.setTimeTrackingData(data);
+      })
+      .catch(() => new Flash('Error occured when fetching sidebar data'));
+  }
+}
diff --git a/app/assets/javascripts/sidebar/stores/sidebar_store.js b/app/assets/javascripts/sidebar/stores/sidebar_store.js
new file mode 100644
index 0000000000000..2d44c05bb8d24
--- /dev/null
+++ b/app/assets/javascripts/sidebar/stores/sidebar_store.js
@@ -0,0 +1,52 @@
+export default class SidebarStore {
+  constructor(store) {
+    if (!SidebarStore.singleton) {
+      const { currentUser, rootPath, editable } = store;
+      this.currentUser = currentUser;
+      this.rootPath = rootPath;
+      this.editable = editable;
+      this.timeEstimate = 0;
+      this.totalTimeSpent = 0;
+      this.humanTimeEstimate = '';
+      this.humanTimeSpent = '';
+      this.assignees = [];
+
+      SidebarStore.singleton = this;
+    }
+
+    return SidebarStore.singleton;
+  }
+
+  setAssigneeData(data) {
+    if (data.assignees) {
+      this.assignees = data.assignees;
+    }
+  }
+
+  setTimeTrackingData(data) {
+    this.timeEstimate = data.time_estimate;
+    this.totalTimeSpent = data.total_time_spent;
+    this.humanTimeEstimate = data.human_time_estimate;
+    this.humanTotalTimeSpent = data.human_total_time_spent;
+  }
+
+  addAssignee(assignee) {
+    if (!this.findAssignee(assignee)) {
+      this.assignees.push(assignee);
+    }
+  }
+
+  findAssignee(findAssignee) {
+    return this.assignees.filter(assignee => assignee.id === findAssignee.id)[0];
+  }
+
+  removeAssignee(removeAssignee) {
+    if (removeAssignee) {
+      this.assignees = this.assignees.filter(assignee => assignee.id !== removeAssignee.id);
+    }
+  }
+
+  removeAllAssignees() {
+    this.assignees = [];
+  }
+}
diff --git a/app/assets/javascripts/signin_tabs_memoizer.js b/app/assets/javascripts/signin_tabs_memoizer.js
index d811d1cd53abf..2587facc58225 100644
--- a/app/assets/javascripts/signin_tabs_memoizer.js
+++ b/app/assets/javascripts/signin_tabs_memoizer.js
@@ -1,5 +1,7 @@
 /* eslint no-param-reassign: ["error", { "props": false }]*/
 /* eslint no-new: "off" */
+import AccessorUtilities from './lib/utils/accessor';
+
 ((global) => {
   /**
    * Memorize the last selected tab after reloading a page.
@@ -9,6 +11,8 @@
     constructor({ currentTabKey = 'current_signin_tab', tabSelector = 'ul.nav-tabs' } = {}) {
       this.currentTabKey = currentTabKey;
       this.tabSelector = tabSelector;
+      this.isLocalStorageAvailable = AccessorUtilities.isLocalStorageAccessSafe();
+
       this.bootstrap();
     }
 
@@ -37,11 +41,15 @@
     }
 
     saveData(val) {
-      localStorage.setItem(this.currentTabKey, val);
+      if (!this.isLocalStorageAvailable) return undefined;
+
+      return window.localStorage.setItem(this.currentTabKey, val);
     }
 
     readData() {
-      return localStorage.getItem(this.currentTabKey);
+      if (!this.isLocalStorageAvailable) return null;
+
+      return window.localStorage.getItem(this.currentTabKey);
     }
   }
 
diff --git a/app/assets/javascripts/subbable_resource.js b/app/assets/javascripts/subbable_resource.js
deleted file mode 100644
index d819160512849..0000000000000
--- a/app/assets/javascripts/subbable_resource.js
+++ /dev/null
@@ -1,51 +0,0 @@
-(() => {
-/*
-*   SubbableResource can be extended to provide a pubsub-style service for one-off REST
-*   calls. Subscribe by passing a callback or render method you will use to handle responses.
- *
-* */
-
-  class SubbableResource {
-    constructor(resourcePath) {
-      this.endpoint = resourcePath;
-
-      // TODO: Switch to axios.create
-      this.resource = $.ajax;
-      this.subscribers = [];
-    }
-
-    subscribe(callback) {
-      this.subscribers.push(callback);
-    }
-
-    publish(newResponse) {
-      const responseCopy = _.extend({}, newResponse);
-      this.subscribers.forEach((fn) => {
-        fn(responseCopy);
-      });
-      return newResponse;
-    }
-
-    get(payload) {
-      return this.resource(payload)
-        .then(data => this.publish(data));
-    }
-
-    post(payload) {
-      return this.resource(payload)
-        .then(data => this.publish(data));
-    }
-
-    put(payload) {
-      return this.resource(payload)
-        .then(data => this.publish(data));
-    }
-
-    delete(payload) {
-      return this.resource(payload)
-        .then(data => this.publish(data));
-    }
-  }
-
-  gl.SubbableResource = SubbableResource;
-})(window.gl || (window.gl = {}));
diff --git a/app/assets/javascripts/subscription_select.js b/app/assets/javascripts/subscription_select.js
index 8b25f43ffc745..0cd591c732086 100644
--- a/app/assets/javascripts/subscription_select.js
+++ b/app/assets/javascripts/subscription_select.js
@@ -19,8 +19,8 @@
               return label;
             };
           })(this),
-          clicked: function(item, $el, e) {
-            return e.preventDefault();
+          clicked: function(options) {
+            return options.e.preventDefault();
           },
           id: function(obj, el) {
             return $(el).data("id");
diff --git a/app/assets/javascripts/users_select.js b/app/assets/javascripts/users_select.js
index 26c90bf8229b9..fecb09cb51ae3 100644
--- a/app/assets/javascripts/users_select.js
+++ b/app/assets/javascripts/users_select.js
@@ -1,6 +1,7 @@
 /* eslint-disable func-names, space-before-function-paren, one-var, no-var, prefer-rest-params, wrap-iife, quotes, max-len, one-var-declaration-per-line, vars-on-top, prefer-arrow-callback, consistent-return, comma-dangle, object-shorthand, no-shadow, no-unused-vars, no-else-return, no-self-compare, prefer-template, no-unused-expressions, no-lonely-if, yoda, prefer-spread, no-void, camelcase, no-param-reassign */
 /* global Issuable */
-/* global ListUser */
+
+import eventHub from './sidebar/event_hub';
 
 (function() {
   var bind = function(fn, me) { return function() { return fn.apply(me, arguments); }; },
@@ -54,42 +55,115 @@
           selectedIdDefault = (defaultNullUser && showNullUser) ? 0 : null;
           selectedId = $dropdown.data('selected') || selectedIdDefault;
 
-          var updateIssueBoardsIssue = function () {
-            $loading.removeClass('hidden').fadeIn();
-            gl.issueBoards.BoardsStore.detail.issue.update($dropdown.attr('data-issue-update'))
-              .then(function () {
-                $loading.fadeOut();
-              })
-              .catch(function () {
-                $loading.fadeOut();
-              });
+          const assignYourself = function () {
+            const unassignedSelected = $dropdown.closest('.selectbox')
+              .find(`input[name='${$dropdown.data('field-name')}'][value=0]`);
+
+            if (unassignedSelected) {
+              unassignedSelected.remove();
+            }
+
+            // Save current selected user to the DOM
+            const input = document.createElement('input');
+            input.type = 'hidden';
+            input.name = $dropdown.data('field-name');
+
+            const currentUserInfo = $dropdown.data('currentUserInfo');
+
+            if (currentUserInfo) {
+              input.value = currentUserInfo.id;
+              input.dataset.meta = currentUserInfo.name;
+            } else if (_this.currentUser) {
+              input.value = _this.currentUser.id;
+            }
+
+            if ($selectbox) {
+              $dropdown.parent().before(input);
+            } else {
+              $dropdown.after(input);
+            }
+          };
+
+          if ($block[0]) {
+            $block[0].addEventListener('assignYourself', assignYourself);
+          }
+
+          const getSelectedUserInputs = function() {
+            return $selectbox
+              .find(`input[name="${$dropdown.data('field-name')}"]`);
+          };
+
+          const getSelected = function() {
+            return getSelectedUserInputs()
+              .map((index, input) => parseInt(input.value, 10))
+              .get();
+          };
+
+          const checkMaxSelect = function() {
+            const maxSelect = $dropdown.data('max-select');
+            if (maxSelect) {
+              const selected = getSelected();
+
+              if (selected.length > maxSelect) {
+                const firstSelectedId = selected[0];
+                const firstSelected = $dropdown.closest('.selectbox')
+                  .find(`input[name='${$dropdown.data('field-name')}'][value=${firstSelectedId}]`);
+
+                firstSelected.remove();
+                eventHub.$emit('sidebar.removeAssignee', {
+                  id: firstSelectedId,
+                });
+              }
+            }
+          };
+
+          const getMultiSelectDropdownTitle = function(selectedUser, isSelected) {
+            const selectedUsers = getSelected()
+              .filter(u => u !== 0);
+
+            const firstUser = getSelectedUserInputs()
+              .map((index, input) => ({
+                name: input.dataset.meta,
+                value: parseInt(input.value, 10),
+              }))
+              .filter(u => u.id !== 0)
+              .get(0);
+
+            if (selectedUsers.length === 0) {
+              return 'Unassigned';
+            } else if (selectedUsers.length === 1) {
+              return firstUser.name;
+            } else if (isSelected) {
+              const otherSelected = selectedUsers.filter(s => s !== selectedUser.id);
+              return `${selectedUser.name} + ${otherSelected.length} more`;
+            } else {
+              return `${firstUser.name} + ${selectedUsers.length - 1} more`;
+            }
           };
 
           $('.assign-to-me-link').on('click', (e) => {
             e.preventDefault();
             $(e.currentTarget).hide();
-            const $input = $(`input[name="${$dropdown.data('field-name')}"]`);
-            $input.val(gon.current_user_id);
-            selectedId = $input.val();
-            $dropdown.find('.dropdown-toggle-text').text(gon.current_user_fullname).removeClass('is-default');
-          });
 
-          $block.on('click', '.js-assign-yourself', function(e) {
-            e.preventDefault();
-
-            if ($dropdown.hasClass('js-issue-board-sidebar')) {
-              gl.issueBoards.boardStoreIssueSet('assignee', new ListUser({
-                id: _this.currentUser.id,
-                username: _this.currentUser.username,
-                name: _this.currentUser.name,
-                avatar_url: _this.currentUser.avatar_url
-              }));
+            if ($dropdown.data('multiSelect')) {
+              assignYourself();
+              checkMaxSelect();
 
-              updateIssueBoardsIssue();
+              const currentUserInfo = $dropdown.data('currentUserInfo');
+              $dropdown.find('.dropdown-toggle-text').text(getMultiSelectDropdownTitle(currentUserInfo)).removeClass('is-default');
             } else {
-              return assignTo(_this.currentUser.id);
+              const $input = $(`input[name="${$dropdown.data('field-name')}"]`);
+              $input.val(gon.current_user_id);
+              selectedId = $input.val();
+              $dropdown.find('.dropdown-toggle-text').text(gon.current_user_fullname).removeClass('is-default');
             }
           });
+
+          $block.on('click', '.js-assign-yourself', (e) => {
+            e.preventDefault();
+            return assignTo(_this.currentUser.id);
+          });
+
           assignTo = function(selected) {
             var data;
             data = {};
@@ -97,6 +171,7 @@
             data[abilityName].assignee_id = selected != null ? selected : null;
             $loading.removeClass('hidden').fadeIn();
             $dropdown.trigger('loading.gl.dropdown');
+
             return $.ajax({
               type: 'PUT',
               dataType: 'json',
@@ -106,7 +181,6 @@
               var user;
               $dropdown.trigger('loaded.gl.dropdown');
               $loading.fadeOut();
-              $selectbox.hide();
               if (data.assignee) {
                 user = {
                   name: data.assignee.name,
@@ -133,51 +207,90 @@
               var isAuthorFilter;
               isAuthorFilter = $('.js-author-search');
               return _this.users(term, options, function(users) {
-                var anyUser, index, j, len, name, obj, showDivider;
-                if (term.length === 0) {
-                  showDivider = 0;
-                  if (firstUser) {
-                    // Move current user to the front of the list
-                    for (index = j = 0, len = users.length; j < len; index = (j += 1)) {
-                      obj = users[index];
-                      if (obj.username === firstUser) {
-                        users.splice(index, 1);
-                        users.unshift(obj);
-                        break;
-                      }
+                // GitLabDropdownFilter returns this.instance
+                // GitLabDropdownRemote returns this.options.instance
+                const glDropdown = this.instance || this.options.instance;
+                glDropdown.options.processData(term, users, callback);
+              }.bind(this));
+            },
+            processData: function(term, users, callback) {
+              let anyUser;
+              let index;
+              let j;
+              let len;
+              let name;
+              let obj;
+              let showDivider;
+              if (term.length === 0) {
+                showDivider = 0;
+                if (firstUser) {
+                  // Move current user to the front of the list
+                  for (index = j = 0, len = users.length; j < len; index = (j += 1)) {
+                    obj = users[index];
+                    if (obj.username === firstUser) {
+                      users.splice(index, 1);
+                      users.unshift(obj);
+                      break;
                     }
                   }
-                  if (showNullUser) {
-                    showDivider += 1;
-                    users.unshift({
-                      beforeDivider: true,
-                      name: 'Unassigned',
-                      id: 0
-                    });
-                  }
-                  if (showAnyUser) {
-                    showDivider += 1;
-                    name = showAnyUser;
-                    if (name === true) {
-                      name = 'Any User';
-                    }
-                    anyUser = {
-                      beforeDivider: true,
-                      name: name,
-                      id: null
-                    };
-                    users.unshift(anyUser);
+                }
+                if (showNullUser) {
+                  showDivider += 1;
+                  users.unshift({
+                    beforeDivider: true,
+                    name: 'Unassigned',
+                    id: 0
+                  });
+                }
+                if (showAnyUser) {
+                  showDivider += 1;
+                  name = showAnyUser;
+                  if (name === true) {
+                    name = 'Any User';
                   }
+                  anyUser = {
+                    beforeDivider: true,
+                    name: name,
+                    id: null
+                  };
+                  users.unshift(anyUser);
                 }
+
                 if (showDivider) {
-                  users.splice(showDivider, 0, "divider");
+                  users.splice(showDivider, 0, 'divider');
                 }
 
-                callback(users);
-                if (showMenuAbove) {
-                  $dropdown.data('glDropdown').positionMenuAbove();
+                if ($dropdown.hasClass('js-multiselect')) {
+                  const selected = getSelected().filter(i => i !== 0);
+
+                  if (selected.length > 0) {
+                    if ($dropdown.data('dropdown-header')) {
+                      showDivider += 1;
+                      users.splice(showDivider, 0, {
+                        header: $dropdown.data('dropdown-header'),
+                      });
+                    }
+
+                    const selectedUsers = users
+                      .filter(u => selected.indexOf(u.id) !== -1)
+                      .sort((a, b) => a.name > b.name);
+
+                    users = users.filter(u => selected.indexOf(u.id) === -1);
+
+                    selectedUsers.forEach((selectedUser) => {
+                      showDivider += 1;
+                      users.splice(showDivider, 0, selectedUser);
+                    });
+
+                    users.splice(showDivider + 1, 0, 'divider');
+                  }
                 }
-              });
+              }
+
+              callback(users);
+              if (showMenuAbove) {
+                $dropdown.data('glDropdown').positionMenuAbove();
+              }
             },
             filterable: true,
             filterRemote: true,
@@ -186,7 +299,22 @@
             },
             selectable: true,
             fieldName: $dropdown.data('field-name'),
-            toggleLabel: function(selected, el) {
+            toggleLabel: function(selected, el, glDropdown) {
+              const inputValue = glDropdown.filterInput.val();
+
+              if (this.multiSelect && inputValue === '') {
+                // Remove non-users from the fullData array
+                const users = glDropdown.filteredFullData();
+                const callback = glDropdown.parseData.bind(glDropdown);
+
+                // Update the data model
+                this.processData(inputValue, users, callback);
+              }
+
+              if (this.multiSelect) {
+                return getMultiSelectDropdownTitle(selected, $(el).hasClass('is-active'));
+              }
+
               if (selected && 'id' in selected && $(el).hasClass('is-active')) {
                 $dropdown.find('.dropdown-toggle-text').removeClass('is-default');
                 if (selected.text) {
@@ -200,22 +328,81 @@
               }
             },
             defaultLabel: defaultLabel,
-            inputId: 'issue_assignee_id',
             hidden: function(e) {
-              $selectbox.hide();
-              // display:block overrides the hide-collapse rule
-              return $value.css('display', '');
+              if ($dropdown.hasClass('js-multiselect')) {
+                eventHub.$emit('sidebar.saveAssignees');
+              }
+
+              if (!$dropdown.data('always-show-selectbox')) {
+                $selectbox.hide();
+
+                // Recalculate where .value is because vue might have changed it
+                $block = $selectbox.closest('.block');
+                $value = $block.find('.value');
+                // display:block overrides the hide-collapse rule
+                $value.css('display', '');
+              }
             },
-            vue: $dropdown.hasClass('js-issue-board-sidebar'),
-            clicked: function(user, $el, e) {
-              var isIssueIndex, isMRIndex, page, selected, isSelecting;
+            multiSelect: $dropdown.hasClass('js-multiselect'),
+            inputMeta: $dropdown.data('input-meta'),
+            clicked: function(options) {
+              const { $el, e, isMarking } = options;
+              const user = options.selectedObj;
+
+              if ($dropdown.hasClass('js-multiselect')) {
+                const isActive = $el.hasClass('is-active');
+                const previouslySelected = $dropdown.closest('.selectbox')
+                    .find("input[name='" + ($dropdown.data('field-name')) + "'][value!=0]");
+
+                // Enables support for limiting the number of users selected
+                // Automatically removes the first on the list if more users are selected
+                checkMaxSelect();
+
+                if (user.beforeDivider && user.name.toLowerCase() === 'unassigned') {
+                  // Unassigned selected
+                  previouslySelected.each((index, element) => {
+                    const id = parseInt(element.value, 10);
+                    element.remove();
+                  });
+                  eventHub.$emit('sidebar.removeAllAssignees');
+                } else if (isActive) {
+                  // user selected
+                  eventHub.$emit('sidebar.addAssignee', user);
+
+                  // Remove unassigned selection (if it was previously selected)
+                  const unassignedSelected = $dropdown.closest('.selectbox')
+                    .find("input[name='" + ($dropdown.data('field-name')) + "'][value=0]");
+
+                  if (unassignedSelected) {
+                    unassignedSelected.remove();
+                  }
+                } else {
+                  if (previouslySelected.length === 0) {
+                  // Select unassigned because there is no more selected users
+                    this.addInput($dropdown.data('field-name'), 0, {});
+                  }
+
+                  // User unselected
+                  eventHub.$emit('sidebar.removeAssignee', user);
+                }
+
+                if (getSelected().find(u => u === gon.current_user_id)) {
+                  $('.assign-to-me-link').hide();
+                } else {
+                  $('.assign-to-me-link').show();
+                }
+              }
+
+              var isIssueIndex, isMRIndex, page, selected;
               page = $('body').data('page');
               isIssueIndex = page === 'projects:issues:index';
               isMRIndex = (page === page && page === 'projects:merge_requests:index');
-              isSelecting = (user.id !== selectedId);
-              selectedId = isSelecting ? user.id : selectedIdDefault;
               if ($dropdown.hasClass('js-filter-bulk-update') || $dropdown.hasClass('js-issuable-form-dropdown')) {
                 e.preventDefault();
+
+                const isSelecting = (user.id !== selectedId);
+                selectedId = isSelecting ? user.id : selectedIdDefault;
+
                 if (selectedId === gon.current_user_id) {
                   $('.assign-to-me-link').hide();
                 } else {
@@ -229,20 +416,7 @@
                 return Issuable.filterResults($dropdown.closest('form'));
               } else if ($dropdown.hasClass('js-filter-submit')) {
                 return $dropdown.closest('form').submit();
-              } else if ($dropdown.hasClass('js-issue-board-sidebar')) {
-                if (user.id && isSelecting) {
-                  gl.issueBoards.boardStoreIssueSet('assignee', new ListUser({
-                    id: user.id,
-                    username: user.username,
-                    name: user.name,
-                    avatar_url: user.avatar_url
-                  }));
-                } else {
-                  gl.issueBoards.boardStoreIssueDelete('assignee');
-                }
-
-                updateIssueBoardsIssue();
-              } else {
+              } else if (!$dropdown.hasClass('js-multiselect')) {
                 selected = $dropdown.closest('.selectbox').find("input[name='" + ($dropdown.data('field-name')) + "']").val();
                 return assignTo(selected);
               }
@@ -256,29 +430,54 @@
                 selectedId = parseInt($dropdown[0].dataset.selected, 10) || selectedIdDefault;
               }
               $el.find('.is-active').removeClass('is-active');
-              $el.find(`li[data-user-id="${selectedId}"] .dropdown-menu-user-link`).addClass('is-active');
+
+              function highlightSelected(id) {
+                $el.find(`li[data-user-id="${id}"] .dropdown-menu-user-link`).addClass('is-active');
+              }
+
+              if ($selectbox[0]) {
+                getSelected().forEach(selectedId => highlightSelected(selectedId));
+              } else {
+                highlightSelected(selectedId);
+              }
             },
+            updateLabel: $dropdown.data('dropdown-title'),
             renderRow: function(user) {
-              var avatar, img, listClosingTags, listWithName, listWithUserName, selected, username;
+              var avatar, img, listClosingTags, listWithName, listWithUserName, username;
               username = user.username ? "@" + user.username : "";
               avatar = user.avatar_url ? user.avatar_url : false;
-              selected = user.id === parseInt(selectedId, 10) ? "is-active" : "";
+
+              let selected = user.id === parseInt(selectedId, 10);
+
+              if (this.multiSelect) {
+                const fieldName = this.fieldName;
+                const field = $dropdown.closest('.selectbox').find("input[name='" + fieldName + "'][value='" + user.id + "']");
+
+                if (field.length) {
+                  selected = true;
+                }
+              }
+
               img = "";
               if (user.beforeDivider != null) {
-                "<li> <a href='#' class='" + selected + "'> " + user.name + " </a> </li>";
+                `<li><a href='#' class='${selected === true ? 'is-active' : ''}'>${user.name}</a></li>`;
               } else {
                 if (avatar) {
-                  img = "<img src='" + avatar + "' class='avatar avatar-inline' width='30' />";
+                  img = "<img src='" + avatar + "' class='avatar avatar-inline' width='32' />";
                 }
               }
-              // split into three parts so we can remove the username section if nessesary
-              listWithName = "<li data-user-id=" + user.id + "> <a href='#' class='dropdown-menu-user-link " + selected + "'> " + img + " <strong class='dropdown-menu-user-full-name'> " + user.name + " </strong>";
-              listWithUserName = "<span class='dropdown-menu-user-username'> " + username + " </span>";
-              listClosingTags = "</a> </li>";
-              if (username === '') {
-                listWithUserName = '';
-              }
-              return listWithName + listWithUserName + listClosingTags;
+
+              return `
+                <li data-user-id=${user.id}>
+                  <a href='#' class='dropdown-menu-user-link ${selected === true ? 'is-active' : ''}'>
+                    ${img}
+                    <strong class='dropdown-menu-user-full-name'>
+                      ${user.name}
+                    </strong>
+                    ${username ? `<span class='dropdown-menu-user-username'>${username}</span>` : ''}
+                  </a>
+                </li>
+              `;
             }
           });
         };
diff --git a/app/assets/javascripts/vue_shared/translate.js b/app/assets/javascripts/vue_shared/translate.js
new file mode 100644
index 0000000000000..f83c4b00761d2
--- /dev/null
+++ b/app/assets/javascripts/vue_shared/translate.js
@@ -0,0 +1,42 @@
+import {
+  __,
+  n__,
+  s__,
+} from '../locale';
+
+export default (Vue) => {
+  Vue.mixin({
+    methods: {
+      /**
+        Translates `text`
+
+        @param text The text to be translated
+        @returns {String} The translated text
+      **/
+      __,
+      /**
+        Translate the text with a number
+        if the number is more than 1 it will use the `pluralText` translation.
+        This method allows for contexts, see below re. contexts
+
+        @param text Singular text to translate (eg. '%d day')
+        @param pluralText Plural text to translate (eg. '%d days')
+        @param count Number to decide which translation to use (eg. 2)
+        @returns {String} Translated text with the number replaced (eg. '2 days')
+      **/
+      n__,
+      /**
+        Translate context based text
+        Either pass in the context translation like `Context|Text to translate`
+        or allow for dynamic text by doing passing in the context first & then the text to translate
+
+        @param keyOrContext Can be either the key to translate including the context
+                            (eg. 'Context|Text') or just the context for the translation
+                            (eg. 'Context')
+        @param key Is the dynamic variable you want to be translated
+        @returns {String} Translated context based text
+      **/
+      s__,
+    },
+  });
+};
diff --git a/app/assets/javascripts/weight_select.js b/app/assets/javascripts/weight_select.js
index 7964023537cf7..ca9a1184add77 100644
--- a/app/assets/javascripts/weight_select.js
+++ b/app/assets/javascripts/weight_select.js
@@ -54,7 +54,10 @@
               return '';
             }
           },
-          clicked: function(selected, $el, e) {
+          clicked: function(options) {
+            const e = options.e;
+            let selected = options.selectedObj;
+
             if ($(dropdown).is(".js-filter-submit")) {
               return $(dropdown).parents('form').submit();
             } else if ($dropdown.is('.js-issuable-form-weight')) {
diff --git a/app/assets/stylesheets/framework/animations.scss b/app/assets/stylesheets/framework/animations.scss
index 7c50b80fd2bb5..3cd7f81da4719 100644
--- a/app/assets/stylesheets/framework/animations.scss
+++ b/app/assets/stylesheets/framework/animations.scss
@@ -159,3 +159,31 @@ a {
 .fade-in {
   animation: fadeIn $fade-in-duration 1;
 }
+
+@keyframes fadeInHalf {
+  0% {
+    opacity: 0;
+  }
+
+  100% {
+    opacity: 0.5;
+  }
+}
+
+.fade-in-half {
+  animation: fadeInHalf $fade-in-duration 1;
+}
+
+@keyframes fadeInFull {
+  0% {
+    opacity: 0.5;
+  }
+
+  100% {
+    opacity: 1;
+  }
+}
+
+.fade-in-full {
+  animation: fadeInFull $fade-in-duration 1;
+}
diff --git a/app/assets/stylesheets/framework/avatar.scss b/app/assets/stylesheets/framework/avatar.scss
index 3f5b78ed44536..91c1ebd5a7de3 100644
--- a/app/assets/stylesheets/framework/avatar.scss
+++ b/app/assets/stylesheets/framework/avatar.scss
@@ -93,3 +93,14 @@
     align-self: center;
   }
 }
+
+.avatar-counter {
+  background-color: $gray-darkest;
+  color: $white-light;
+  border: 1px solid $border-color;
+  border-radius: 1em;
+  font-family: $regular_font;
+  font-size: 9px;
+  line-height: 16px;
+  text-align: center;
+}
diff --git a/app/assets/stylesheets/framework/dropdowns.scss b/app/assets/stylesheets/framework/dropdowns.scss
index 73ded9f30d470..5c9b71a452cc0 100644
--- a/app/assets/stylesheets/framework/dropdowns.scss
+++ b/app/assets/stylesheets/framework/dropdowns.scss
@@ -251,14 +251,16 @@
   }
 
   .dropdown-header {
-    color: $gl-text-color;
+    color: $gl-text-color-secondary;
     font-size: 13px;
-    font-weight: 600;
     line-height: 22px;
-    text-transform: capitalize;
     padding: 0 16px;
   }
 
+  &.capitalize-header .dropdown-header {
+    text-transform: capitalize;
+  }
+
   .separator + .dropdown-header {
     padding-top: 2px;
   }
@@ -337,8 +339,8 @@
 .dropdown-menu-user {
   .avatar {
     float: left;
-    width: 30px;
-    height: 30px;
+    width: 2 * $gl-padding;
+    height: 2 * $gl-padding;
     margin: 0 10px 0 0;
   }
 }
@@ -381,6 +383,7 @@
 .dropdown-menu-selectable {
   a {
     padding-left: 26px;
+    position: relative;
 
     &.is-indeterminate,
     &.is-active {
@@ -406,6 +409,9 @@
 
     &.is-active::before {
       content: "\f00c";
+      position: absolute;
+      top: 50%;
+      transform: translateY(-50%);
     }
   }
 }
diff --git a/app/assets/stylesheets/framework/files.scss b/app/assets/stylesheets/framework/files.scss
index c197bf6b9f578..1dd0e5ab581b4 100644
--- a/app/assets/stylesheets/framework/files.scss
+++ b/app/assets/stylesheets/framework/files.scss
@@ -162,6 +162,18 @@
     &.code {
       padding: 0;
     }
+
+    .list-inline.previews {
+      display: flex;
+      flex-wrap: wrap;
+      justify-content: center;
+      align-content: flex-start;
+      align-items: baseline;
+
+      .preview {
+        padding: $gl-padding;
+      }
+    }
   }
 }
 
diff --git a/app/assets/stylesheets/framework/filters.scss b/app/assets/stylesheets/framework/filters.scss
index 03c206f26a4ae..3f75148f8b8c1 100644
--- a/app/assets/stylesheets/framework/filters.scss
+++ b/app/assets/stylesheets/framework/filters.scss
@@ -114,11 +114,21 @@
     padding-right: 8px;
 
     .fa-close {
-      color: $gl-text-color-disabled;
+      color: $gl-text-color-secondary;
     }
 
     &:hover .fa-close {
-      color: $gl-text-color-secondary;
+      color: $gl-text-color;
+    }
+
+    &.inverted {
+      .fa-close {
+        color: $gl-text-color-secondary-inverted;
+      }
+
+      &:hover .fa-close {
+        color: $gl-text-color-inverted;
+      }
     }
   }
 
diff --git a/app/assets/stylesheets/framework/lists.scss b/app/assets/stylesheets/framework/lists.scss
index cb51323809c87..0b71cf16bd72c 100644
--- a/app/assets/stylesheets/framework/lists.scss
+++ b/app/assets/stylesheets/framework/lists.scss
@@ -261,6 +261,7 @@ ul.controls {
       .avatar-inline {
         margin-left: 0;
         margin-right: 0;
+        margin-bottom: 0;
       }
     }
   }
diff --git a/app/assets/stylesheets/framework/variables.scss b/app/assets/stylesheets/framework/variables.scss
index 60f02921c0a63..8615801700183 100644
--- a/app/assets/stylesheets/framework/variables.scss
+++ b/app/assets/stylesheets/framework/variables.scss
@@ -101,6 +101,8 @@ $gl-font-size: 14px;
 $gl-text-color: rgba(0, 0, 0, .85);
 $gl-text-color-secondary: rgba(0, 0, 0, .55);
 $gl-text-color-disabled: rgba(0, 0, 0, .35);
+$gl-text-color-inverted: rgba(255, 255, 255, 1.0);
+$gl-text-color-secondary-inverted: rgba(255, 255, 255, .85);
 $gl-text-green: $green-600;
 $gl-text-red: $red-500;
 $gl-text-orange: $orange-600;
diff --git a/app/assets/stylesheets/pages/boards.scss b/app/assets/stylesheets/pages/boards.scss
index 353dd13dff41d..46548189b0488 100644
--- a/app/assets/stylesheets/pages/boards.scss
+++ b/app/assets/stylesheets/pages/boards.scss
@@ -236,8 +236,13 @@
     margin-bottom: 5px;
   }
 
-  &.is-active {
+  &.is-active,
+  &.is-active .card-assignee:hover a {
     background-color: $row-hover;
+
+    &:first-child:not(:only-child) {
+      box-shadow: -10px 0 10px 1px $row-hover;
+    }
   }
 
   .label {
@@ -253,7 +258,7 @@
 }
 
 .card-title {
-  margin: 0;
+  margin: 0 30px 0 0;
   font-size: 1em;
   line-height: inherit;
 
@@ -269,10 +274,68 @@
   min-height: 20px;
 
   .card-assignee {
-    margin-left: auto;
-    margin-right: 5px;
-    padding-left: 10px;
+    display: flex;
+    justify-content: flex-end;
+    position: absolute;
+    right: 15px;
     height: 20px;
+    width: 20px;
+
+    .avatar-counter {
+      display: none;
+      vertical-align: middle;
+      line-height: 18px;
+      height: 20px;
+      padding-left: 3px;
+      padding-right: 3px;
+      border-radius: 2em;
+    }
+
+    img {
+      vertical-align: top;
+    }
+
+    a {
+      position: relative;
+      margin-left: -15px;
+    }
+
+    a:nth-child(1) {
+      z-index: 3;
+    }
+
+    a:nth-child(2) {
+      z-index: 2;
+    }
+
+    a:nth-child(3) {
+      z-index: 1;
+    }
+
+    a:nth-child(4) {
+      display: none;
+    }
+
+    &:hover {
+      .avatar-counter {
+        display: inline-block;
+      }
+
+      a {
+        position: static;
+        background-color: $white-light;
+        transition: background-color 0s;
+        margin-left: auto;
+
+        &:nth-child(4) {
+          display: block;
+        }
+
+        &:first-child:not(:only-child) {
+          box-shadow: -10px 0 10px 1px $white-light;
+        }
+      }
+    }
   }
 
   .avatar {
diff --git a/app/assets/stylesheets/pages/cycle_analytics.scss b/app/assets/stylesheets/pages/cycle_analytics.scss
index 403724cd68a84..d29944207c5ba 100644
--- a/app/assets/stylesheets/pages/cycle_analytics.scss
+++ b/app/assets/stylesheets/pages/cycle_analytics.scss
@@ -3,6 +3,25 @@
   margin: 24px auto 0;
   position: relative;
 
+  .landing {
+    margin-top: 10px;
+
+    .inner-content {
+      white-space: normal;
+
+      h4,
+      p {
+        margin: 7px 0 0;
+        max-width: 480px;
+        padding: 0 $gl-padding;
+
+        @media (max-width: $screen-sm-min) {
+          margin: 0 auto;
+        }
+      }
+    }
+  }
+
   .col-headers {
     ul {
       margin: 0;
@@ -175,7 +194,7 @@
     }
 
     .stage-nav-item {
-      display: block;
+      display: flex;
       line-height: 65px;
       border-top: 1px solid transparent;
       border-bottom: 1px solid transparent;
@@ -209,14 +228,10 @@
       }
 
       .stage-nav-item-cell {
-        float: left;
-
-        &.stage-name {
-          width: 65%;
-        }
-
         &.stage-median {
-          width: 35%;
+          margin-left: auto;
+          margin-right: $gl-padding;
+          min-width: calc(35% - #{$gl-padding});
         }
       }
 
diff --git a/app/assets/stylesheets/pages/diff.scss b/app/assets/stylesheets/pages/diff.scss
index feefaad8a15c6..77f2638683a1d 100644
--- a/app/assets/stylesheets/pages/diff.scss
+++ b/app/assets/stylesheets/pages/diff.scss
@@ -570,14 +570,7 @@
 
 .diff-comments-more-count,
 .diff-notes-collapse {
-  background-color: $gray-darkest;
-  color: $white-light;
-  border: 1px solid $white-light;
-  border-radius: 1em;
-  font-family: $regular_font;
-  font-size: 9px;
-  line-height: 17px;
-  text-align: center;
+  @extend .avatar-counter;
 }
 
 .diff-notes-collapse {
diff --git a/app/assets/stylesheets/pages/issuable.scss b/app/assets/stylesheets/pages/issuable.scss
index ad6eb9f6fe040..485ea369f3d17 100644
--- a/app/assets/stylesheets/pages/issuable.scss
+++ b/app/assets/stylesheets/pages/issuable.scss
@@ -95,10 +95,15 @@
 }
 
 .right-sidebar {
-  a {
+  a,
+  .btn-link {
     color: inherit;
   }
 
+  .btn-link {
+    outline: none;
+  }
+
   .issuable-header-text {
     margin-top: 7px;
   }
@@ -215,6 +220,10 @@
       }
     }
 
+    .assign-yourself .btn-link {
+      padding-left: 0;
+    }
+
     .light {
       font-weight: normal;
     }
@@ -239,6 +248,10 @@
       margin-left: 0;
     }
 
+    .assignee .user-list .avatar {
+      margin: 0;
+    }
+
     .username {
       display: block;
       margin-top: 4px;
@@ -301,6 +314,10 @@
         margin-top: 0;
       }
 
+      .sidebar-avatar-counter {
+        padding-top: 2px;
+      }
+
       .todo-undone {
         color: $gl-link-color;
       }
@@ -309,10 +326,15 @@
         display: none;
       }
 
-      .avatar:hover {
+      .avatar:hover,
+      .avatar-counter:hover {
         border-color: $issuable-sidebar-color;
       }
 
+      .avatar-counter:hover {
+        color: $issuable-sidebar-color;
+      }
+
       .btn-clipboard {
         border: none;
         color: $issuable-sidebar-color;
@@ -322,6 +344,17 @@
           color: $gl-text-color;
         }
       }
+
+      &.multiple-users {
+        display: flex;
+        justify-content: center;
+      }
+    }
+
+    .sidebar-avatar-counter {
+      width: 24px;
+      height: 24px;
+      border-radius: 12px;
     }
 
     .sidebar-collapsed-user {
@@ -332,6 +365,37 @@
     .issuable-header-btn {
       display: none;
     }
+
+    .multiple-users {
+      height: 24px;
+      margin-bottom: 17px;
+      margin-top: 4px;
+      padding-bottom: 4px;
+
+      .btn-link {
+        padding: 0;
+        border: 0;
+
+        .avatar {
+          margin: 0;
+        }
+      }
+
+      .btn-link:first-child {
+        position: absolute;
+        left: 10px;
+        z-index: 1;
+      }
+
+      .btn-link:last-child {
+        position: absolute;
+        right: 10px;
+
+        &:hover {
+          text-decoration: none;
+        }
+      }
+    }
   }
 
   a {
@@ -380,17 +444,21 @@
 }
 
 .participants-list {
+  display: flex;
+  flex-wrap: wrap;
+  justify-content: space-between;
   margin: -5px;
 }
 
+.user-list {
+  display: flex;
+  flex-wrap: wrap;
+}
+
 .participants-author {
-  display: inline-block;
+  flex-basis: 14%;
   padding: 5px;
 
-  &:nth-of-type(7n) {
-    padding-right: 0;
-  }
-
   .author_link {
     display: block;
   }
@@ -400,13 +468,39 @@
   }
 }
 
-.participants-more {
+.user-item {
+  display: inline-block;
+  padding: 5px;
+  flex-basis: 20%;
+
+  .user-link {
+    display: inline-block;
+  }
+}
+
+.participants-more,
+.user-list-more {
   margin-top: 5px;
   margin-left: 5px;
 
-  a {
+  a,
+  .btn-link {
     color: $gl-text-color-secondary;
   }
+
+  .btn-link {
+    outline: none;
+    padding: 0;
+  }
+
+  .btn-link:hover {
+    @extend a:hover;
+    text-decoration: none;
+  }
+
+  .btn-link:focus {
+    text-decoration: none;
+  }
 }
 
 .issuable-form-padding-top {
@@ -499,6 +593,19 @@
   }
 }
 
+.issuable-list li,
+.issue-info-container .controls {
+  .avatar-counter {
+    display: inline-block;
+    vertical-align: middle;
+    min-width: 16px;
+    line-height: 14px;
+    height: 16px;
+    padding-left: 2px;
+    padding-right: 2px;
+  }
+}
+
 .time_tracker {
   padding-bottom: 0;
   border-bottom: 0;
diff --git a/app/assets/stylesheets/pages/labels.scss b/app/assets/stylesheets/pages/labels.scss
index e1ef0b029a59a..c10588ac58e6a 100644
--- a/app/assets/stylesheets/pages/labels.scss
+++ b/app/assets/stylesheets/pages/labels.scss
@@ -116,7 +116,7 @@
 }
 
 .manage-labels-list {
-  > li:not(.empty-message) {
+  > li:not(.empty-message):not(.is-not-draggable) {
     background-color: $white-light;
     cursor: move;
     cursor: -webkit-grab;
diff --git a/app/assets/stylesheets/pages/merge_requests.scss b/app/assets/stylesheets/pages/merge_requests.scss
index 10e402f33e266..c5046dbeeff2e 100644
--- a/app/assets/stylesheets/pages/merge_requests.scss
+++ b/app/assets/stylesheets/pages/merge_requests.scss
@@ -482,6 +482,10 @@
   }
 }
 
+.target-branch-select-dropdown-container {
+  position: relative;
+}
+
 .assign-to-me-link {
   padding-left: 12px;
   white-space: nowrap;
diff --git a/app/assets/stylesheets/pages/notes.scss b/app/assets/stylesheets/pages/notes.scss
index f89150ebead7e..69c328d09ffab 100644
--- a/app/assets/stylesheets/pages/notes.scss
+++ b/app/assets/stylesheets/pages/notes.scss
@@ -57,6 +57,25 @@ ul.notes {
     position: relative;
     border-bottom: 1px solid $white-normal;
 
+    &.being-posted {
+      pointer-events: none;
+      opacity: 0.5;
+
+      .dummy-avatar {
+        display: inline-block;
+        height: 40px;
+        width: 40px;
+        border-radius: 50%;
+        background-color: $kdb-border;
+        border: 1px solid darken($kdb-border, 25%);
+      }
+
+      .note-headline-light,
+      .fa-spinner {
+        margin-left: 3px;
+      }
+    }
+
     &.note-discussion {
       &.timeline-entry {
         padding: 14px 10px;
@@ -111,12 +130,6 @@ ul.notes {
     }
 
     .note-header {
-      padding-bottom: 8px;
-      padding-right: 20px;
-
-      @media (min-width: $screen-sm-min) {
-        padding-right: 0;
-      }
 
       @media (max-width: $screen-xs-min) {
         .inline {
@@ -365,10 +378,15 @@ ul.notes {
 .note-header {
   display: flex;
   justify-content: space-between;
+
+  @media (max-width: $screen-xs-max) {
+    flex-flow: row wrap;
+  }
 }
 
 .note-header-info {
   min-width: 0;
+  padding-bottom: 5px;
 }
 
 .note-headline-light {
@@ -416,6 +434,11 @@ ul.notes {
   margin-left: 10px;
   color: $gray-darkest;
 
+  @media (max-width: $screen-xs-max) {
+    float: none;
+    margin-left: 0;
+  }
+
   .note-action-button {
     margin-left: 8px;
   }
@@ -687,6 +710,10 @@ ul.notes {
   }
 }
 
+.discussion-notes .flash-container {
+  margin-bottom: 0;
+}
+
 // Merge request notes in diffs
 .diff-file {
   // Diff is side by side
diff --git a/app/assets/stylesheets/pages/pipelines.scss b/app/assets/stylesheets/pages/pipelines.scss
index 60f79665135af..106b0ee45d2b5 100644
--- a/app/assets/stylesheets/pages/pipelines.scss
+++ b/app/assets/stylesheets/pages/pipelines.scss
@@ -277,6 +277,7 @@
   .stage-container {
     display: inline-block;
     position: relative;
+    vertical-align: middle;
     height: 22px;
     margin: 3px 6px 3px 0;
 
@@ -320,6 +321,32 @@
   }
 }
 
+.build-failures {
+  .build-state {
+    padding: 20px 2px;
+
+    .build-name {
+      float: right;
+      font-weight: 500;
+    }
+
+    .ci-status-icon-failed svg {
+      vertical-align: middle;
+    }
+
+    .stage {
+      color: $gl-text-color-secondary;
+      font-weight: 500;
+      vertical-align: middle;
+    }
+  }
+
+  .build-log {
+    border: none;
+    line-height: initial;
+  }
+}
+
 // Pipeline graph
 .pipeline-graph {
   width: 100%;
diff --git a/app/assets/stylesheets/pages/todos.scss b/app/assets/stylesheets/pages/todos.scss
index a39815319f377..de652a7936988 100644
--- a/app/assets/stylesheets/pages/todos.scss
+++ b/app/assets/stylesheets/pages/todos.scss
@@ -54,8 +54,9 @@
     background-color: $white-light;
 
     &:hover {
-      border-color: $white-dark;
+      border-color: $white-normal;
       background-color: $gray-light;
+      border-top: 1px solid transparent;
 
       .todo-avatar,
       .todo-item {
diff --git a/app/controllers/admin/application_settings_controller.rb b/app/controllers/admin/application_settings_controller.rb
index 0bc99451d5e39..8fdb887a7f933 100644
--- a/app/controllers/admin/application_settings_controller.rb
+++ b/app/controllers/admin/application_settings_controller.rb
@@ -134,6 +134,8 @@ def application_setting_params_ce
       :signup_enabled,
       :sentry_dsn,
       :sentry_enabled,
+      :clientside_sentry_dsn,
+      :clientside_sentry_enabled,
       :send_user_confirmation_email,
       :shared_runners_enabled,
       :shared_runners_text,
diff --git a/app/controllers/admin/services_controller.rb b/app/controllers/admin/services_controller.rb
index 37a1a23178eb7..4c3d336b3aff8 100644
--- a/app/controllers/admin/services_controller.rb
+++ b/app/controllers/admin/services_controller.rb
@@ -16,6 +16,8 @@ def edit
 
   def update
     if service.update_attributes(service_params[:service])
+      PropagateServiceTemplateWorker.perform_async(service.id) if service.active?
+
       redirect_to admin_application_settings_services_path,
         notice: 'Application settings saved successfully'
     else
diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index 1f8a12a44d004..09f84b7a5997f 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -21,6 +21,8 @@ class ApplicationController < ActionController::Base
   before_action :configure_permitted_parameters, if: :devise_controller?
   before_action :require_email, unless: :devise_controller?
 
+  around_action :set_locale
+
   protect_from_forgery with: :exception
 
   helper_method :can?, :current_application_settings
@@ -273,4 +275,12 @@ def gitlab_project_import_enabled?
   def u2f_app_id
     request.base_url
   end
+
+  def set_locale
+    Gitlab::I18n.set_locale(current_user)
+
+    yield
+  ensure
+    Gitlab::I18n.reset_locale
+  end
 end
diff --git a/app/controllers/concerns/issuable_actions.rb b/app/controllers/concerns/issuable_actions.rb
index 3ccf2a9ce33e1..b199f18da1e6d 100644
--- a/app/controllers/concerns/issuable_actions.rb
+++ b/app/controllers/concerns/issuable_actions.rb
@@ -66,6 +66,7 @@ def bulk_update_params
       :milestone_id,
       :state_event,
       :subscription_event,
+      assignee_ids: [],
       label_ids: [],
       add_label_ids: [],
       remove_label_ids: []
diff --git a/app/controllers/concerns/issuable_collections.rb b/app/controllers/concerns/issuable_collections.rb
index ebc494051f8c6..5fc0702ab990d 100644
--- a/app/controllers/concerns/issuable_collections.rb
+++ b/app/controllers/concerns/issuable_collections.rb
@@ -43,7 +43,7 @@ def issuable_meta_data(issuable_collection, collection_type)
   end
 
   def issues_collection
-    issues_finder.execute.preload(:project, :author, :assignee, :labels, :milestone, project: :namespace)
+    issues_finder.execute.preload(:project, :author, :assignees, :labels, :milestone, project: :namespace)
   end
 
   def merge_requests_collection
diff --git a/app/controllers/dashboard/labels_controller.rb b/app/controllers/dashboard/labels_controller.rb
index d5031da867af6..dd1d46a68c71c 100644
--- a/app/controllers/dashboard/labels_controller.rb
+++ b/app/controllers/dashboard/labels_controller.rb
@@ -3,7 +3,7 @@ def index
     labels = LabelsFinder.new(current_user).execute
 
     respond_to do |format|
-      format.json { render json: labels.as_json(only: [:id, :title, :color]) }
+      format.json { render json: LabelSerializer.new.represent_appearance(labels) }
     end
   end
 end
diff --git a/app/controllers/groups/labels_controller.rb b/app/controllers/groups/labels_controller.rb
index facb25525b5ef..3fa0516fb0ce4 100644
--- a/app/controllers/groups/labels_controller.rb
+++ b/app/controllers/groups/labels_controller.rb
@@ -15,7 +15,7 @@ def index
 
       format.json do
         available_labels = LabelsFinder.new(current_user, group_id: @group.id).execute
-        render json: available_labels.as_json(only: [:id, :title, :color])
+        render json: LabelSerializer.new.represent_appearance(available_labels)
       end
     end
   end
diff --git a/app/controllers/omniauth_callbacks_controller.rb b/app/controllers/omniauth_callbacks_controller.rb
index 4a719ebdbd61d..a27848de3a6b8 100644
--- a/app/controllers/omniauth_callbacks_controller.rb
+++ b/app/controllers/omniauth_callbacks_controller.rb
@@ -67,7 +67,7 @@ def saml
   def omniauth_error
     @provider = params[:provider]
     @error = params[:error]
-    render 'errors/omniauth_error', layout: "errors", status: 422
+    render 'errors/omniauth_error', layout: "oauth_error", status: 422
   end
 
   def cas3
diff --git a/app/controllers/profiles_controller.rb b/app/controllers/profiles_controller.rb
index 987b95e89b9e6..57e23cea00eb5 100644
--- a/app/controllers/profiles_controller.rb
+++ b/app/controllers/profiles_controller.rb
@@ -85,7 +85,8 @@ def user_params
       :twitter,
       :username,
       :website_url,
-      :organization
+      :organization,
+      :preferred_language
     )
   end
 end
diff --git a/app/controllers/projects/artifacts_controller.rb b/app/controllers/projects/artifacts_controller.rb
index a13588b4218c2..1224e9503c93d 100644
--- a/app/controllers/projects/artifacts_controller.rb
+++ b/app/controllers/projects/artifacts_controller.rb
@@ -1,11 +1,13 @@
 class Projects::ArtifactsController < Projects::ApplicationController
   include ExtractsPath
+  include RendersBlob
 
   layout 'project'
   before_action :authorize_read_build!
   before_action :authorize_update_build!, only: [:keep]
   before_action :extract_ref_name_and_path
   before_action :validate_artifacts!
+  before_action :set_path_and_entry, only: [:file, :raw]
 
   def download
     if artifacts_file.file_storage?
@@ -24,15 +26,24 @@ def browse
   end
 
   def file
-    entry = build.artifacts_metadata_entry(params[:path])
+    blob = @entry.blob
+    override_max_blob_size(blob)
 
-    if entry.exists?
-      send_artifacts_entry(build, entry)
-    else
-      render_404
+    respond_to do |format|
+      format.html do
+        render 'file'
+      end
+
+      format.json do
+        render_blob_json(blob)
+      end
     end
   end
 
+  def raw
+    send_artifacts_entry(build, @entry)
+  end
+
   def keep
     build.keep_artifacts!
     redirect_to namespace_project_build_path(project.namespace, project, build)
@@ -81,4 +92,11 @@ def build_from_ref
   def artifacts_file
     @artifacts_file ||= build.artifacts_file
   end
+
+  def set_path_and_entry
+    @path = params[:path]
+    @entry = build.artifacts_metadata_entry(@path)
+
+    render_404 unless @entry.exists?
+  end
 end
diff --git a/app/controllers/projects/boards/issues_controller.rb b/app/controllers/projects/boards/issues_controller.rb
index 1093414462ccf..8085a9c450b35 100644
--- a/app/controllers/projects/boards/issues_controller.rb
+++ b/app/controllers/projects/boards/issues_controller.rb
@@ -82,7 +82,7 @@ def serialize_as_json(resource)
           labels: true,
           only: [:id, :iid, :title, :confidential, :due_date, :relative_position],
           include: {
-            assignee: { only: [:id, :name, :username], methods: [:avatar_url] },
+            assignees: { only: [:id, :name, :username], methods: [:avatar_url] },
             milestone: { only: [:id, :title] }
           },
           user: current_user
diff --git a/app/controllers/projects/deploy_keys_controller.rb b/app/controllers/projects/deploy_keys_controller.rb
index 5e0d7fd02a42c..d46c7ac9b985d 100644
--- a/app/controllers/projects/deploy_keys_controller.rb
+++ b/app/controllers/projects/deploy_keys_controller.rb
@@ -8,7 +8,12 @@ class Projects::DeployKeysController < Projects::ApplicationController
   layout "project_settings"
 
   def index
-    redirect_to_repository_settings(@project)
+    respond_to do |format|
+      format.html { redirect_to_repository_settings(@project) }
+      format.json do
+        render json: Projects::Settings::DeployKeysPresenter.new(@project, current_user: current_user).as_json
+      end
+    end
   end
 
   def new
@@ -23,6 +28,7 @@ def create
     else
       log_audit_event(@key.title, action: :create)
     end
+
     redirect_to_repository_settings(@project)
   end
 
@@ -31,7 +37,10 @@ def enable
     Projects::EnableDeployKeyService.new(@project, current_user, params).execute
     log_audit_event(@key.title, action: :create)
 
-    redirect_to_repository_settings(@project)
+    respond_to do |format|
+      format.html { redirect_to_repository_settings(@project) }
+      format.json { head :ok }
+    end
   end
 
   def disable
@@ -42,7 +51,10 @@ def disable
     deploy_key_project.destroy!
     log_audit_event(@key.title, action: :destroy)
 
-    redirect_to_repository_settings(@project)
+    respond_to do |format|
+      format.html { redirect_to_repository_settings(@project) }
+      format.json { head :ok }
+    end
   end
 
   protected
diff --git a/app/controllers/projects/issues_controller.rb b/app/controllers/projects/issues_controller.rb
index 5672ff7e1dcaa..86cd3c5cd699d 100644
--- a/app/controllers/projects/issues_controller.rb
+++ b/app/controllers/projects/issues_controller.rb
@@ -68,7 +68,7 @@ def index
 
   def new
     params[:issue] ||= ActionController::Parameters.new(
-      assignee_id: ""
+      assignee_ids: ""
     )
     build_params = issue_params.merge(
       merge_request_to_resolve_discussions_of: params[:merge_request_to_resolve_discussions_of],
@@ -151,7 +151,7 @@ def update
         if @issue.valid?
           render json: @issue.to_json(methods: [:task_status, :task_status_short],
                                       include: { milestone: {},
-                                                 assignee: { only: [:name, :username], methods: [:avatar_url] },
+                                                 assignees: { only: [:id, :name, :username], methods: [:avatar_url] },
                                                  labels: { methods: :text_color } })
         else
           render json: { errors: @issue.errors.full_messages }, status: :unprocessable_entity
@@ -291,8 +291,8 @@ def redirect_old
 
   def issue_params
     params.require(:issue).permit(
-      :title, :assignee_id, :position, :description, :confidential, :weight,
-      :milestone_id, :due_date, :state_event, :task_num, :lock_version, label_ids: []
+      :title, :position, :description, :confidential, :weight,
+      :milestone_id, :due_date, :state_event, :task_num, :lock_version, label_ids: [], assignee_ids: [],
     )
   end
 
diff --git a/app/controllers/projects/labels_controller.rb b/app/controllers/projects/labels_controller.rb
index 2f55ba4e700b0..71bfb7163da14 100644
--- a/app/controllers/projects/labels_controller.rb
+++ b/app/controllers/projects/labels_controller.rb
@@ -19,7 +19,7 @@ def index
     respond_to do |format|
       format.html
       format.json do
-        render json: @available_labels.as_json(only: [:id, :title, :color])
+        render json: LabelSerializer.new.represent_appearance(@available_labels)
       end
     end
   end
diff --git a/app/controllers/projects/pipelines_controller.rb b/app/controllers/projects/pipelines_controller.rb
index 454b8ee17af2b..f9adedcb074b8 100644
--- a/app/controllers/projects/pipelines_controller.rb
+++ b/app/controllers/projects/pipelines_controller.rb
@@ -1,11 +1,13 @@
 class Projects::PipelinesController < Projects::ApplicationController
   before_action :pipeline, except: [:index, :new, :create, :charts]
-  before_action :commit, only: [:show, :builds]
+  before_action :commit, only: [:show, :builds, :failures]
   before_action :authorize_read_pipeline!
   before_action :authorize_create_pipeline!, only: [:new, :create]
   before_action :authorize_update_pipeline!, only: [:retry, :cancel]
   before_action :builds_enabled, only: :charts
 
+  wrap_parameters Ci::Pipeline
+
   def index
     @scope = params[:scope]
     @pipelines = PipelinesFinder
@@ -67,10 +69,14 @@ def show
   end
 
   def builds
-    respond_to do |format|
-      format.html do
-        render 'show'
-      end
+    render_show
+  end
+
+  def failures
+    if @pipeline.statuses.latest.failed.present?
+      render_show
+    else
+      redirect_to pipeline_path(@pipeline)
     end
   end
 
@@ -92,13 +98,25 @@ def stage
   def retry
     pipeline.retry_failed(current_user)
 
-    redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
+    respond_to do |format|
+      format.html do
+        redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
+      end
+
+      format.json { head :no_content }
+    end
   end
 
   def cancel
     pipeline.cancel_running
 
-    redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
+    respond_to do |format|
+      format.html do
+        redirect_back_or_default default: namespace_project_pipelines_path(project.namespace, project)
+      end
+
+      format.json { head :no_content }
+    end
   end
 
   def charts
@@ -111,6 +129,14 @@ def charts
 
   private
 
+  def render_show
+    respond_to do |format|
+      format.html do
+        render 'show'
+      end
+    end
+  end
+
   def create_params
     params.require(:pipeline).permit(:ref)
   end
diff --git a/app/finders/issuable_finder.rb b/app/finders/issuable_finder.rb
index 82994dd43ac0c..caa0d128630f6 100644
--- a/app/finders/issuable_finder.rb
+++ b/app/finders/issuable_finder.rb
@@ -236,7 +236,7 @@ def by_scope(items)
     when 'created-by-me', 'authored'
       items.where(author_id: current_user.id)
     when 'assigned-to-me'
-      items.where(assignee_id: current_user.id)
+      items.assigned_to(current_user)
     else
       items
     end
diff --git a/app/finders/issues_finder.rb b/app/finders/issues_finder.rb
index 34ed2c1fce8a9..6d2c0277fec81 100644
--- a/app/finders/issues_finder.rb
+++ b/app/finders/issues_finder.rb
@@ -26,17 +26,28 @@ def init_collection
     IssuesFinder.not_restricted_by_confidentiality(current_user)
   end
 
+  def by_assignee(items)
+    if assignee
+      items.assigned_to(assignee)
+    elsif no_assignee?
+      items.unassigned
+    elsif assignee_id? || assignee_username? # assignee not found
+      items.none
+    else
+      items
+    end
+  end
+
   def self.not_restricted_by_confidentiality(user)
-    return Issue.where('issues.confidential IS NULL OR issues.confidential IS FALSE') if user.blank?
+    return Issue.where('issues.confidential IS NOT TRUE') if user.blank?
 
     return Issue.all if user.admin_or_auditor?
 
     Issue.where('
-      issues.confidential IS NULL
-      OR issues.confidential IS FALSE
+      issues.confidential IS NOT TRUE
       OR (issues.confidential = TRUE
         AND (issues.author_id = :user_id
-          OR issues.assignee_id = :user_id
+          OR EXISTS (SELECT TRUE FROM issue_assignees WHERE user_id = :user_id AND issue_id = issues.id)
           OR issues.project_id IN(:project_ids)))',
       user_id: user.id,
       project_ids: user.authorized_projects(Gitlab::Access::REPORTER).select(:id))
diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb
index c410a1a46e218..be097fe854cfa 100644
--- a/app/helpers/application_helper.rb
+++ b/app/helpers/application_helper.rb
@@ -183,16 +183,16 @@ def time_ago_with_tooltip(time, placement: 'top', html_class: '', short_format:
     element
   end
 
-  def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago', include_author: false)
-    return if object.updated_at == object.created_at
+  def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago', exclude_author: false)
+    return if object.last_edited_at == object.created_at || object.last_edited_at.blank?
 
-    content_tag :small, class: "edited-text" do
-      output = content_tag(:span, "Edited ")
-      output << time_ago_with_tooltip(object.updated_at, placement: placement, html_class: html_class)
+    content_tag :small, class: 'edited-text' do
+      output = content_tag(:span, 'Edited ')
+      output << time_ago_with_tooltip(object.last_edited_at, placement: placement, html_class: html_class)
 
-      if include_author && object.updated_by && object.updated_by != object.author
-        output << content_tag(:span, " by ")
-        output << link_to_member(object.project, object.updated_by, avatar: false, author_class: nil)
+      if !exclude_author && object.last_edited_by
+        output << content_tag(:span, ' by ')
+        output << link_to_member(object.project, object.last_edited_by, avatar: false, author_class: nil)
       end
 
       output
diff --git a/app/helpers/blob_helper.rb b/app/helpers/blob_helper.rb
index 37b6f4ad5cc6b..af430270ae424 100644
--- a/app/helpers/blob_helper.rb
+++ b/app/helpers/blob_helper.rb
@@ -119,7 +119,9 @@ def blob_icon(mode, name)
   end
 
   def blob_raw_url
-    if @snippet
+    if @build && @entry
+      raw_namespace_project_build_artifacts_path(@project.namespace, @project, @build, path: @entry.path)
+    elsif @snippet
       if @snippet.project_id
         raw_namespace_project_snippet_path(@project.namespace, @project, @snippet)
       else
@@ -250,6 +252,8 @@ def blob_render_error_reason(viewer)
       case viewer.blob.external_storage
       when :lfs
         'it is stored in LFS'
+      when :build_artifact
+        'it is stored as a job artifact'
       else
         'it is stored externally'
       end
diff --git a/app/helpers/boards_helper.rb b/app/helpers/boards_helper.rb
index 1bd59822a54fb..2a9dc8b6858bd 100644
--- a/app/helpers/boards_helper.rb
+++ b/app/helpers/boards_helper.rb
@@ -10,6 +10,7 @@ def board_data
       issue_link_base: namespace_project_issues_path(@project.namespace, @project),
       root_path: root_path,
       bulk_update_path: bulk_update_namespace_project_issues_path(@project.namespace, @project),
+      default_avatar: image_path(default_avatar)
     }
   end
 
diff --git a/app/helpers/builds_helper.rb b/app/helpers/builds_helper.rb
index 2fcb7a59fc32a..2eb2c6c738945 100644
--- a/app/helpers/builds_helper.rb
+++ b/app/helpers/builds_helper.rb
@@ -1,4 +1,16 @@
 module BuildsHelper
+  def build_summary(build, skip: false)
+    if build.has_trace?
+      if skip
+        link_to "View job trace", pipeline_build_url(build.pipeline, build)
+      else
+        build.trace.html(last_lines: 10).html_safe
+      end
+    else
+      "No job trace"
+    end
+  end
+
   def sidebar_build_class(build, current_build)
     build_class = ''
     build_class += ' active' if build.id === current_build.id
diff --git a/app/helpers/form_helper.rb b/app/helpers/form_helper.rb
index 1182939f65695..53962b846185b 100644
--- a/app/helpers/form_helper.rb
+++ b/app/helpers/form_helper.rb
@@ -15,4 +15,36 @@ def form_errors(model)
         end
     end
   end
+
+  def issue_dropdown_options(issuable, has_multiple_assignees = true)
+    options = {
+      toggle_class: 'js-user-search js-assignee-search js-multiselect js-save-user-data',
+      title: 'Select assignee',
+      filter: true,
+      dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-assignee',
+      placeholder: 'Search users',
+      data: {
+        first_user: current_user&.username,
+        null_user: true,
+        current_user: true,
+        project_id: issuable.project.try(:id),
+        field_name: "#{issuable.class.model_name.param_key}[assignee_ids][]",
+        default_label: 'Assignee',
+        'max-select': 1,
+        'dropdown-header': 'Assignee',
+        multi_select: true,
+        'input-meta': 'name',
+        'always-show-selectbox': true,
+        current_user_info: current_user.to_json(only: [:id, :name])
+      }
+    }
+
+    if has_multiple_assignees
+      options[:title] = 'Select assignee(s)'
+      options[:data][:'dropdown-header'] = 'Assignee(s)'
+      options[:data].delete(:'max-select')
+    end
+
+    options
+  end
 end
diff --git a/app/helpers/gitlab_routing_helper.rb b/app/helpers/gitlab_routing_helper.rb
index 1f02f8b99af1f..3769830de2a1f 100644
--- a/app/helpers/gitlab_routing_helper.rb
+++ b/app/helpers/gitlab_routing_helper.rb
@@ -216,6 +216,8 @@ def artifacts_action_path(path, project, build)
       browse_namespace_project_build_artifacts_path(*args)
     when 'file'
       file_namespace_project_build_artifacts_path(*args)
+    when 'raw'
+      raw_namespace_project_build_artifacts_path(*args)
     end
   end
 
diff --git a/app/helpers/issuables_helper.rb b/app/helpers/issuables_helper.rb
index 69c0a61bd6555..f21d33a96cf98 100644
--- a/app/helpers/issuables_helper.rb
+++ b/app/helpers/issuables_helper.rb
@@ -63,6 +63,16 @@ def template_dropdown_tag(issuable, &block)
     end
   end
 
+  def users_dropdown_label(selected_users)
+    if selected_users.length == 0
+      "Unassigned"
+    elsif selected_users.length == 1
+      selected_users[0].name
+    else
+      "#{selected_users[0].name} + #{selected_users.length - 1} more"
+    end
+  end
+
   def user_dropdown_label(user_id, default_label)
     return default_label if user_id.nil?
     return "Unassigned" if user_id == "0"
diff --git a/app/helpers/sorting_helper.rb b/app/helpers/sorting_helper.rb
index 75a2d24435e4e..c61170d711c2a 100644
--- a/app/helpers/sorting_helper.rb
+++ b/app/helpers/sorting_helper.rb
@@ -72,6 +72,14 @@ def branches_sort_options_hash
     }
   end
 
+  def tags_sort_options_hash
+    {
+      sort_value_name => sort_title_name,
+      sort_value_recently_updated => sort_title_recently_updated,
+      sort_value_oldest_updated => sort_title_oldest_updated
+    }
+  end
+
   def sort_title_priority
     'Priority'
   end
diff --git a/app/helpers/system_note_helper.rb b/app/helpers/system_note_helper.rb
index 419729d88490d..ac4eb8598b777 100644
--- a/app/helpers/system_note_helper.rb
+++ b/app/helpers/system_note_helper.rb
@@ -1,6 +1,7 @@
 module SystemNoteHelper
   ICON_NAMES_BY_ACTION = {
     'commit' => 'icon_commit',
+    'description' => 'icon_edit',
     'merge' => 'icon_merge',
     'merged' => 'icon_merged',
     'opened' => 'icon_status_open',
diff --git a/app/mailers/emails/issues.rb b/app/mailers/emails/issues.rb
index d64e48f774b92..0f84784129511 100644
--- a/app/mailers/emails/issues.rb
+++ b/app/mailers/emails/issues.rb
@@ -11,10 +11,12 @@ def new_mention_in_issue_email(recipient_id, issue_id, updated_by_user_id)
       mail_answer_thread(@issue, issue_thread_options(updated_by_user_id, recipient_id))
     end
 
-    def reassigned_issue_email(recipient_id, issue_id, previous_assignee_id, updated_by_user_id)
+    def reassigned_issue_email(recipient_id, issue_id, previous_assignee_ids, updated_by_user_id)
       setup_issue_mail(issue_id, recipient_id)
 
-      @previous_assignee = User.find_by(id: previous_assignee_id) if previous_assignee_id
+      @previous_assignees = []
+      @previous_assignees = User.where(id: previous_assignee_ids) if previous_assignee_ids.any?
+
       mail_answer_thread(@issue, issue_thread_options(updated_by_user_id, recipient_id))
     end
 
diff --git a/app/models/application_setting.rb b/app/models/application_setting.rb
index e4a17186ff7eb..17338cea73e84 100644
--- a/app/models/application_setting.rb
+++ b/app/models/application_setting.rb
@@ -63,6 +63,10 @@ class ApplicationSetting < ActiveRecord::Base
             presence: true,
             if: :sentry_enabled
 
+  validates :clientside_sentry_dsn,
+            presence: true,
+            if: :clientside_sentry_enabled
+
   validates :akismet_api_key,
             presence: true,
             if: :akismet_enabled
diff --git a/app/models/blob.rb b/app/models/blob.rb
index a4fae22a0c459..eaf0b71312288 100644
--- a/app/models/blob.rb
+++ b/app/models/blob.rb
@@ -26,6 +26,7 @@ class Blob < SimpleDelegator
 
     BlobViewer::Image,
     BlobViewer::Sketch,
+    BlobViewer::Balsamiq,
 
     BlobViewer::Video,
 
diff --git a/app/models/blob_viewer/balsamiq.rb b/app/models/blob_viewer/balsamiq.rb
new file mode 100644
index 0000000000000..f982521db9973
--- /dev/null
+++ b/app/models/blob_viewer/balsamiq.rb
@@ -0,0 +1,12 @@
+module BlobViewer
+  class Balsamiq < Base
+    include Rich
+    include ClientSide
+
+    self.partial_name = 'balsamiq'
+    self.extensions = %w(bmpr)
+    self.binary = true
+    self.switcher_icon = 'file-image-o'
+    self.switcher_title = 'preview'
+  end
+end
diff --git a/app/models/ci/artifact_blob.rb b/app/models/ci/artifact_blob.rb
new file mode 100644
index 0000000000000..b35febc9ac5a5
--- /dev/null
+++ b/app/models/ci/artifact_blob.rb
@@ -0,0 +1,35 @@
+module Ci
+  class ArtifactBlob
+    include BlobLike
+
+    attr_reader :entry
+
+    def initialize(entry)
+      @entry = entry
+    end
+
+    delegate :name, :path, to: :entry
+
+    def id
+      Digest::SHA1.hexdigest(path)
+    end
+
+    def size
+      entry.metadata[:size]
+    end
+
+    def data
+      "Build artifact #{path}"
+    end
+
+    def mode
+      entry.metadata[:mode]
+    end
+
+    def external_storage
+      :build_artifact
+    end
+
+    alias_method :external_size, :size
+  end
+end
diff --git a/app/models/concerns/elastic/issues_search.rb b/app/models/concerns/elastic/issues_search.rb
index 6fd6156ce3c24..181ade2f5e700 100644
--- a/app/models/concerns/elastic/issues_search.rb
+++ b/app/models/concerns/elastic/issues_search.rb
@@ -17,7 +17,14 @@ module IssuesSearch
         indexes :state,       type: :text
         indexes :project_id,  type: :integer
         indexes :author_id,   type: :integer
+
+        # The field assignee_id does not exist in issues table anymore.
+        # Nevertheless we'll keep this field as is because we don't want users to rebuild index
+        # + the ES treats arrays transparently so
+        # to any integer field you can write any array of integers and you don't have to change mapping.
+        # More over you can query those items just like a single integer value.
         indexes :assignee_id, type: :integer
+
         indexes :confidential, type: :boolean
       end
 
@@ -26,10 +33,12 @@ def as_indexed_json(options = {})
 
         # We don't use as_json(only: ...) because it calls all virtual and serialized attributtes
         # https://gitlab.com/gitlab-org/gitlab-ee/issues/349
-        [:id, :iid, :title, :description, :created_at, :updated_at, :state, :project_id, :author_id, :assignee_id, :confidential].each do |attr|
+        [:id, :iid, :title, :description, :created_at, :updated_at, :state, :project_id, :author_id, :confidential].each do |attr|
           data[attr.to_s] = safely_read_attribute_for_elasticsearch(attr)
         end
 
+        data['assignee_id'] = safely_read_attribute_for_elasticsearch(:assignee_ids)
+
         data
       end
 
diff --git a/app/models/concerns/elastic/notes_search.rb b/app/models/concerns/elastic/notes_search.rb
index 5576bff63dbb8..7e524a59bf225 100644
--- a/app/models/concerns/elastic/notes_search.rb
+++ b/app/models/concerns/elastic/notes_search.rb
@@ -34,7 +34,7 @@ def as_indexed_json(options = {})
 
         if noteable.is_a?(Issue)
           data['issue'] = {
-            assignee_id: noteable.assignee_id,
+            assignee_id: noteable.assignee_ids,
             author_id: noteable.author_id,
             confidential: noteable.confidential
           }
diff --git a/app/models/concerns/issuable.rb b/app/models/concerns/issuable.rb
index b61b5585ca1fd..7e843207fdd4c 100644
--- a/app/models/concerns/issuable.rb
+++ b/app/models/concerns/issuable.rb
@@ -26,8 +26,8 @@ module Issuable
     cache_markdown_field :description, issuable_state_filter_enabled: true
 
     belongs_to :author, class_name: "User"
-    belongs_to :assignee, class_name: "User"
     belongs_to :updated_by, class_name: "User"
+    belongs_to :last_edited_by, class_name: 'User'
     belongs_to :milestone
     has_many :notes, as: :noteable, inverse_of: :noteable, dependent: :destroy do
       def authors_loaded?
@@ -65,11 +65,8 @@ def award_emojis_loaded?
     validates :title, presence: true, length: { maximum: 255 }
 
     scope :authored, ->(user) { where(author_id: user) }
-    scope :assigned_to, ->(u) { where(assignee_id: u.id)}
     scope :recent, -> { reorder(id: :desc) }
     scope :order_position_asc, -> { reorder(position: :asc) }
-    scope :assigned, -> { where("assignee_id IS NOT NULL") }
-    scope :unassigned, -> { where("assignee_id IS NULL") }
     scope :of_projects, ->(ids) { where(project_id: ids) }
     scope :of_milestones, ->(ids) { where(milestone_id: ids) }
     scope :with_milestone, ->(title) { left_joins_milestones.where(milestones: { title: title }) }
@@ -95,23 +92,14 @@ def award_emojis_loaded?
     attr_mentionable :description
 
     participant :author
-    participant :assignee
     participant :notes_with_associations
 
     strip_attributes :title
 
     acts_as_paranoid
 
-    after_save :update_assignee_cache_counts, if: :assignee_id_changed?
     after_save :record_metrics, unless: :imported?
 
-    def update_assignee_cache_counts
-      # make sure we flush the cache for both the old *and* new assignees(if they exist)
-      previous_assignee = User.find_by_id(assignee_id_was) if assignee_id_was
-      previous_assignee&.update_cache_counts
-      assignee&.update_cache_counts
-    end
-
     # We want to use optimistic lock for cases when only title or description are involved
     # http://api.rubyonrails.org/classes/ActiveRecord/Locking/Optimistic.html
     def locking_enabled?
@@ -248,10 +236,6 @@ def new?
     today? && created_at == updated_at
   end
 
-  def is_being_reassigned?
-    assignee_id_changed?
-  end
-
   def open?
     opened? || reopened?
   end
@@ -280,7 +264,11 @@ def to_hook_data(user)
       # DEPRECATED
       repository: project.hook_attrs.slice(:name, :url, :description, :homepage)
     }
-    hook_data[:assignee] = assignee.hook_attrs if assignee
+    if self.is_a?(Issue)
+      hook_data[:assignees] = assignees.map(&:hook_attrs) if assignees.any?
+    else
+      hook_data[:assignee] = assignee.hook_attrs if assignee
+    end
 
     hook_data
   end
@@ -342,11 +330,6 @@ def can_move?(*)
     false
   end
 
-  def assignee_or_author?(user)
-    # We're comparing IDs here so we don't need to load any associations.
-    author_id == user.id || assignee_id == user.id
-  end
-
   def record_metrics
     metrics = self.metrics || create_metrics
     metrics.record!
diff --git a/app/models/concerns/milestoneish.rb b/app/models/concerns/milestoneish.rb
index f449229864d44..a3472af5c55c9 100644
--- a/app/models/concerns/milestoneish.rb
+++ b/app/models/concerns/milestoneish.rb
@@ -40,7 +40,7 @@ def elapsed_days
   def issues_visible_to_user(user)
     memoize_per_user(user, :issues_visible_to_user) do
       IssuesFinder.new(user, issues_finder_params)
-        .execute.where(milestone_id: milestoneish_ids)
+        .execute.includes(:assignees).where(milestone_id: milestoneish_ids)
     end
   end
 
diff --git a/app/models/global_milestone.rb b/app/models/global_milestone.rb
index 0afbca2cb325e..538615130a762 100644
--- a/app/models/global_milestone.rb
+++ b/app/models/global_milestone.rb
@@ -36,7 +36,7 @@ def self.states_count(projects)
     closed = count_by_state(milestones_by_state_and_title, 'closed')
     all = milestones_by_state_and_title.map { |(_, title), _| title }.uniq.count
 
-    { 
+    {
       opened: opened,
       closed: closed,
       all: all
@@ -86,7 +86,7 @@ def closed?
   end
 
   def issues
-    @issues ||= Issue.of_milestones(milestoneish_ids).includes(:project, :assignee, :labels)
+    @issues ||= Issue.of_milestones(milestoneish_ids).includes(:project, :assignees, :labels)
   end
 
   def merge_requests
@@ -94,7 +94,7 @@ def merge_requests
   end
 
   def participants
-    @participants ||= milestones.includes(:participants).map(&:participants).flatten.compact.uniq
+    @participants ||= milestones.map(&:participants).flatten.uniq
   end
 
   def labels
diff --git a/app/models/issue.rb b/app/models/issue.rb
index 28a6c1173b78e..06da4c7a2b5b4 100644
--- a/app/models/issue.rb
+++ b/app/models/issue.rb
@@ -32,10 +32,17 @@ class Issue < ActiveRecord::Base
 
   has_many :merge_requests_closing_issues, class_name: 'MergeRequestsClosingIssues', dependent: :delete_all
 
+  has_many :issue_assignees
+  has_many :assignees, class_name: "User", through: :issue_assignees
+
   validates :project, presence: true
 
   scope :in_projects, ->(project_ids) { where(project_id: project_ids) }
 
+  scope :assigned, -> { where('EXISTS (SELECT TRUE FROM issue_assignees WHERE issue_id = issues.id)') }
+  scope :unassigned, -> { where('NOT EXISTS (SELECT TRUE FROM issue_assignees WHERE issue_id = issues.id)') }
+  scope :assigned_to, ->(u) { where('EXISTS (SELECT TRUE FROM issue_assignees WHERE user_id = ? AND issue_id = issues.id)', u.id)}
+
   scope :without_due_date, -> { where(due_date: nil) }
   scope :due_before, ->(date) { where('issues.due_date < ?', date) }
   scope :due_between, ->(from_date, to_date) { where('issues.due_date >= ?', from_date).where('issues.due_date <= ?', to_date) }
@@ -47,13 +54,15 @@ class Issue < ActiveRecord::Base
 
   scope :created_after, -> (datetime) { where("created_at >= ?", datetime) }
 
-  scope :include_associations, -> { includes(:assignee, :labels, project: :namespace) }
+  scope :include_associations, -> { includes(:labels, project: :namespace) }
 
   after_save :expire_etag_cache
 
   attr_spammable :title, spam_title: true
   attr_spammable :description, spam_description: true
 
+  participant :assignees
+
   state_machine :state, initial: :opened do
     event :close do
       transition [:reopened, :opened] => :closed
@@ -73,10 +82,14 @@ class Issue < ActiveRecord::Base
   end
 
   def hook_attrs
+    assignee_ids = self.assignee_ids
+
     attrs = {
       total_time_spent: total_time_spent,
       human_total_time_spent: human_total_time_spent,
-      human_time_estimate: human_time_estimate
+      human_time_estimate: human_time_estimate,
+      assignee_ids: assignee_ids,
+      assignee_id: assignee_ids.first # This key is deprecated
     }
 
     attributes.merge!(attrs)
@@ -126,6 +139,22 @@ def self.order_by_position_and_priority
               "id DESC")
   end
 
+  # Returns a Hash of attributes to be used for Twitter card metadata
+  def card_attributes
+    {
+      'Author'   => author.try(:name),
+      'Assignee' => assignee_list
+    }
+  end
+
+  def assignee_or_author?(user)
+    author_id == user.id || assignees.exists?(user.id)
+  end
+
+  def assignee_list
+    assignees.map(&:name).to_sentence
+  end
+
   # `from` argument can be a Namespace or Project.
   def to_reference(from = nil, full: false)
     reference = "#{self.class.reference_prefix}#{iid}"
@@ -268,7 +297,7 @@ def readable_by?(user)
       true
     elsif confidential?
       author == user ||
-        assignee == user ||
+        assignees.include?(user) ||
         project.team.member?(user, Gitlab::Access::REPORTER)
     else
       project.public? ||
diff --git a/app/models/issue_assignee.rb b/app/models/issue_assignee.rb
new file mode 100644
index 0000000000000..b94b55bb1d1ec
--- /dev/null
+++ b/app/models/issue_assignee.rb
@@ -0,0 +1,29 @@
+class IssueAssignee < ActiveRecord::Base
+  extend Gitlab::CurrentSettings
+
+  belongs_to :issue
+  belongs_to :assignee, class_name: "User", foreign_key: :user_id
+
+  after_create :update_assignee_cache_counts
+  after_destroy :update_assignee_cache_counts
+
+  # EE-specific
+  after_create :update_elasticsearch_index
+  after_destroy :update_elasticsearch_index
+  # EE-specific
+
+  def update_assignee_cache_counts
+    assignee&.update_cache_counts
+  end
+
+  def update_elasticsearch_index
+    if current_application_settings.elasticsearch_indexing?
+      ElasticIndexerWorker.perform_async(
+        :update,
+        'Issue',
+        issue.id,
+        changed_fields: ['assignee_ids']
+      )
+    end
+  end
+end
diff --git a/app/models/merge_request.rb b/app/models/merge_request.rb
index cf176b98fdf18..81c2fe2fdede9 100644
--- a/app/models/merge_request.rb
+++ b/app/models/merge_request.rb
@@ -22,6 +22,8 @@ class MergeRequest < ActiveRecord::Base
 
   has_many :merge_requests_closing_issues, class_name: 'MergeRequestsClosingIssues', dependent: :delete_all
 
+  belongs_to :assignee, class_name: "User"
+
   serialize :merge_params, Hash
 
   after_create :ensure_merge_request_diff, unless: :importing?
@@ -119,10 +121,15 @@ class MergeRequest < ActiveRecord::Base
   scope :from_source_branches, ->(branches) { where(source_branch: branches) }
   scope :join_project, -> { joins(:target_project) }
   scope :references_project, -> { references(:target_project) }
+  scope :assigned, -> { where("assignee_id IS NOT NULL") }
+  scope :unassigned, -> { where("assignee_id IS NULL") }
+  scope :assigned_to, ->(u) { where(assignee_id: u.id)}
 
   participant :approvers_left
+  participant :assignee
 
   after_save :keep_around_commit
+  after_save :update_assignee_cache_counts, if: :assignee_id_changed?
 
   def self.reference_prefix
     '!'
@@ -184,6 +191,30 @@ def self.wip_title(title)
     work_in_progress?(title) ? title : "WIP: #{title}"
   end
 
+  def update_assignee_cache_counts
+    # make sure we flush the cache for both the old *and* new assignees(if they exist)
+    previous_assignee = User.find_by_id(assignee_id_was) if assignee_id_was
+    previous_assignee&.update_cache_counts
+    assignee&.update_cache_counts
+  end
+
+  # Returns a Hash of attributes to be used for Twitter card metadata
+  def card_attributes
+    {
+      'Author'   => author.try(:name),
+      'Assignee' => assignee.try(:name)
+    }
+  end
+
+  # This method is needed for compatibility with issues to not mess view and other code
+  def assignees
+    Array(assignee)
+  end
+
+  def assignee_or_author?(user)
+    author_id == user.id || assignee_id == user.id
+  end
+
   # `from` argument can be a Namespace or Project.
   def to_reference(from = nil, full: false)
     reference = "#{self.class.reference_prefix}#{iid}"
diff --git a/app/models/milestone.rb b/app/models/milestone.rb
index 66f017d057fe0..7c5ceb28d1cd1 100644
--- a/app/models/milestone.rb
+++ b/app/models/milestone.rb
@@ -23,7 +23,6 @@ class Milestone < ActiveRecord::Base
   has_many :issues
   has_many :labels, -> { distinct.reorder('labels.title') },  through: :issues
   has_many :merge_requests
-  has_many :participants, -> { distinct.reorder('users.name') }, through: :issues, source: :assignee
   has_many :events, as: :target, dependent: :destroy
 
   scope :active, -> { with_state(:active) }
@@ -109,6 +108,10 @@ def self.upcoming_ids_by_projects(projects)
     end
   end
 
+  def participants
+    User.joins(assigned_issues: :milestone).where("milestones.id = ?", id)
+  end
+
   def self.sort(method)
     case method.to_s
     when 'due_date_asc'
diff --git a/app/models/note.rb b/app/models/note.rb
index dc8d60b7e0cff..a7d15e6a701cb 100644
--- a/app/models/note.rb
+++ b/app/models/note.rb
@@ -19,6 +19,11 @@ class Note < ActiveRecord::Base
 
   cache_markdown_field :note, pipeline: :note, issuable_state_filter_enabled: true
 
+  # Aliases to make application_helper#edited_time_ago_with_tooltip helper work properly with notes.
+  # See https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10392/diffs#note_28719102
+  alias_attribute :last_edited_at, :updated_at
+  alias_attribute :last_edited_by, :updated_by
+
   # Attribute containing rendered and redacted Markdown as generated by
   # Banzai::ObjectRenderer.
   attr_accessor :redacted_note_html
@@ -39,6 +44,7 @@ class Note < ActiveRecord::Base
   belongs_to :noteable, polymorphic: true, touch: true
   belongs_to :author, class_name: "User"
   belongs_to :updated_by, class_name: "User"
+  belongs_to :last_edited_by, class_name: 'User'
 
   has_many :todos, dependent: :destroy
   has_many :events, as: :target, dependent: :destroy
diff --git a/app/models/snippet.rb b/app/models/snippet.rb
index d6ac49c570b4c..0dfb6da484304 100644
--- a/app/models/snippet.rb
+++ b/app/models/snippet.rb
@@ -13,6 +13,11 @@ class Snippet < ActiveRecord::Base
   cache_markdown_field :title, pipeline: :single_line
   cache_markdown_field :content
 
+  # Aliases to make application_helper#edited_time_ago_with_tooltip helper work properly with snippets.
+  # See https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/10392/diffs#note_28719102
+  alias_attribute :last_edited_at, :updated_at
+  alias_attribute :last_edited_by, :updated_by
+
   # If file_name changes, it invalidates content
   alias_method :default_content_html_invalidator, :content_html_invalidated?
   def content_html_invalidated?
diff --git a/app/models/system_note_metadata.rb b/app/models/system_note_metadata.rb
index 729a2b6be91bd..789843fd34e5d 100644
--- a/app/models/system_note_metadata.rb
+++ b/app/models/system_note_metadata.rb
@@ -1,6 +1,6 @@
 class SystemNoteMetadata < ActiveRecord::Base
   ICON_TYPES = %w[
-    commit merge confidential visible label assignee cross_reference
+    commit description merge confidential visible label assignee cross_reference
     title time_tracking branch milestone discussion task moved opened closed merged
     approved unapproved
   ].freeze
diff --git a/app/models/user.rb b/app/models/user.rb
index 00fe0803d0ba2..85ca243525eff 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -25,6 +25,7 @@ class User < ActiveRecord::Base
   default_value_for :hide_no_password, false
   default_value_for :project_view, :files
   default_value_for :notified_of_own_activity, false
+  default_value_for :preferred_language, I18n.default_locale
 
   attr_encrypted :otp_secret,
     key:       Gitlab::Application.secrets.otp_key_base,
@@ -89,7 +90,8 @@ class User < ActiveRecord::Base
   has_many :merge_requests,           dependent: :destroy, foreign_key: :author_id
   has_many :events,                   dependent: :destroy, foreign_key: :author_id
   has_many :subscriptions,            dependent: :destroy
-  has_many :recent_events, -> { order "id DESC" }, foreign_key: :author_id,   class_name: "Event"
+  has_many :recent_events, -> { order "id DESC" }, foreign_key: :author_id, class_name: "Event"
+
   has_many :oauth_applications,       class_name: 'Doorkeeper::Application', as: :owner, dependent: :destroy
   has_one  :abuse_report,             dependent: :destroy, foreign_key: :user_id
   has_many :reported_abuse_reports,   dependent: :destroy, foreign_key: :reporter_id, class_name: "AbuseReport"
@@ -109,6 +111,10 @@ class User < ActiveRecord::Base
   has_many :protected_branch_push_access_levels, dependent: :destroy, class_name: ProtectedBranch::PushAccessLevel
   has_many :triggers,                 dependent: :destroy, class_name: 'Ci::Trigger', foreign_key: :owner_id
 
+  has_many :issue_assignees
+  has_many :assigned_issues, class_name: "Issue", through: :issue_assignees, source: :issue
+  has_many :assigned_merge_requests,  dependent: :nullify, foreign_key: :assignee_id, class_name: "MergeRequest"
+
   # Issues that a user owns are expected to be moved to the "ghost" user before
   # the user is destroyed. If the user owns any issues during deletion, this
   # should be treated as an exceptional condition.
diff --git a/app/presenters/projects/settings/deploy_keys_presenter.rb b/app/presenters/projects/settings/deploy_keys_presenter.rb
index 86ac513b3c07e..070b0c35e3673 100644
--- a/app/presenters/projects/settings/deploy_keys_presenter.rb
+++ b/app/presenters/projects/settings/deploy_keys_presenter.rb
@@ -48,6 +48,17 @@ def any_available_public_keys_enabled?
         available_public_keys.any?
       end
 
+      def as_json
+        serializer = DeployKeySerializer.new
+        opts = { user: current_user }
+
+        {
+          enabled_keys: serializer.represent(enabled_keys, opts),
+          available_project_keys: serializer.represent(available_project_keys, opts),
+          public_keys: serializer.represent(available_public_keys, opts)
+        }
+      end
+
       def to_partial_path
         'projects/deploy_keys/index'
       end
diff --git a/app/serializers/README.md b/app/serializers/README.md
new file mode 100644
index 0000000000000..0337f88db5f65
--- /dev/null
+++ b/app/serializers/README.md
@@ -0,0 +1,325 @@
+# Serializers
+
+This is a documentation for classes located in `app/serializers` directory.
+
+In GitLab, we use [grape-entities][grape-entity-project], accompanied by a
+serializer, to convert a Ruby object to its JSON representation.
+
+Serializers are typically used in controllers to build a JSON response
+that is usually consumed by a frontend code.
+
+## Why using a serializer is important?
+
+Using serializers, instead of `to_json` method, has several benefits:
+
+* it helps to prevent exposure of a sensitive data stored in the database
+* it makes it easier to test what should and should not be exposed
+* it makes it easier to reuse serialization entities that are building blocks
+* it makes it easier to move complexity from controllers to easily testable
+  classes
+* it encourages hiding complexity behind intentions-revealing interfaces
+* it makes it easier to take care about serialization performance concerns
+* it makes it easier to reduce merge conflicts between CE -> EE
+* it makes it easier to benefit from domain driven development techniques
+
+## What is a serializer?
+
+A serializer is a class that encapsulates all business rules for building a
+JSON response using serialization entities.
+
+It is designed to be testable and to support passing additional context from
+the controller.
+
+## What is a serialization entity?
+
+Entities are lightweight structures that allow to represent domain models
+in a consistent and abstracted way, and reuse them as building blocks to
+create a payload.
+
+Entities located in `app/serializers` are usually derived from a
+[`Grape::Entity`][grape-entity-class] class.
+
+Serialization entities that do require to have a knowledge about specific
+elements of the request, need to mix `RequestAwareEntity` in.
+
+A serialization entity usually maps a domain model class into its JSON
+representation. It rarely happens that a serialization entity exists without
+a corresponding domain model class. As an example, we have an `Issue` class and
+a corresponding `IssueSerializer`.
+
+Serialization entites are designed to reuse other serialization entities, which
+is a convenient way to create a multi-level JSON representation of a piece of
+a domain model you want to serialize.
+
+See [documentation for Grape Entites][grape-entity-readme] for more details.
+
+## How to implement a serializer?
+
+### Base implementation
+
+In order to effectively implement a serializer it is necessary to create a new
+class in `app/serializers`. See existing serializers as an example.
+
+A new serializer should inherit from a `BaseSerializer` class. It is necessary
+to specify which serialization entity will be used to serialize a resource.
+
+```ruby
+class MyResourceSerializer < BaseSerialize
+  entity MyResourceEntity
+end
+```
+
+The example above shows how a most simple serializer can look like.
+
+Given that the entity `MyResourceEntity` exists, you can now use
+`MyResourceSerializer` in the controller by creating an instance of it, and
+calling `MyResourceSerializer#represent(resource)` method.
+
+Note that a `resource` can be either a single object, an array of objects or an
+`ActiveRecord::Relation` object. A serialization entity should be smart enough
+to accurately represent each of these.
+
+It should not be necessary to use `Enumerable#map`, and it should be avoided
+from the performance reasons.
+
+### Choosing what gets serialized
+
+It often happens that you might want to use the same serializer in many places,
+but sometimes the intention is to only expose a small subset of object's
+attributes in one place, and a different subset in another.
+
+`BaseSerializer#represent(resource, opts = {})` method can take an additional
+hash argument, `opts`, that defines what is going to be serialized.
+
+`BaseSerializer` will pass these options to a serialization entity. See
+how it is [documented in the upstream project][grape-entity-only].
+
+With this approach you can extend the serializer to respond to methods that will
+create a JSON response according to your needs.
+
+```ruby
+class PipelineSerializer < BaseSerializer
+  entity PipelineEntity
+
+  def represent_details(resource)
+    represent(resource, only: [:details])
+  end
+
+  def represent_status(resource)
+    represent(resource, only: [:status])
+  end
+end
+```
+
+It is possible to use `only` and `except` keywords. Both keywords do support
+nested attributes, like `except: [:id, { user: [:id] }]`.
+
+Passing `only` and `except` to the `represent` method from a controller is
+possible, but it defies principles of encapsulation and testability, and it is
+better to avoid it, and to add a specific method to the serializer instead.
+
+### Reusing serialization entities from the API
+
+Public API in GitLab is implemented using [Grape][grape-project].
+
+Under the hood it also uses [`Grape::Entity`][grape-entity-class] classes.
+This means that it is possible to reuse these classes to implement internal
+serializers.
+
+You can either use such entity directly:
+
+```ruby
+class MyResourceSerializer < BaseSerializer
+  entity API::Entities::SomeEntity
+end
+```
+
+Or derive a new serialization entity class from it:
+
+```ruby
+class MyEntity < API::Entities::SomeEntity
+  include RequestAwareEntity
+
+  unexpose :something
+end
+```
+
+It might be a good idea to write specs for entities that do inherit from
+the API, because when API payloads are changed / extended, it is easy to forget
+about the impact on the internal API through a serializer that reuses API
+entities.
+
+It is usually safe to do that, because API entities rarely break backward
+compatibility, but additional exposure may have a performance impact when API
+gets extended significantly. Write tests that check if only necessary data is
+exposed.
+
+## How to write tests for a serializer?
+
+Like every other class in the project, creating a serializer warrants writing
+tests for it.
+
+It is usually a good idea to test each public method in the serializer against
+a valid payload. `BaseSerializer#represent` returns a hash, so it is possible
+to use usual RSpec matchers like `include`.
+
+Sometimes, when the payload is large, it makes sense to validate it entirely
+using `match_response_schema` matcher along with a new fixture that can be
+stored in `spec/fixtures/api/schemas/`. This matcher is using a `json-schema`
+gem, which is quite flexible, see a [documentation][json-schema-gem] for it.
+
+## How to use a serializer in a controller?
+
+Once a new serializer is implemented, it is possible to use it in a controller.
+
+Create an instance of the serializer and render the response.
+
+```ruby
+def index
+  format.json do
+    render json: MyResourceSerializer
+      .new(current_user: @current_user)
+      .represent_details(@project.resources)
+  nd
+end
+```
+
+If it is necessary to include additional information in the payload, it is
+possible to extend what is going to be rendered, the usual way:
+
+```ruby
+def index
+  format.json do
+    render json: {
+      resources: MyResourceSerializer
+        .new(current_user: @current_user)
+        .represent_details(@project.resources),
+      count: @project.resources.count
+    }
+  nd
+end
+```
+
+Note that in these examples an additional context is being passed to the
+serializer (`current_user: @current_user`).
+
+## How to pass an additional context from the controller?
+
+It is possible to pass an additional context from a controller to a
+serializer and each serialization entity that is used in the process.
+
+Serialization entities that do require an additional context have
+`RequestAwareEntity` concern mixed in. This piece of the code exposes a method
+called `request` in every serialization entity that is instantiated during
+serialization.
+
+An object returned by this method is an instance of `EntityRequest`, which
+behaves like an `OpenStruct` object, with the difference that it will raise
+an error if an unknown method is called.
+
+In other words, in the previous example, `request` method will return an
+instance of `EntityRequest` that responds to `current_user` method. It will be
+available in every serialization entity instantiated by `MyResourceSerializer`.
+
+`EntityRequest` is a workaround for [#20045][issue-20045] and is meant to be
+refactored soon. Please avoid passing an additional context that is not
+required by a serialization entity.
+
+At the moment, the context that is passed to entities most often is
+`current_user` and `project`.
+
+## How is this related to using presenters?
+
+Payload created by a serializer is usually a representation of the backed code,
+combined with the current request data. Therefore, technically, serializers
+are presenters that create payload consumed by a frontend code, usually Vue
+components.
+
+In GitLab, it is possible to use [presenters][presenters-readme], but
+`BaseSerializer` still needs to learn how to use it, see [#30898][issue-30898].
+
+It is possible to use presenters when serializer is used to represent only
+a single object. It is not supported when  `ActiveRecord::Relation` is being
+serialized.
+
+```ruby
+MyObjectSerializer.new.represent(object.present)
+```
+
+## Best practices
+
+1. Do not invoke a serializer from within a serialization entity.
+
+    If you need to use a serializer from within a serialization entity, it is
+    possible that you are missing a class for an important domain concept.
+
+    Consider creating a new domain class and a corresponding serialization
+    entity for it.
+
+1. Use only one approach to switch behavior of the serializer.
+
+    It is possible to use a few approaches to switch a behavior of the
+    serializer. Most common are using a [Fluent Interface][fluent-interface]
+    and creating a separate `represent_something` methods.
+
+    Whatever you choose, it might be better to use only one approach at a time.
+
+1. Do not forget about creating specs for serialization entities.
+
+    Writing tests for the serializer indeed does cover testing a behavior of
+    serialization entities that the serializer instantiates. However it might
+    be a good idea to write separate tests for entities as well, because these
+    are meant to be reused in different serializers, and a serializer can
+    change a behavior of a serialization entity.
+
+1. Use `ActiveRecord::Relation` where possible
+
+    Using an `ActiveRecord::Relation` might help from the performance perspective.
+
+1. Be diligent about passing an additional context from the controller.
+
+    Using `EntityRequest` and `RequestAwareEntity` is a workaround for the lack
+    of high-level mechanism. It is meant to be refactored, and current
+    implementation is error prone. Imagine the situation that one serialization
+    entity requires `request.user` attribute, but the second one wants
+    `request.current_user`. When it happens that these two entities are used in
+    the same serialization request, you might need to pass both parameters to
+    the serializer, which is obviously not a perfect situation.
+
+    When in doubt, pass only `current_user` and `project` if these are required.
+
+1. Keep performance concerns in mind
+
+    Using a serializer incorrectly can have significant impact on the
+    performance.
+
+    Because serializers are technically presenters, it is often necessary
+    to calculate, for example, paths to various controller-actions.
+    Since using URL helpers usually involve passing `project` and `namespace`
+    adding `includes(project: :namespace)` in the serializer, can help to avoid
+    N+1 queries.
+
+    Also, try to avoid using `Enumerable#map` or other methods that will
+    execute a database query eagerly.
+
+1. Avoid passing `only` and `except` from the controller.
+1. Write tests checking for N+1 queries.
+1. Write controller tests for actions / formats using serializers.
+1. Write tests that check if only necessary data is exposed.
+1. Write tests that check if no sensitive data is exposed.
+
+## Future
+
+* [Next iteration of serializers][issue-27569]
+
+[grape-project]: http://www.ruby-grape.org
+[grape-entity-project]: https://github.com/ruby-grape/grape-entity
+[grape-entity-readme]: https://github.com/ruby-grape/grape-entity/blob/master/README.md
+[grape-entity-class]: https://github.com/ruby-grape/grape-entity/blob/master/lib/grape_entity/entity.rb
+[grape-entity-only]: https://github.com/ruby-grape/grape-entity/blob/master/README.md#returning-only-the-fields-you-want
+[presenters-readme]: https://gitlab.com/gitlab-org/gitlab-ce/blob/master/app/presenters/README.md
+[fluent-interface]: https://en.wikipedia.org/wiki/Fluent_interface
+[json-schema-gem]: https://github.com/ruby-json-schema/json-schema
+[issue-20045]: https://gitlab.com/gitlab-org/gitlab-ce/issues/20045
+[issue-30898]: https://gitlab.com/gitlab-org/gitlab-ce/issues/30898
+[issue-27569]: https://gitlab.com/gitlab-org/gitlab-ce/issues/27569
diff --git a/app/serializers/analytics_stage_entity.rb b/app/serializers/analytics_stage_entity.rb
index 69bf693de8d8e..564612202b5ec 100644
--- a/app/serializers/analytics_stage_entity.rb
+++ b/app/serializers/analytics_stage_entity.rb
@@ -2,6 +2,7 @@ class AnalyticsStageEntity < Grape::Entity
   include EntityDateHelper
 
   expose :title
+  expose :name
   expose :legend
   expose :description
 
diff --git a/app/serializers/analytics_summary_entity.rb b/app/serializers/analytics_summary_entity.rb
index 91803ec07f5ff..9c37afd53e1ac 100644
--- a/app/serializers/analytics_summary_entity.rb
+++ b/app/serializers/analytics_summary_entity.rb
@@ -1,7 +1,4 @@
 class AnalyticsSummaryEntity < Grape::Entity
   expose :value, safe: true
-
-  expose :title do |object|
-    object.title.pluralize(object.value)
-  end
+  expose :title
 end
diff --git a/app/serializers/deploy_key_entity.rb b/app/serializers/deploy_key_entity.rb
new file mode 100644
index 0000000000000..d75a83d0fa529
--- /dev/null
+++ b/app/serializers/deploy_key_entity.rb
@@ -0,0 +1,14 @@
+class DeployKeyEntity < Grape::Entity
+  expose :id
+  expose :user_id
+  expose :title
+  expose :fingerprint
+  expose :can_push
+  expose :destroyed_when_orphaned?, as: :destroyed_when_orphaned
+  expose :almost_orphaned?, as: :almost_orphaned
+  expose :created_at
+  expose :updated_at
+  expose :projects, using: ProjectEntity do |deploy_key|
+    deploy_key.projects.select { |project| options[:user].can?(:read_project, project) }
+  end
+end
diff --git a/app/serializers/deploy_key_serializer.rb b/app/serializers/deploy_key_serializer.rb
new file mode 100644
index 0000000000000..8f849eb88b717
--- /dev/null
+++ b/app/serializers/deploy_key_serializer.rb
@@ -0,0 +1,3 @@
+class DeployKeySerializer < BaseSerializer
+  entity DeployKeyEntity
+end
diff --git a/app/serializers/issuable_entity.rb b/app/serializers/issuable_entity.rb
index 29aecb50849fc..65b204d4dd27b 100644
--- a/app/serializers/issuable_entity.rb
+++ b/app/serializers/issuable_entity.rb
@@ -1,7 +1,6 @@
 class IssuableEntity < Grape::Entity
   expose :id
   expose :iid
-  expose :assignee_id
   expose :author_id
   expose :description
   expose :lock_version
diff --git a/app/serializers/issue_entity.rb b/app/serializers/issue_entity.rb
index 46cdf2c0aa51d..dc85232499dec 100644
--- a/app/serializers/issue_entity.rb
+++ b/app/serializers/issue_entity.rb
@@ -1,6 +1,7 @@
 class IssueEntity < IssuableEntity
   expose :branch_name
   expose :confidential
+  expose :assignees, using: API::Entities::UserBasic
   expose :due_date
   expose :moved_to_id
   expose :project_id
diff --git a/app/serializers/label_entity.rb b/app/serializers/label_entity.rb
index 304fd9de08f8e..ad565654342ba 100644
--- a/app/serializers/label_entity.rb
+++ b/app/serializers/label_entity.rb
@@ -6,6 +6,7 @@ class LabelEntity < Grape::Entity
   expose :group_id
   expose :project_id
   expose :template
+  expose :text_color
   expose :created_at
   expose :updated_at
 end
diff --git a/app/serializers/label_serializer.rb b/app/serializers/label_serializer.rb
new file mode 100644
index 0000000000000..ad6ba8c46c99e
--- /dev/null
+++ b/app/serializers/label_serializer.rb
@@ -0,0 +1,7 @@
+class LabelSerializer < BaseSerializer
+  entity LabelEntity
+
+  def represent_appearance(resource)
+    represent(resource, { only: [:id, :title, :color, :text_color] })
+  end
+end
diff --git a/app/serializers/merge_request_entity.rb b/app/serializers/merge_request_entity.rb
index 59c94c613f6c0..d2a9d76108ddd 100644
--- a/app/serializers/merge_request_entity.rb
+++ b/app/serializers/merge_request_entity.rb
@@ -1,5 +1,6 @@
 class MergeRequestEntity < IssuableEntity
   expose :approvals_before_merge
+  expose :assignee_id
   expose :in_progress_merge_commit_sha
   expose :locked_at
   expose :merge_commit_sha
diff --git a/app/serializers/project_entity.rb b/app/serializers/project_entity.rb
new file mode 100644
index 0000000000000..a471a7e6a8825
--- /dev/null
+++ b/app/serializers/project_entity.rb
@@ -0,0 +1,14 @@
+class ProjectEntity < Grape::Entity
+  include RequestAwareEntity
+  
+  expose :id
+  expose :name
+
+  expose :full_path do |project|
+    namespace_project_path(project.namespace, project)
+  end
+
+  expose :full_name do |project|
+    project.full_name
+  end
+end
diff --git a/app/services/issuable/bulk_update_service.rb b/app/services/issuable/bulk_update_service.rb
index 60891cbb255fb..40ff9b8b8679a 100644
--- a/app/services/issuable/bulk_update_service.rb
+++ b/app/services/issuable/bulk_update_service.rb
@@ -7,10 +7,14 @@ def execute(type)
       ids = params.delete(:issuable_ids).split(",")
       items = model_class.where(id: ids)
 
-      %i(state_event milestone_id assignee_id add_label_ids remove_label_ids subscription_event).each do |key|
+      %i(state_event milestone_id assignee_id assignee_ids add_label_ids remove_label_ids subscription_event).each do |key|
         params.delete(key) unless params[key].present?
       end
 
+      if params[:assignee_ids] == [IssuableFinder::NONE.to_s]
+        params[:assignee_ids] = []
+      end
+
       items.each do |issuable|
         next unless can?(current_user, :"update_#{type}", issuable)
 
diff --git a/app/services/issuable_base_service.rb b/app/services/issuable_base_service.rb
index 97058db922923..2802ae36226fd 100644
--- a/app/services/issuable_base_service.rb
+++ b/app/services/issuable_base_service.rb
@@ -1,11 +1,6 @@
 class IssuableBaseService < BaseService
   private
 
-  def create_assignee_note(issuable)
-    SystemNoteService.change_assignee(
-      issuable, issuable.project, current_user, issuable.assignee)
-  end
-
   def create_milestone_note(issuable)
     SystemNoteService.change_milestone(
       issuable, issuable.project, current_user, issuable.milestone)
@@ -24,6 +19,10 @@ def create_title_change_note(issuable, old_title)
       issuable, issuable.project, current_user, old_title)
   end
 
+  def create_description_change_note(issuable)
+    SystemNoteService.change_description(issuable, issuable.project, current_user)
+  end
+
   def create_branch_change_note(issuable, branch_type, old_branch, new_branch)
     SystemNoteService.change_branch(
       issuable, issuable.project, current_user, branch_type,
@@ -53,6 +52,7 @@ def filter_params(issuable)
       params.delete(:add_label_ids)
       params.delete(:remove_label_ids)
       params.delete(:label_ids)
+      params.delete(:assignee_ids)
       params.delete(:assignee_id)
       params.delete(:due_date)
     end
@@ -77,7 +77,7 @@ def filter_assignee(issuable)
   def assignee_can_read?(issuable, assignee_id)
     new_assignee = User.find_by_id(assignee_id)
 
-    return false unless new_assignee.present?
+    return false unless new_assignee
 
     ability_name = :"read_#{issuable.to_ability_name}"
     resource     = issuable.persisted? ? issuable : project
@@ -207,6 +207,7 @@ def update(issuable)
     filter_params(issuable)
     old_labels = issuable.labels.to_a
     old_mentioned_users = issuable.mentioned_users.to_a
+    old_assignees = issuable.assignees.to_a
 
     label_ids = process_label_ids(params, existing_label_ids: issuable.label_ids)
     params[:label_ids] = label_ids if labels_changing?(issuable.label_ids, label_ids)
@@ -214,6 +215,10 @@ def update(issuable)
     if issuable.changed? || params.present?
       issuable.assign_attributes(params.merge(updated_by: current_user))
 
+      if has_title_or_description_changed?(issuable)
+        issuable.assign_attributes(last_edited_at: Time.now, last_edited_by: current_user)
+      end
+
       before_update(issuable)
 
       if issuable.with_transaction_returning_status { issuable.save }
@@ -222,7 +227,13 @@ def update(issuable)
           handle_common_system_notes(issuable, old_labels: old_labels)
         end
 
-        handle_changes(issuable, old_labels: old_labels, old_mentioned_users: old_mentioned_users)
+        handle_changes(
+          issuable,
+          old_labels: old_labels,
+          old_mentioned_users: old_mentioned_users,
+          old_assignees: old_assignees
+        )
+
         after_update(issuable)
         issuable.create_new_cross_references!(current_user)
         execute_hooks(issuable, 'update')
@@ -236,6 +247,10 @@ def labels_changing?(old_label_ids, new_label_ids)
     old_label_ids.sort != new_label_ids.sort
   end
 
+  def has_title_or_description_changed?(issuable)
+    issuable.title_changed? || issuable.description_changed?
+  end
+
   def change_state(issuable)
     case params.delete(:state_event)
     when 'reopen'
@@ -272,7 +287,7 @@ def toggle_award(issuable)
     end
   end
 
-  def has_changes?(issuable, old_labels: [])
+  def has_changes?(issuable, old_labels: [], old_assignees: [])
     valid_attrs = [:title, :description, :assignee_id, :milestone_id, :target_branch]
 
     attrs_changed = valid_attrs.any? do |attr|
@@ -281,7 +296,9 @@ def has_changes?(issuable, old_labels: [])
 
     labels_changed = issuable.labels != old_labels
 
-    attrs_changed || labels_changed
+    assignees_changed = issuable.assignees != old_assignees
+
+    attrs_changed || labels_changed || assignees_changed
   end
 
   def handle_common_system_notes(issuable, old_labels: [])
@@ -289,6 +306,10 @@ def handle_common_system_notes(issuable, old_labels: [])
       create_title_change_note(issuable, issuable.previous_changes['title'].first)
     end
 
+    if issuable.previous_changes.include?('description')
+      create_description_change_note(issuable)
+    end
+
     if issuable.previous_changes.include?('description') && issuable.tasks?
       create_task_status_note(issuable)
     end
diff --git a/app/services/issues/base_service.rb b/app/services/issues/base_service.rb
index ee1b40db71891..eedbfa724ff66 100644
--- a/app/services/issues/base_service.rb
+++ b/app/services/issues/base_service.rb
@@ -9,11 +9,30 @@ def hook_data(issue, action)
 
     private
 
+    def create_assignee_note(issue, old_assignees)
+      SystemNoteService.change_issue_assignees(
+        issue, issue.project, current_user, old_assignees)
+    end
+
     def execute_hooks(issue, action = 'open')
       issue_data  = hook_data(issue, action)
       hooks_scope = issue.confidential? ? :confidential_issue_hooks : :issue_hooks
       issue.project.execute_hooks(issue_data, hooks_scope)
       issue.project.execute_services(issue_data, hooks_scope)
     end
+
+    def filter_assignee(issuable)
+      return if params[:assignee_ids].blank?
+
+      assignee_ids = params[:assignee_ids].select { |assignee_id| assignee_can_read?(issuable, assignee_id) }
+
+      if params[:assignee_ids].map(&:to_s) == [IssuableFinder::NONE]
+        params[:assignee_ids] = []
+      elsif assignee_ids.any?
+        params[:assignee_ids] = assignee_ids
+      else
+        params.delete(:assignee_ids)
+      end
+    end
   end
 end
diff --git a/app/services/issues/export_csv_service.rb b/app/services/issues/export_csv_service.rb
index fe76c0f1f0511..333437dfc3020 100644
--- a/app/services/issues/export_csv_service.rb
+++ b/app/services/issues/export_csv_service.rb
@@ -21,7 +21,7 @@ def email(user, project)
 
     def csv_builder
       @csv_builder ||=
-        CsvBuilder.new(@issues.includes(:author, :assignee), header_to_value_hash)
+        CsvBuilder.new(@issues.includes(:author, :assignees), header_to_value_hash)
     end
 
     private
@@ -35,8 +35,8 @@ def header_to_value_hash
        'Description' => 'description',
        'Author' => 'author_name',
        'Author Username' => -> (issue) { issue.author&.username },
-       'Assignee' => 'assignee_name',
-       'Assignee Username' => -> (issue) { issue.assignee&.username },
+       'Assignee' => -> (issue) { issue.assignees.map(&:name).join(', ') },
+       'Assignee Username' => -> (issue) { issue.assignees.map(&:username).join(', ') },
        'Confidential' => -> (issue) { issue.confidential? ? 'Yes' : 'No' },
        'Due Date' => -> (issue) { issue.due_date&.to_s(:csv) },
        'Created At (UTC)' => -> (issue) { issue.created_at&.to_s(:csv) },
diff --git a/app/services/issues/update_service.rb b/app/services/issues/update_service.rb
index b7fe5cb168b97..cd9f9a4a16e02 100644
--- a/app/services/issues/update_service.rb
+++ b/app/services/issues/update_service.rb
@@ -12,8 +12,12 @@ def before_update(issue)
       spam_check(issue, current_user)
     end
 
-    def handle_changes(issue, old_labels: [], old_mentioned_users: [])
-      if has_changes?(issue, old_labels: old_labels)
+    def handle_changes(issue, options)
+      old_labels = options[:old_labels] || []
+      old_mentioned_users = options[:old_mentioned_users] || []
+      old_assignees = options[:old_assignees] || []
+
+      if has_changes?(issue, old_labels: old_labels, old_assignees: old_assignees)
         todo_service.mark_pending_todos_as_done(issue, current_user)
       end
 
@@ -26,9 +30,9 @@ def handle_changes(issue, old_labels: [], old_mentioned_users: [])
         create_milestone_note(issue)
       end
 
-      if issue.previous_changes.include?('assignee_id')
-        create_assignee_note(issue)
-        notification_service.reassigned_issue(issue, current_user)
+      if issue.assignees != old_assignees
+        create_assignee_note(issue, old_assignees)
+        notification_service.reassigned_issue(issue, current_user, old_assignees)
         todo_service.reassigned_issue(issue, current_user)
       end
 
diff --git a/app/services/members/authorized_destroy_service.rb b/app/services/members/authorized_destroy_service.rb
index 1711be7211c6f..a85b9465c8483 100644
--- a/app/services/members/authorized_destroy_service.rb
+++ b/app/services/members/authorized_destroy_service.rb
@@ -26,15 +26,22 @@ def execute
 
     def unassign_issues_and_merge_requests(member)
       if member.is_a?(GroupMember)
-        IssuesFinder.new(user, group_id: member.source_id, assignee_id: member.user_id).
-          execute.
-          update_all(assignee_id: nil)
+        issue_ids = IssuesFinder.new(user, group_id: member.source_id, assignee_id: member.user_id).
+          execute.pluck(:id)
+
+        IssueAssignee.destroy_all(issue_id: issue_ids, user_id: member.user_id)
+
         MergeRequestsFinder.new(user, group_id: member.source_id, assignee_id: member.user_id).
           execute.
           update_all(assignee_id: nil)
       else
         project = member.source
-        project.issues.opened.assigned_to(member.user).update_all(assignee_id: nil)
+
+        IssueAssignee.destroy_all(
+          user_id: member.user_id,
+          issue_id: project.issues.opened.assigned_to(member.user).select(:id)
+        )
+
         project.merge_requests.opened.assigned_to(member.user).update_all(assignee_id: nil)
         member.user.update_cache_counts
       end
diff --git a/app/services/merge_requests/assign_issues_service.rb b/app/services/merge_requests/assign_issues_service.rb
index 066efa1acc3dd..8c6c484102039 100644
--- a/app/services/merge_requests/assign_issues_service.rb
+++ b/app/services/merge_requests/assign_issues_service.rb
@@ -4,7 +4,7 @@ def assignable_issues
       @assignable_issues ||= begin
         if current_user == merge_request.author
           closes_issues.select do |issue|
-            !issue.is_a?(ExternalIssue) && !issue.assignee_id? && can?(current_user, :admin_issue, issue)
+            !issue.is_a?(ExternalIssue) && !issue.assignees.present? && can?(current_user, :admin_issue, issue)
           end
         else
           []
@@ -14,7 +14,7 @@ def assignable_issues
 
     def execute
       assignable_issues.each do |issue|
-        Issues::UpdateService.new(issue.project, current_user, assignee_id: current_user.id).execute(issue)
+        Issues::UpdateService.new(issue.project, current_user, assignee_ids: [current_user.id]).execute(issue)
       end
 
       {
diff --git a/app/services/merge_requests/base_service.rb b/app/services/merge_requests/base_service.rb
index 582d5c47b6603..3542a41ac831b 100644
--- a/app/services/merge_requests/base_service.rb
+++ b/app/services/merge_requests/base_service.rb
@@ -38,6 +38,11 @@ def execute_hooks(merge_request, action = 'open', oldrev = nil)
 
     private
 
+    def create_assignee_note(merge_request)
+      SystemNoteService.change_assignee(
+        merge_request, merge_request.project, current_user, merge_request.assignee)
+    end
+
     # Returns all origin and fork merge requests from `@project` satisfying passed arguments.
     def merge_requests_for(source_branch, mr_states: [:opened, :reopened])
       MergeRequest
diff --git a/app/services/merge_requests/update_service.rb b/app/services/merge_requests/update_service.rb
index 6a6d606a8b6c9..0fb9be6f9ae46 100644
--- a/app/services/merge_requests/update_service.rb
+++ b/app/services/merge_requests/update_service.rb
@@ -31,7 +31,10 @@ def execute(merge_request)
       merge_request
     end
 
-    def handle_changes(merge_request, old_labels: [], old_mentioned_users: [])
+    def handle_changes(merge_request, options)
+      old_labels = options[:old_labels] || []
+      old_mentioned_users = options[:old_mentioned_users] || []
+
       if has_changes?(merge_request, old_labels: old_labels)
         todo_service.mark_pending_todos_as_done(merge_request, current_user)
       end
diff --git a/app/services/notification_recipient_service.rb b/app/services/notification_recipient_service.rb
index 8bb995158de7e..988bd0a7cdbe9 100644
--- a/app/services/notification_recipient_service.rb
+++ b/app/services/notification_recipient_service.rb
@@ -19,9 +19,14 @@ def build_recipients(target, current_user, action: nil, previous_assignee: nil,
     # Re-assign is considered as a mention of the new assignee so we add the
     # new assignee to the list of recipients after we rejected users with
     # the "on mention" notification level
-    if [:reassign_merge_request, :reassign_issue].include?(custom_action)
+    case custom_action
+    when :reassign_merge_request
       recipients << previous_assignee if previous_assignee
       recipients << target.assignee
+    when :reassign_issue
+      previous_assignees = Array(previous_assignee)
+      recipients.concat(previous_assignees)
+      recipients.concat(target.assignees)
     end
 
     recipients = reject_muted_users(recipients)
diff --git a/app/services/notification_service.rb b/app/services/notification_service.rb
index b48afe5e4b2e3..c4022c7c8e1d8 100644
--- a/app/services/notification_service.rb
+++ b/app/services/notification_service.rb
@@ -68,8 +68,25 @@ def close_issue(issue, current_user)
   #  * issue new assignee if their notification level is not Disabled
   #  * users with custom level checked with "reassign issue"
   #
-  def reassigned_issue(issue, current_user)
-    reassign_resource_email(issue, issue.project, current_user, :reassigned_issue_email)
+  def reassigned_issue(issue, current_user, previous_assignees = [])
+    recipients = NotificationRecipientService.new(issue.project).build_recipients(
+      issue,
+      current_user,
+      action: "reassign",
+      previous_assignee: previous_assignees
+    )
+
+    previous_assignee_ids = previous_assignees.map(&:id)
+
+    recipients.each do |recipient|
+      mailer.send(
+        :reassigned_issue_email,
+        recipient.id,
+        issue.id,
+        previous_assignee_ids,
+        current_user.id
+      ).deliver_later
+    end
   end
 
   # When we add labels to an issue we should send an email to:
@@ -414,10 +431,10 @@ def mailer
   end
 
   def previous_record(object, attribute)
-    if object && attribute
-      if object.previous_changes.include?(attribute)
-        object.previous_changes[attribute].first
-      end
+    return unless object && attribute
+
+    if object.previous_changes.include?(attribute)
+      object.previous_changes[attribute].first
     end
   end
 end
diff --git a/app/services/projects/propagate_service_template.rb b/app/services/projects/propagate_service_template.rb
new file mode 100644
index 0000000000000..a8ef2108492e7
--- /dev/null
+++ b/app/services/projects/propagate_service_template.rb
@@ -0,0 +1,103 @@
+module Projects
+  class PropagateServiceTemplate
+    BATCH_SIZE = 100
+
+    def self.propagate(*args)
+      new(*args).propagate
+    end
+
+    def initialize(template)
+      @template = template
+    end
+
+    def propagate
+      return unless @template.active?
+
+      Rails.logger.info("Propagating services for template #{@template.id}")
+
+      propagate_projects_with_template
+    end
+
+    private
+
+    def propagate_projects_with_template
+      loop do
+        batch = project_ids_batch
+
+        bulk_create_from_template(batch) unless batch.empty?
+
+        break if batch.size < BATCH_SIZE
+      end
+    end
+
+    def bulk_create_from_template(batch)
+      service_list = batch.map do |project_id|
+        service_hash.values << project_id
+      end
+
+      Project.transaction do
+        bulk_insert_services(service_hash.keys << 'project_id', service_list)
+        run_callbacks(batch)
+      end
+    end
+
+    def project_ids_batch
+      Project.connection.select_values(
+        <<-SQL
+          SELECT id
+          FROM projects
+          WHERE NOT EXISTS (
+            SELECT true
+            FROM services
+            WHERE services.project_id = projects.id
+            AND services.type = '#{@template.type}'
+          )
+          AND projects.pending_delete = false
+          AND projects.archived = false
+          LIMIT #{BATCH_SIZE}
+      SQL
+      )
+    end
+
+    def bulk_insert_services(columns, values_array)
+      ActiveRecord::Base.connection.execute(
+        <<-SQL.strip_heredoc
+          INSERT INTO services (#{columns.join(', ')})
+          VALUES #{values_array.map { |tuple| "(#{tuple.join(', ')})" }.join(', ')}
+      SQL
+      )
+    end
+
+    def service_hash
+      @service_hash ||=
+        begin
+          template_hash = @template.as_json(methods: :type).except('id', 'template', 'project_id')
+
+          template_hash.each_with_object({}) do |(key, value), service_hash|
+            value = value.is_a?(Hash) ? value.to_json : value
+
+            service_hash[ActiveRecord::Base.connection.quote_column_name(key)] =
+              ActiveRecord::Base.sanitize(value)
+          end
+        end
+    end
+
+    def run_callbacks(batch)
+      if active_external_issue_tracker?
+        Project.where(id: batch).update_all(has_external_issue_tracker: true)
+      end
+
+      if active_external_wiki?
+        Project.where(id: batch).update_all(has_external_wiki: true)
+      end
+    end
+
+    def active_external_issue_tracker?
+      @template.issue_tracker? && !@template.default
+    end
+
+    def active_external_wiki?
+      @template.type == 'ExternalWikiService'
+    end
+  end
+end
diff --git a/app/services/slash_commands/interpret_service.rb b/app/services/slash_commands/interpret_service.rb
index e906f52ce33b4..2c239e123ab19 100644
--- a/app/services/slash_commands/interpret_service.rb
+++ b/app/services/slash_commands/interpret_service.rb
@@ -91,32 +91,47 @@ def extractor
     end
 
     desc 'Assign'
-    explanation do |user|
-      "Assigns #{user.to_reference}." if user
+    explanation do |users|
+      "Assigns #{users.map(&:to_reference).to_sentence}." if users.any?
     end
     params '@user'
     condition do
       current_user.can?(:"admin_#{issuable.to_ability_name}", project)
     end
     parse_params do |assignee_param|
-      extract_references(assignee_param, :user).first ||
-        User.find_by(username: assignee_param)
+      users = extract_references(assignee_param, :user)
+
+      if users.empty?
+        users = User.where(username: assignee_param.split(' ').map(&:strip))
+      end
+
+      users
     end
-    command :assign do |user|
-      @updates[:assignee_id] = user.id if user
+    command :assign do |users|
+      next if users.empty?
+
+      if issuable.is_a?(Issue)
+        @updates[:assignee_ids] = users.map(&:id)
+      else
+        @updates[:assignee_id] = users.last.id
+      end
     end
 
     desc 'Remove assignee'
     explanation do
-      "Removes assignee #{issuable.assignee.to_reference}."
+      "Removes assignee #{issuable.assignees.first.to_reference}."
     end
     condition do
       issuable.persisted? &&
-        issuable.assignee_id? &&
+        issuable.assignees.any? &&
         current_user.can?(:"admin_#{issuable.to_ability_name}", project)
     end
     command :unassign do
-      @updates[:assignee_id] = nil
+      if issuable.is_a?(Issue)
+        @updates[:assignee_ids] = []
+      else
+        @updates[:assignee_id] = nil
+      end
     end
 
     desc 'Set milestone'
diff --git a/app/services/system_note_service.rb b/app/services/system_note_service.rb
index 5c52870016bf8..3ee5c05094c5a 100644
--- a/app/services/system_note_service.rb
+++ b/app/services/system_note_service.rb
@@ -49,6 +49,44 @@ def change_assignee(noteable, project, author, assignee)
     create_note(NoteSummary.new(noteable, project, author, body, action: 'assignee'))
   end
 
+  # Called when the assignees of an Issue is changed or removed
+  #
+  # issue - Issue object
+  # project  - Project owning noteable
+  # author   - User performing the change
+  # assignees - Users being assigned, or nil
+  #
+  # Example Note text:
+  #
+  #   "removed all assignees"
+  #
+  #   "assigned to @user1 additionally to @user2"
+  #
+  #   "assigned to @user1, @user2 and @user3 and unassigned from @user4 and @user5"
+  #
+  #   "assigned to @user1 and @user2"
+  #
+  # Returns the created Note object
+  def change_issue_assignees(issue, project, author, old_assignees)
+    body =
+      if issue.assignees.any? && old_assignees.any?
+        unassigned_users = old_assignees - issue.assignees
+        added_users = issue.assignees.to_a - old_assignees
+
+        text_parts = []
+        text_parts << "assigned to #{added_users.map(&:to_reference).to_sentence}" if added_users.any?
+        text_parts << "unassigned #{unassigned_users.map(&:to_reference).to_sentence}" if unassigned_users.any?
+
+        text_parts.join(' and ')
+      elsif old_assignees.any?
+        "removed all assignees"
+      elsif issue.assignees.any?
+        "assigned to #{issue.assignees.map(&:to_reference).to_sentence}"
+      end
+
+    create_note(NoteSummary.new(issue, project, author, body, action: 'assignee'))
+  end
+
   # Called when one or more labels on a Noteable are added and/or removed
   #
   # noteable       - Noteable object
@@ -261,6 +299,23 @@ def change_title(noteable, project, author, old_title)
     create_note(NoteSummary.new(noteable, project, author, body, action: 'title'))
   end
 
+  # Called when the description of a Noteable is changed
+  #
+  # noteable  - Noteable object that responds to `description`
+  # project   - Project owning noteable
+  # author    - User performing the change
+  #
+  # Example Note text:
+  #
+  #   "changed the description"
+  #
+  # Returns the created Note object
+  def change_description(noteable, project, author)
+    body = 'changed the description'
+
+    create_note(NoteSummary.new(noteable, project, author, body, action: 'description'))
+  end
+
   # Called when the confidentiality changes
   #
   # issue   - Issue object
diff --git a/app/services/todo_service.rb b/app/services/todo_service.rb
index 3ddaf7bed5939..cfa34faab8684 100644
--- a/app/services/todo_service.rb
+++ b/app/services/todo_service.rb
@@ -264,9 +264,9 @@ def handle_note(note, author, skip_users = [])
   end
 
   def create_assignment_todo(issuable, author)
-    if issuable.assignee
+    if issuable.assignees.any?
       attributes = attributes_for_todo(issuable.project, issuable, author, Todo::ASSIGNED)
-      create_todos(issuable.assignee, attributes)
+      create_todos(issuable.assignees, attributes)
     end
   end
 
diff --git a/app/views/admin/application_settings/_form.html.haml b/app/views/admin/application_settings/_form.html.haml
index 0afe327a539d8..8116569444946 100644
--- a/app/views/admin/application_settings/_form.html.haml
+++ b/app/views/admin/application_settings/_form.html.haml
@@ -414,8 +414,6 @@
 
   %fieldset
     %legend Error Reporting and Logging
-    %p
-      These settings require a restart to take effect.
     .form-group
       .col-sm-offset-2.col-sm-10
         .checkbox
@@ -423,6 +421,7 @@
             = f.check_box :sentry_enabled
             Enable Sentry
           .help-block
+            %p This setting requires a restart to take effect.
             Sentry is an error reporting and logging tool which is currently not shipped with GitLab, get it here:
             %a{ href: 'https://getsentry.com', target: '_blank', rel: 'noopener noreferrer' } https://getsentry.com
 
@@ -431,6 +430,21 @@
       .col-sm-10
         = f.text_field :sentry_dsn, class: 'form-control'
 
+    .form-group
+      .col-sm-offset-2.col-sm-10
+        .checkbox
+          = f.label :clientside_sentry_enabled do
+            = f.check_box :clientside_sentry_enabled
+            Enable Clientside Sentry
+          .help-block
+            Sentry can also be used for reporting and logging clientside exceptions.
+            %a{ href: 'https://sentry.io/for/javascript/', target: '_blank', rel: 'noopener noreferrer' } https://sentry.io/for/javascript/
+
+    .form-group
+      = f.label :clientside_sentry_dsn, 'Clientside Sentry DSN', class: 'control-label col-sm-2'
+      .col-sm-10
+        = f.text_field :clientside_sentry_dsn, class: 'form-control'
+
   %fieldset
     %legend Repository Storage
     .form-group
diff --git a/app/views/discussions/_notes.html.haml b/app/views/discussions/_notes.html.haml
index 964473ee3e08c..7ba3f3f6c42a5 100644
--- a/app/views/discussions/_notes.html.haml
+++ b/app/views/discussions/_notes.html.haml
@@ -1,6 +1,7 @@
 .discussion-notes
   %ul.notes{ data: { discussion_id: discussion.id } }
     = render partial: "shared/notes/note", collection: discussion.notes, as: :note
+    .flash-container
 
   - if current_user
     .discussion-reply-holder
diff --git a/app/views/errors/omniauth_error.html.haml b/app/views/errors/omniauth_error.html.haml
index 72508b9113411..20b7fa471a0a3 100644
--- a/app/views/errors/omniauth_error.html.haml
+++ b/app/views/errors/omniauth_error.html.haml
@@ -1,16 +1,15 @@
 - content_for(:title, 'Auth Error')
-%img{ :alt => "GitLab Logo", :src => image_path('logo.svg') }
-  %h1
-    422
+
 .container
+  = render "shared/errors/graphic_422.svg"
   %h3 Sign-in using #{@provider} auth failed
-  %hr
-  %p Sign-in failed because #{@error}.
-  %p There are couple of steps you can take:
 
-%ul
-  %li Try logging in using your email
-  %li Try logging in using your username
-  %li If you have forgotten your password, try recovering it using #{ link_to "Password recovery", new_password_path(resource_name) }
+  %p.light.subtitle Sign-in failed because #{@error}.
+
+  %p Try logging in using your username or email. If you have forgotten your password, try recovering it
 
-%p If none of the options work, try contacting the GitLab administrator.
+  = link_to "Sign in", new_session_path(:user), class: 'btn primary'
+  = link_to "Recover password", new_password_path(resource_name), class: 'btn secondary'
+
+  %hr
+  %p.light If none of the options work, try contacting a GitLab administrator.
diff --git a/app/views/issues/_issue.atom.builder b/app/views/issues/_issue.atom.builder
index 23a884480552a..2ed78bb3b6586 100644
--- a/app/views/issues/_issue.atom.builder
+++ b/app/views/issues/_issue.atom.builder
@@ -23,10 +23,19 @@ xml.entry do
     end
   end
 
-  if issue.assignee
+  if issue.assignees.any?
+    xml.assignees do
+      issue.assignees.each do |assignee|
+        xml.assignee do
+          xml.name assignee.name
+          xml.email assignee.public_email
+        end
+      end
+    end
+
     xml.assignee do
-      xml.name issue.assignee.name
-      xml.email issue.assignee_public_email
+      xml.name issue.assignees.first.name
+      xml.email issue.assignees.first.public_email
     end
   end
 end
diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml
index 19473b6ab276d..afcc2b6e4f31f 100644
--- a/app/views/layouts/_head.html.haml
+++ b/app/views/layouts/_head.html.haml
@@ -28,9 +28,12 @@
   = stylesheet_link_tag "application", media: "all"
   = stylesheet_link_tag "print",       media: "print"
 
+  = Gon::Base.render_data
+
   = webpack_bundle_tag "runtime"
   = webpack_bundle_tag "common"
   = webpack_bundle_tag "main"
+  = webpack_bundle_tag "raven" if current_application_settings.clientside_sentry_enabled
 
   - if content_for?(:page_specific_javascripts)
     = yield :page_specific_javascripts
diff --git a/app/views/layouts/application.html.haml b/app/views/layouts/application.html.haml
index 36543edc040f9..7e011ac3e75b7 100644
--- a/app/views/layouts/application.html.haml
+++ b/app/views/layouts/application.html.haml
@@ -1,9 +1,7 @@
 !!! 5
-%html{ lang: "en", class: "#{page_class}" }
+%html{ lang: I18n.locale, class: "#{page_class}" }
   = render "layouts/head"
   %body{ class: @body_class, data: { page: body_data_page, project: "#{@project.path if @project}", group: "#{@group.path if @group}" } }
-    = Gon::Base.render_data
-
     = render "layouts/header/default", title: header_title
     = render 'layouts/page', sidebar: sidebar, nav: nav
 
diff --git a/app/views/layouts/devise.html.haml b/app/views/layouts/devise.html.haml
index 07982e6dd920a..2018007b887e1 100644
--- a/app/views/layouts/devise.html.haml
+++ b/app/views/layouts/devise.html.haml
@@ -3,7 +3,6 @@
   = render "layouts/head"
   %body.ui_charcoal.login-page.application.navless{ data: { page: body_data_page } }
     .page-wrap
-      = Gon::Base.render_data
       = render "layouts/header/empty"
       = render "layouts/broadcast"
       .container.navless-container
diff --git a/app/views/layouts/devise_empty.html.haml b/app/views/layouts/devise_empty.html.haml
index 7466423a934ec..ed6731bde9540 100644
--- a/app/views/layouts/devise_empty.html.haml
+++ b/app/views/layouts/devise_empty.html.haml
@@ -2,7 +2,6 @@
 %html{ lang: "en" }
   = render "layouts/head"
   %body.ui_charcoal.login-page.application.navless
-    = Gon::Base.render_data
     = render "layouts/header/empty"
     = render "layouts/broadcast"
     .container.navless-container
diff --git a/app/views/layouts/oauth_error.html.haml b/app/views/layouts/oauth_error.html.haml
new file mode 100644
index 0000000000000..34bcd2a8b3a60
--- /dev/null
+++ b/app/views/layouts/oauth_error.html.haml
@@ -0,0 +1,127 @@
+!!! 5
+%html{ lang: "en" }
+  %head
+    %meta{ :content => "width=device-width, initial-scale=1, maximum-scale=1", :name => "viewport" }
+    %title= yield(:title)
+    :css
+      body {
+        color: #666;
+        text-align: center;
+        font-family: "Helvetica Neue", Helvetica, Arial, sans-serif;
+        margin: auto;
+        font-size: 16px;
+      }
+
+      .container {
+        margin: auto 20px;
+      }
+
+      h3 {
+        color: #456;
+        font-size: 22px;
+        font-weight: bold;
+        margin-bottom: 6px;
+      }
+
+      p {
+        max-width: 470px;
+        margin: 16px auto;
+      }
+
+      .subtitle {
+        margin: 0 auto 20px;
+      }
+
+      svg {
+        width: 280px;
+        height: 280px;
+        display: block;
+        margin: 40px auto;
+      }
+
+      .tv-screen path {
+        animation: move-lines 1s linear infinite;
+      }
+
+
+      @keyframes move-lines {
+        0% {transform: translateY(0)}
+        50% {transform: translateY(-10px)}
+        100% {transform: translateY(-20px)}
+      }
+
+      .tv-screen path:nth-child(1) {
+        animation-delay: .2s
+      }
+
+      .tv-screen path:nth-child(2) {
+        animation-delay: .4s
+      }
+
+      .tv-screen path:nth-child(3) {
+        animation-delay: .6s
+      }
+
+      .tv-screen path:nth-child(4) {
+        animation-delay: .8s
+      }
+
+      .tv-screen path:nth-child(5) {
+        animation-delay: 2s
+      }
+
+      .text-422 {
+        animation: flicker 1s infinite;
+      }
+
+      @keyframes flicker {
+        0% {opacity: 0.3;}
+        10% {opacity: 1;}
+        15% {opacity: .3;}
+        20% {opacity: .5;}
+        25% {opacity: 1;}
+      }
+
+      .light {
+        color: #8D8D8D;
+      }
+
+      hr {
+        max-width: 600px;
+        margin: 18px auto;
+        border: 0;
+        border-top: 1px solid #EEE;
+      }
+
+      .btn {
+        padding: 8px 14px;
+        border-radius: 3px;
+        border: 1px solid;
+        display: inline-block;
+        text-decoration: none;
+        margin: 4px 8px;
+        font-size: 14px;
+      }
+
+      .primary {
+        color: #fff;
+        background-color: #1aaa55;
+        border-color: #168f48;
+      }
+
+      .primary:hover {
+        background-color: #168f48;
+      }
+
+      .secondary {
+        color: #1aaa55;
+        background-color: #fff;
+        border-color: #1aaa55;
+      }
+
+      .secondary:hover {
+        background-color: #f3fff8;
+      }
+
+%body
+  = yield
diff --git a/app/views/notify/_reassigned_issuable_email.text.erb b/app/views/notify/_reassigned_issuable_email.text.erb
deleted file mode 100644
index daf20a226dd7b..0000000000000
--- a/app/views/notify/_reassigned_issuable_email.text.erb
+++ /dev/null
@@ -1,6 +0,0 @@
-Reassigned <%= issuable.class.model_name.human.titleize %> <%= issuable.iid %>
-
-<%= url_for([issuable.project.namespace.becomes(Namespace), issuable.project, issuable, { only_path: false }]) %>
-
-Assignee changed <%= "from #{@previous_assignee.name}" if @previous_assignee -%>
- to <%= "#{issuable.assignee_id ? issuable.assignee_name : 'Unassigned'}" %>
diff --git a/app/views/notify/new_issue_email.html.haml b/app/views/notify/new_issue_email.html.haml
index c762578971a5f..eb5157ccac920 100644
--- a/app/views/notify/new_issue_email.html.haml
+++ b/app/views/notify/new_issue_email.html.haml
@@ -2,9 +2,9 @@
   %p.details
     #{link_to @issue.author_name, user_url(@issue.author)} created an issue:
 
-- if @issue.assignee_id.present?
+- if @issue.assignees.any?
   %p
-    Assignee: #{@issue.assignee_name}
+    Assignee: #{@issue.assignee_list}
 
 - if @issue.description
   %div
diff --git a/app/views/notify/new_issue_email.text.erb b/app/views/notify/new_issue_email.text.erb
index ca5c2f2688c0c..13f1ac08e9454 100644
--- a/app/views/notify/new_issue_email.text.erb
+++ b/app/views/notify/new_issue_email.text.erb
@@ -2,6 +2,6 @@ New Issue was created.
 
 Issue <%= @issue.iid %>: <%= url_for(namespace_project_issue_url(@issue.project.namespace, @issue.project, @issue)) %>
 Author:    <%= @issue.author_name %>
-Assignee:  <%= @issue.assignee_name %>
+Assignee:  <%= @issue.assignee_list %>
 
 <%= @issue.description %>
diff --git a/app/views/notify/new_mention_in_issue_email.text.erb b/app/views/notify/new_mention_in_issue_email.text.erb
index 457e94b48001b..f19ac3adfc7aa 100644
--- a/app/views/notify/new_mention_in_issue_email.text.erb
+++ b/app/views/notify/new_mention_in_issue_email.text.erb
@@ -2,6 +2,6 @@ You have been mentioned in an issue.
 
 Issue <%= @issue.iid %>: <%= url_for(namespace_project_issue_url(@issue.project.namespace, @issue.project, @issue)) %>
 Author:    <%= @issue.author_name %>
-Assignee:  <%= @issue.assignee_name %>
+Assignee:  <%= @issue.assignee_list %>
 
 <%= @issue.description %>
diff --git a/app/views/notify/reassigned_issue_email.html.haml b/app/views/notify/reassigned_issue_email.html.haml
index 498ba8b83651b..ee2f40e1683a4 100644
--- a/app/views/notify/reassigned_issue_email.html.haml
+++ b/app/views/notify/reassigned_issue_email.html.haml
@@ -1 +1,10 @@
-= render 'reassigned_issuable_email', issuable: @issue
+%p
+  Assignee changed
+  - if @previous_assignees.any?
+    from
+    %strong= @previous_assignees.map(&:name).to_sentence
+  to
+  - if @issue.assignees.any?
+    %strong= @issue.assignee_list
+  - else
+    %strong Unassigned
diff --git a/app/views/notify/reassigned_issue_email.text.erb b/app/views/notify/reassigned_issue_email.text.erb
index 710253be98429..6c357f1074a48 100644
--- a/app/views/notify/reassigned_issue_email.text.erb
+++ b/app/views/notify/reassigned_issue_email.text.erb
@@ -1 +1,6 @@
-<%= render 'reassigned_issuable_email', issuable: @issue %>
+Reassigned Issue <%= @issue.iid %>
+
+<%= url_for([@issue.project.namespace.becomes(Namespace), @issue.project, @issue, { only_path: false }]) %>
+
+Assignee changed <%= "from #{@previous_assignees.map(&:name).to_sentence}" if @previous_assignees.any? -%>
+ to <%= "#{@issue.assignees.any? ? @issue.assignee_list : 'Unassigned'}" %>
diff --git a/app/views/notify/reassigned_merge_request_email.html.haml b/app/views/notify/reassigned_merge_request_email.html.haml
index 2a650130f59ce..841df872857fc 100644
--- a/app/views/notify/reassigned_merge_request_email.html.haml
+++ b/app/views/notify/reassigned_merge_request_email.html.haml
@@ -1 +1,9 @@
-= render 'reassigned_issuable_email', issuable: @merge_request
+Reassigned Merge Request #{ @merge_request.iid }
+
+= url_for([@merge_request.project.namespace.becomes(Namespace), @merge_request.project, @merge_request, { only_path: false }])
+
+Assignee changed
+- if @previous_assignee
+  from #{@previous_assignee.name}
+to
+= @merge_request.assignee_id ? @merge_request.assignee_name : 'Unassigned'
diff --git a/app/views/notify/reassigned_merge_request_email.text.erb b/app/views/notify/reassigned_merge_request_email.text.erb
index b5b4f1ff99a6b..998a40fefdeb3 100644
--- a/app/views/notify/reassigned_merge_request_email.text.erb
+++ b/app/views/notify/reassigned_merge_request_email.text.erb
@@ -1 +1,6 @@
-<%= render 'reassigned_issuable_email', issuable: @merge_request %>
+Reassigned Merge Request <%= @merge_request.iid %>
+
+<%= url_for([@merge_request.project.namespace.becomes(Namespace), @merge_request.project, @merge_request, { only_path: false }]) %>
+
+Assignee changed <%= "from #{@previous_assignee.name}" if @previous_assignee -%>
+ to <%= "#{@merge_request.assignee_id ? @merge_request.assignee_name : 'Unassigned'}" %>
diff --git a/app/views/profiles/show.html.haml b/app/views/profiles/show.html.haml
index c74b3249a1305..4a1438aa68e92 100644
--- a/app/views/profiles/show.html.haml
+++ b/app/views/profiles/show.html.haml
@@ -72,6 +72,11 @@
         = f.label :public_email, class: "label-light"
         = f.select :public_email, options_for_select(@user.all_emails, selected: @user.public_email), { include_blank: 'Do not show on profile' }, class: "select2"
         %span.help-block This email will be displayed on your public profile.
+      .form-group
+        = f.label :preferred_language, class: "label-light"
+        = f.select :preferred_language, Gitlab::I18n::AVAILABLE_LANGUAGES.map { |value, label| [label, value] },
+          {}, class: "select2"
+        %span.help-block This feature is experimental and translations are not complete yet.
       .form-group
         = f.label :skype, class: "label-light"
         = f.text_field :skype, class: "form-control"
diff --git a/app/views/projects/artifacts/_tree_file.html.haml b/app/views/projects/artifacts/_tree_file.html.haml
index 36fb4c998c96b..ce7e25d774b62 100644
--- a/app/views/projects/artifacts/_tree_file.html.haml
+++ b/app/views/projects/artifacts/_tree_file.html.haml
@@ -1,9 +1,10 @@
 - path_to_file = file_namespace_project_build_artifacts_path(@project.namespace, @project, @build, path: file.path)
 
 %tr.tree-item{ 'data-link' => path_to_file }
+  - blob = file.blob
   %td.tree-item-file-name
-    = tree_icon('file', '664', file.name)
-    %span.str-truncated
-      = link_to file.name, path_to_file
+    = tree_icon('file', blob.mode, blob.name)
+    = link_to path_to_file do
+      %span.str-truncated= blob.name
   %td
-    = number_to_human_size(file.metadata[:size], precision: 2)
+    = number_to_human_size(blob.size, precision: 2)
diff --git a/app/views/projects/artifacts/file.html.haml b/app/views/projects/artifacts/file.html.haml
new file mode 100644
index 0000000000000..d8da83b9a80ba
--- /dev/null
+++ b/app/views/projects/artifacts/file.html.haml
@@ -0,0 +1,33 @@
+- page_title @path, 'Artifacts', "#{@build.name} (##{@build.id})", 'Jobs'
+= render "projects/pipelines/head"
+
+= render "projects/builds/header", show_controls: false
+
+#tree-holder.tree-holder
+  .nav-block
+    %ul.breadcrumb.repo-breadcrumb
+      %li
+        = link_to 'Artifacts', browse_namespace_project_build_artifacts_path(@project.namespace, @project, @build)
+      - path_breadcrumbs do |title, path|
+        - title = truncate(title, length: 40)
+        %li
+          - if path == @path
+            = link_to file_namespace_project_build_artifacts_path(@project.namespace, @project, @build, path) do
+              %strong= title
+          - else
+            = link_to title, browse_namespace_project_build_artifacts_path(@project.namespace, @project, @build, path)
+
+
+  %article.file-holder
+    - blob = @entry.blob
+    .js-file-title.file-title-flex-parent
+      = render 'projects/blob/header_content', blob: blob
+
+      .file-actions.hidden-xs
+        = render 'projects/blob/viewer_switcher', blob: blob
+
+        .btn-group{ role: "group" }<
+          = copy_blob_source_button(blob)
+          = open_raw_blob_button(blob)
+
+    = render 'projects/blob/content', blob: blob
diff --git a/app/views/projects/blob/viewers/_balsamiq.html.haml b/app/views/projects/blob/viewers/_balsamiq.html.haml
new file mode 100644
index 0000000000000..28670e7de9755
--- /dev/null
+++ b/app/views/projects/blob/viewers/_balsamiq.html.haml
@@ -0,0 +1,4 @@
+- content_for :page_specific_javascripts do
+  = page_specific_javascript_bundle_tag('balsamiq_viewer')
+
+.file-content.balsamiq-viewer#js-balsamiq-viewer{ data: { endpoint: blob_raw_url } }
diff --git a/app/views/projects/boards/components/_board.html.haml b/app/views/projects/boards/components/_board.html.haml
index 5a4eaf92b1613..bc5c727bf0df2 100644
--- a/app/views/projects/boards/components/_board.html.haml
+++ b/app/views/projects/boards/components/_board.html.haml
@@ -13,8 +13,8 @@
             %button.btn.btn-small.btn-default.pull-right.has-tooltip{ type: "button",
               "@click" => "showNewIssueForm",
               "v-if" => 'list.type !== "closed"',
-              "aria-label" => "Add an issue",
-              "title" => "Add an issue",
+              "aria-label" => "New issue",
+              "title" => "New issue",
               data: { placement: "top", container: "body" } }
               = icon("plus")
         - if can?(current_user, :admin_list, @project)
diff --git a/app/views/projects/boards/components/sidebar/_assignee.html.haml b/app/views/projects/boards/components/sidebar/_assignee.html.haml
index 0f42433452134..0c09e71feeed5 100644
--- a/app/views/projects/boards/components/sidebar/_assignee.html.haml
+++ b/app/views/projects/boards/components/sidebar/_assignee.html.haml
@@ -1,40 +1,28 @@
-.block.assignee
-  .title.hide-collapsed
-    Assignee
-    - if can?(current_user, :admin_issue, @project)
-      = icon("spinner spin", class: "block-loading")
-      = link_to "Edit", "#", class: "edit-link pull-right"
-  .value.hide-collapsed
-    %span.assign-yourself.no-value{ "v-if" => "!issue.assignee" }
-      No assignee
-      - if can?(current_user, :admin_issue, @project)
-        \-
-        %a.js-assign-yourself{ href: "#" }
-          assign yourself
-    %a.author_link.bold{ ":href" => "'#{root_url}' + issue.assignee.username",
-      "v-if" => "issue.assignee" }
-      %img.avatar.avatar-inline.s32{ ":src" => "issue.assignee.avatar",
-        width: "32", alt: "Avatar" }
-      %span.author
-        {{ issue.assignee.name }}
-      %span.username
-        = precede "@" do
-          {{ issue.assignee.username }}
+.block.assignee{ ref: "assigneeBlock" }
+  %template{ "v-if" => "issue.assignees" }
+    %assignee-title{ ":number-of-assignees" => "issue.assignees.length",
+      ":loading" => "loadingAssignees",
+      ":editable" => can?(current_user, :admin_issue, @project) }
+    %assignees.value{ "root-path" => "#{root_url}",
+      ":users" => "issue.assignees",
+      ":editable" => can?(current_user, :admin_issue, @project),
+      "@assign-self" => "assignSelf" }
+
   - if can?(current_user, :admin_issue, @project)
     .selectbox.hide-collapsed
       %input{ type: "hidden",
-        name: "issue[assignee_id]",
-        id: "issue_assignee_id",
-        ":value" => "issue.assignee.id",
-        "v-if" => "issue.assignee" }
+        name: "issue[assignee_ids][]",
+        ":value" => "assignee.id",
+        "v-if" => "issue.assignees",
+        "v-for" => "assignee in issue.assignees" }
       .dropdown
-        %button.dropdown-menu-toggle.js-user-search.js-author-search.js-issue-board-sidebar{ type: "button", data: { toggle: "dropdown", field_name: "issue[assignee_id]", first_user: (current_user.username if current_user), current_user: "true", project_id: @project.id, null_user: "true", null_user_default: "true" },
+        %button.dropdown-menu-toggle.js-user-search.js-author-search.js-multiselect.js-save-user-data.js-issue-board-sidebar{ type: "button", ref: "assigneeDropdown", data: { toggle: "dropdown", field_name: "issue[assignee_ids][]", first_user: (current_user.username if current_user), current_user: "true", project_id: @project.id, null_user: "true", multi_select: "true", dropdown: { header: 'Assignee(s)'} },
           ":data-issuable-id" => "issue.id",
           ":data-selected" => "assigneeId",
           ":data-issue-update" => "'#{namespace_project_issues_path(@project.namespace, @project)}/' + issue.id + '.json'" }
-          Select assignee
+          Select assignee(s)
           = icon("chevron-down")
-        .dropdown-menu.dropdown-menu-user.dropdown-menu-selectable.dropdown-menu-author
+        .dropdown-menu.dropdown-select.dropdown-menu-user.dropdown-menu-selectable.dropdown-menu-author
           = dropdown_title("Assign to")
           = dropdown_filter("Search users")
           = dropdown_content
diff --git a/app/views/projects/branches/new.html.haml b/app/views/projects/branches/new.html.haml
index d3c3e40d5185a..796ecdfd0145a 100644
--- a/app/views/projects/branches/new.html.haml
+++ b/app/views/projects/branches/new.html.haml
@@ -1,4 +1,5 @@
 - page_title "New Branch"
+- default_ref = params[:ref] || @project.default_branch
 
 - if @error
   .alert.alert-danger
@@ -16,12 +17,11 @@
       .help-block.text-danger.js-branch-name-error
   .form-group
     = label_tag :ref, 'Create from', class: 'control-label'
-    .col-sm-10
-      = hidden_field_tag :ref, params[:ref] || @project.default_branch
-      = dropdown_tag(params[:ref] || @project.default_branch,
-                     options: { toggle_class: 'js-branch-select wide',
-                                filter: true, dropdown_class: "dropdown-menu-selectable", placeholder: "Search branches",
-                                data: { selected: params[:ref] || @project.default_branch, field_name: 'ref' } })
+    .col-sm-10.dropdown.create-from
+      = hidden_field_tag :ref, default_ref
+      = button_tag type: 'button', title: default_ref, class: 'dropdown-toggle form-control js-branch-select', required: true, data: { toggle: 'dropdown', selected: default_ref, field_name: 'ref' } do
+        .text-left.dropdown-toggle-text= default_ref
+      = render 'shared/ref_dropdown', dropdown_class: 'wide'
       .help-block Existing branch name, tag, or commit SHA
   .form-actions
     = button_tag 'Create branch', class: 'btn btn-create', tabindex: 3
diff --git a/app/views/projects/compare/_form.html.haml b/app/views/projects/compare/_form.html.haml
index 0f080b6aceefb..1f4c9fac54c73 100644
--- a/app/views/projects/compare/_form.html.haml
+++ b/app/views/projects/compare/_form.html.haml
@@ -9,7 +9,7 @@
         = hidden_field_tag :from, params[:from]
         = button_tag type: 'button', title: params[:from], class: "form-control compare-dropdown-toggle js-compare-dropdown has-tooltip", required: true, data: { refs_url: refs_namespace_project_path(@project.namespace, @project), toggle: "dropdown", target: ".js-compare-from-dropdown", selected: params[:from], field_name: :from } do
           .dropdown-toggle-text.str-truncated= params[:from] || 'Select branch/tag'
-      = render "ref_dropdown"
+      = render 'shared/ref_dropdown'
     .compare-ellipsis.inline ...
     .form-group.dropdown.compare-form-group.to.js-compare-to-dropdown
       .input-group.inline-input-group
@@ -17,7 +17,7 @@
         = hidden_field_tag :to, params[:to]
         = button_tag type: 'button', title: params[:to], class: "form-control compare-dropdown-toggle js-compare-dropdown has-tooltip", required: true, data: { refs_url: refs_namespace_project_path(@project.namespace, @project), toggle: "dropdown", target: ".js-compare-to-dropdown", selected: params[:to], field_name: :to } do
           .dropdown-toggle-text.str-truncated= params[:to] || 'Select branch/tag'
-      = render "ref_dropdown"
+      = render 'shared/ref_dropdown'
     &nbsp;
     = button_tag "Compare", class: "btn btn-create commits-compare-btn"
     - if @merge_request.present?
diff --git a/app/views/projects/cycle_analytics/_empty_stage.html.haml b/app/views/projects/cycle_analytics/_empty_stage.html.haml
index c3f95860e9237..cdad0bc723188 100644
--- a/app/views/projects/cycle_analytics/_empty_stage.html.haml
+++ b/app/views/projects/cycle_analytics/_empty_stage.html.haml
@@ -2,6 +2,6 @@
   .empty-stage
     .icon-no-data
       = custom_icon ('icon_no_data')
-    %h4 We don't have enough data to show this stage.
+    %h4 {{ __('We don\'t have enough data to show this stage.') }}
     %p
       {{currentStage.emptyStageText}}
diff --git a/app/views/projects/cycle_analytics/_no_access.html.haml b/app/views/projects/cycle_analytics/_no_access.html.haml
index 0ffc79b318122..c3eda39823422 100644
--- a/app/views/projects/cycle_analytics/_no_access.html.haml
+++ b/app/views/projects/cycle_analytics/_no_access.html.haml
@@ -2,6 +2,6 @@
   .no-access-stage
     .icon-lock
       = custom_icon ('icon_lock')
-    %h4 You need permission.
+    %h4 {{ __('You need permission.') }}
     %p
-      Want to see the data? Please ask administrator for access.
+      {{ __('Want to see the data? Please ask an administrator for access.') }}
diff --git a/app/views/projects/cycle_analytics/show.html.haml b/app/views/projects/cycle_analytics/show.html.haml
index dd3fa814716ea..b158a81471cc4 100644
--- a/app/views/projects/cycle_analytics/show.html.haml
+++ b/app/views/projects/cycle_analytics/show.html.haml
@@ -2,29 +2,30 @@
 - page_title "Cycle Analytics"
 - content_for :page_specific_javascripts do
   = page_specific_javascript_bundle_tag('common_vue')
+  = page_specific_javascript_bundle_tag('locale')
   = page_specific_javascript_bundle_tag('cycle_analytics')
 
 = render "projects/head"
 
 #cycle-analytics{ class: container_class, "v-cloak" => "true", data: { request_path: project_cycle_analytics_path(@project) } }
   - if @cycle_analytics_no_data
-    .bordered-box.landing.content-block{ "v-if" => "!isOverviewDialogDismissed" }
-      = icon("times", class: "dismiss-icon", "@click" => "dismissOverviewDialog()")
-      .row
-        .col-sm-3.col-xs-12.svg-container
-          = custom_icon('icon_cycle_analytics_splash')
-        .col-sm-8.col-xs-12.inner-content
-          %h4
-            Introducing Cycle Analytics
-          %p
-            Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.
-
-          = link_to "Read more",  help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn'
+    .landing.content-block{ "v-if" => "!isOverviewDialogDismissed" }
+      %button.dismiss-button{ type: 'button', 'aria-label': 'Dismiss Cycle Analytics introduction box' }
+        = icon("times", "@click" => "dismissOverviewDialog()")
+      .svg-container
+        = custom_icon('icon_cycle_analytics_splash')
+      .inner-content
+        %h4
+          {{ __('Introducing Cycle Analytics') }}
+        %p
+          {{ __('Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project.') }}
+        %p
+          = link_to _('Read more'), help_page_path('user/project/cycle_analytics'), target: '_blank', class: 'btn'
   = icon("spinner spin", "v-show" => "isLoading")
   .wrapper{ "v-show" => "!isLoading && !hasError" }
     .panel.panel-default
       .panel-heading
-        Pipeline Health
+        {{ __('Pipeline Health') }}
       .content-block
         .container-fluid
           .row
@@ -34,15 +35,15 @@
             .col-sm-3.col-xs-12.column
               .dropdown.inline.js-ca-dropdown
                 %button.dropdown-menu-toggle{ "data-toggle" => "dropdown", :type => "button" }
-                  %span.dropdown-label Last 30 days
+                  %span.dropdown-label {{ n__('Last %d day', 'Last %d days', 30) }}
                   %i.fa.fa-chevron-down
                 %ul.dropdown-menu.dropdown-menu-align-right
                   %li
                     %a{ "href" => "#", "data-value" => "30" }
-                      Last 30 days
+                      {{ n__('Last %d day', 'Last %d days', 30) }}
                   %li
                     %a{ "href" => "#", "data-value" => "90" }
-                      Last 90 days
+                      {{ n__('Last %d day', 'Last %d days', 90) }}
     .stage-panel-container
       .panel.panel-default.stage-panel
         .panel-heading
@@ -50,20 +51,20 @@
             %ul
               %li.stage-header
                 %span.stage-name
-                  Stage
-                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The phase of the development lifecycle.", "aria-hidden" => "true" }
+                  {{ __('ProjectLifecycle|Stage') }}
+                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The phase of the development lifecycle."), "aria-hidden" => "true" }
               %li.median-header
                 %span.stage-name
-                  Median
-                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6.", "aria-hidden" => "true" }
+                  {{ __('Median') }}
+                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."), "aria-hidden" => "true" }
               %li.event-header
                 %span.stage-name
-                  {{ currentStage ? currentStage.legend : 'Related Issues' }}
-                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The collection of events added to the data gathered for that stage.", "aria-hidden" => "true" }
+                  {{ currentStage ? __(currentStage.legend) : __('Related Issues') }}
+                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The collection of events added to the data gathered for that stage."), "aria-hidden" => "true" }
               %li.total-time-header
                 %span.stage-name
-                  Total Time
-                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: "The time taken by each data entry gathered by that stage.", "aria-hidden" => "true" }
+                  {{ __('Total Time') }}
+                %i.has-tooltip.fa.fa-question-circle{ "data-placement" => "top", title: _("The time taken by each data entry gathered by that stage."), "aria-hidden" => "true" }
         .stage-panel-body
           %nav.stage-nav
             %ul
@@ -75,10 +76,10 @@
                     %span{ "v-if" => "stage.value" }
                       {{ stage.value }}
                     %span.stage-empty{ "v-else" => true }
-                      Not enough data
+                      {{ __('Not enough data') }}
                   %template{ "v-else" => true }
                     %span.not-available
-                      Not available
+                      {{ __('Not available') }}
           .section.stage-events
             %template{ "v-if" => "isLoadingStage" }
               = icon("spinner spin")
diff --git a/app/views/projects/deploy_keys/_index.html.haml b/app/views/projects/deploy_keys/_index.html.haml
index 4cfbd9add009d..74756b58439a0 100644
--- a/app/views/projects/deploy_keys/_index.html.haml
+++ b/app/views/projects/deploy_keys/_index.html.haml
@@ -10,25 +10,4 @@
     = render @deploy_keys.form_partial_path
   .col-lg-9.col-lg-offset-3
     %hr
-  .col-lg-9.col-lg-offset-3.append-bottom-default.deploy-keys
-    %h5.prepend-top-0
-      Enabled deploy keys for this project (#{@deploy_keys.enabled_keys_size})
-    - if @deploy_keys.any_keys_enabled?
-      %ul.well-list
-        = render partial: 'projects/deploy_keys/deploy_key', collection: @deploy_keys.enabled_keys, as: :deploy_key
-    - else
-      .settings-message.text-center
-        No deploy keys found. Create one with the form above.
-    %h5.prepend-top-default
-      Deploy keys from projects you have access to (#{@deploy_keys.available_project_keys_size})
-    - if @deploy_keys.any_available_project_keys_enabled?
-      %ul.well-list
-        = render partial: 'projects/deploy_keys/deploy_key', collection: @deploy_keys.available_project_keys, as: :deploy_key
-    - else
-      .settings-message.text-center
-        No deploy keys from your projects could be found. Create one with the form above or add existing one below.
-    - if @deploy_keys.any_available_public_keys_enabled?
-      %h5.prepend-top-default
-        Public deploy keys available to any project (#{@deploy_keys.available_public_keys_size})
-      %ul.well-list
-        = render partial: 'projects/deploy_keys/deploy_key', collection: @deploy_keys.available_public_keys, as: :deploy_key
+  #js-deploy-keys{ data: { endpoint: namespace_project_deploy_keys_path } }
diff --git a/app/views/projects/group_links/_index.html.haml b/app/views/projects/group_links/_index.html.haml
index b6116dbec414a..debb0214d068c 100644
--- a/app/views/projects/group_links/_index.html.haml
+++ b/app/views/projects/group_links/_index.html.haml
@@ -6,11 +6,9 @@
     %p
       Projects can be stored in only one group at once. However you can share a project with other groups here.
   .col-lg-9
-    %h5.prepend-top-0
-      Set a group to share
     = form_tag namespace_project_group_links_path(@project.namespace, @project), class: 'js-requires-input', method: :post do
       .form-group
-        = label_tag :link_group_id, "Group", class: "label-light"
+        = label_tag :link_group_id, "Select a group to share with", class: "label-light"
         = groups_select_tag(:link_group_id, data: { skip_groups: @skip_groups }, required: true)
       .form-group
         = label_tag :link_group_access, "Max access level", class: "label-light"
diff --git a/app/views/projects/issues/_issue.html.haml b/app/views/projects/issues/_issue.html.haml
index 74befc5f2e154..57e380b4879d6 100644
--- a/app/views/projects/issues/_issue.html.haml
+++ b/app/views/projects/issues/_issue.html.haml
@@ -13,9 +13,9 @@
             %li
               CLOSED
 
-          - if issue.assignee
+          - if issue.assignees.any?
             %li
-              = link_to_member(@project, issue.assignee, name: false, title: "Assigned to :name")
+              = render 'shared/issuable/assignees', project: @project, issue: issue
 
           = render 'shared/issuable_meta_data', issuable: issue
 
diff --git a/app/views/projects/merge_requests/_show.html.haml b/app/views/projects/merge_requests/_show.html.haml
index 881ee9fd59656..9e306d4543ce9 100644
--- a/app/views/projects/merge_requests/_show.html.haml
+++ b/app/views/projects/merge_requests/_show.html.haml
@@ -6,7 +6,7 @@
   = page_specific_javascript_bundle_tag('common_vue')
   = page_specific_javascript_bundle_tag('diff_notes')
 
-.merge-request{ 'data-url' => merge_request_path(@merge_request), 'data-project-path' => project_path(@merge_request.project) }
+.merge-request{ 'data-url' => merge_request_path(@merge_request, format: :json), 'data-project-path' => project_path(@merge_request.project) }
   = render "projects/merge_requests/show/mr_title"
 
   .merge-request-details.issuable-details{ data: { id: @merge_request.project.id } }
diff --git a/app/views/projects/notes/_actions.html.haml b/app/views/projects/notes/_actions.html.haml
index 718b52dd82e81..d70ec8a6062a5 100644
--- a/app/views/projects/notes/_actions.html.haml
+++ b/app/views/projects/notes/_actions.html.haml
@@ -31,14 +31,14 @@
 - if current_user
   - if note.emoji_awardable?
     - user_authored = note.user_authored?(current_user)
-    = link_to '#', title: 'Award Emoji', class: "note-action-button note-emoji-button js-add-award js-note-emoji #{'js-user-authored' if user_authored}", data: { position: 'right' } do
+    = link_to '#', title: 'Award Emoji', class: "note-action-button note-emoji-button js-add-award js-note-emoji #{'js-user-authored' if user_authored} has-tooltip", data: { position: 'right' } do
       = icon('spinner spin')
       %span{ class: 'link-highlight award-control-icon-neutral' }= custom_icon('emoji_slightly_smiling_face')
       %span{ class: 'link-highlight award-control-icon-positive' }= custom_icon('emoji_smiley')
       %span{ class: 'link-highlight award-control-icon-super-positive' }= custom_icon('emoji_smile')
 
   - if note_editable
-    = link_to '#', title: 'Edit comment', class: 'note-action-button js-note-edit' do
+    = link_to '#', title: 'Edit comment', class: 'note-action-button js-note-edit has-tooltip' do
       = icon('pencil', class: 'link-highlight')
-    = link_to namespace_project_note_path(note.project.namespace, note.project, note), title: 'Remove comment', method: :delete, data: { confirm: 'Are you sure you want to remove this comment?' }, remote: true, class: 'note-action-button js-note-delete danger' do
+    = link_to namespace_project_note_path(note.project.namespace, note.project, note), title: 'Remove comment', method: :delete, data: { confirm: 'Are you sure you want to remove this comment?' }, remote: true, class: 'note-action-button js-note-delete danger has-tooltip' do
       = icon('trash-o', class: 'danger-highlight')
diff --git a/app/views/projects/pipelines/_with_tabs.html.haml b/app/views/projects/pipelines/_with_tabs.html.haml
index d7cefb8613e4a..1aa48bf98131b 100644
--- a/app/views/projects/pipelines/_with_tabs.html.haml
+++ b/app/views/projects/pipelines/_with_tabs.html.haml
@@ -1,3 +1,5 @@
+- failed_builds = @pipeline.statuses.latest.failed
+
 .tabs-holder
   %ul.pipelines-tabs.nav-links.no-top.no-bottom
     %li.js-pipeline-tab-link
@@ -7,8 +9,11 @@
       = link_to builds_namespace_project_pipeline_path(@project.namespace, @project, @pipeline), data: {target: 'div#js-tab-builds', action: 'builds', toggle: 'tab' }, class: 'builds-tab' do
         Jobs
         %span.badge.js-builds-counter= pipeline.statuses.count
-
-
+    - if failed_builds.present?
+      %li.js-failures-tab-link
+        = link_to failures_namespace_project_pipeline_path(@project.namespace, @project, @pipeline), data: {target: 'div#js-tab-failures', action: 'failures', toggle: 'tab' }, class: 'failures-tab' do
+          Failed Jobs
+          %span.badge.js-failures-counter= failed_builds.count
 
 .tab-content
   #js-tab-pipeline.tab-pane
@@ -39,3 +44,13 @@
             %th Coverage
             %th
         = render partial: "projects/stage/stage", collection: pipeline.stages, as: :stage
+  - if failed_builds.present?
+    #js-tab-failures.build-failures.tab-pane
+      - failed_builds.each_with_index do |build, index|
+        .build-state
+          %span.ci-status-icon-failed= custom_icon('icon_status_failed')
+          %span.stage
+            = build.stage.titleize
+          %span.build-name
+            = link_to build.name, pipeline_build_url(pipeline, build)
+        %pre.build-log= build_summary(build, skip: index >= 10)
diff --git a/app/views/projects/project_members/_index.html.haml b/app/views/projects/project_members/_index.html.haml
index 543801451f9f0..ea93ebfbe285e 100644
--- a/app/views/projects/project_members/_index.html.haml
+++ b/app/views/projects/project_members/_index.html.haml
@@ -18,7 +18,7 @@
         = render "projects/project_members/new_project_member"
 
         = render 'shared/members/requests', membership_source: @project, requesters: @requesters
-        .append-bottom-default.clearfix
+        .clearfix
           %h5.member.existing-title
             Existing members and groups
         - if @group_links.any?
diff --git a/app/views/projects/protected_branches/_create_protected_branch.html.haml b/app/views/projects/protected_branches/_create_protected_branch.html.haml
index 77b05c6bf4b1c..f35718a90d46d 100644
--- a/app/views/projects/protected_branches/_create_protected_branch.html.haml
+++ b/app/views/projects/protected_branches/_create_protected_branch.html.haml
@@ -25,7 +25,7 @@
             .merge_access_levels-container
               = dropdown_tag('Select',
                              options: { toggle_class: 'js-allowed-to-merge js-multiselect wide',
-                             dropdown_class: 'dropdown-menu-user dropdown-menu-selectable', filter: true,
+                             dropdown_class: 'dropdown-menu-user dropdown-menu-selectable  capitalize-header', filter: true,
                              data: { input_id: 'merge_access_levels_attributes', default_label: 'Select' } })
         .form-group
           %label.col-md-2.text-right{ for: 'push_access_levels_attributes' }
@@ -34,7 +34,7 @@
             .push_access_levels-container
               = dropdown_tag('Select',
                              options: { toggle_class: 'js-allowed-to-push js-multiselect wide',
-                             dropdown_class: 'dropdown-menu-user dropdown-menu-selectable', filter: true,
+                             dropdown_class: 'dropdown-menu-user dropdown-menu-selectable  capitalize-header', filter: true,
                              data: { input_id: 'push_access_levels_attributes', default_label: 'Select' } })
             .help-block
               Only groups that
diff --git a/app/views/projects/protected_branches/_update_protected_branch.html.haml b/app/views/projects/protected_branches/_update_protected_branch.html.haml
index d6044aacaec85..0a757b7f0901b 100644
--- a/app/views/projects/protected_branches/_update_protected_branch.html.haml
+++ b/app/views/projects/protected_branches/_update_protected_branch.html.haml
@@ -1,10 +1,10 @@
 %td
   = hidden_field_tag "allowed_to_merge_#{protected_branch.id}", protected_branch.merge_access_levels.first.access_level
   = dropdown_tag( (protected_branch.merge_access_levels.first.humanize || 'Select') ,
-                 options: { toggle_class: 'js-allowed-to-merge', dropdown_class: 'dropdown-menu-selectable js-allowed-to-merge-container',
+                 options: { toggle_class: 'js-allowed-to-merge', dropdown_class: 'dropdown-menu-selectable js-allowed-to-merge-container  capitalize-header',
                  data: { field_name: "allowed_to_merge_#{protected_branch.id}", access_level_id: protected_branch.merge_access_levels.first.id }})
 %td
   = hidden_field_tag "allowed_to_push_#{protected_branch.id}", protected_branch.push_access_levels.first.access_level
   = dropdown_tag( (protected_branch.push_access_levels.first.humanize || 'Select') ,
-                 options: { toggle_class: 'js-allowed-to-push', dropdown_class: 'dropdown-menu-selectable js-allowed-to-push-container',
+                 options: { toggle_class: 'js-allowed-to-push', dropdown_class: 'dropdown-menu-selectable js-allowed-to-push-container  capitalize-header',
                  data: { field_name: "allowed_to_push_#{protected_branch.id}", access_level_id: protected_branch.push_access_levels.first.id }})
diff --git a/app/views/projects/protected_tags/_dropdown.html.haml b/app/views/projects/protected_tags/_dropdown.html.haml
index 748515190779f..c50515cfe06e9 100644
--- a/app/views/projects/protected_tags/_dropdown.html.haml
+++ b/app/views/projects/protected_tags/_dropdown.html.haml
@@ -2,7 +2,7 @@
 
 = dropdown_tag('Select tag or create wildcard',
                options: { toggle_class: 'js-protected-tag-select js-filter-submit wide',
-                          filter: true, dropdown_class: "dropdown-menu-selectable", placeholder: "Search protected tag",
+                          filter: true, dropdown_class: "dropdown-menu-selectable  capitalize-header", placeholder: "Search protected tag",
                           footer_content: true,
                           data: { show_no: true, show_any: true, show_upcoming: true,
                                   selected: params[:protected_tag_name],
diff --git a/app/views/projects/protected_tags/_update_protected_tag.haml b/app/views/projects/protected_tags/_update_protected_tag.haml
index 62823bee46e1f..cc80bd04dd067 100644
--- a/app/views/projects/protected_tags/_update_protected_tag.haml
+++ b/app/views/projects/protected_tags/_update_protected_tag.haml
@@ -1,5 +1,5 @@
 %td
   = hidden_field_tag "allowed_to_create_#{protected_tag.id}", protected_tag.create_access_levels.first.access_level
   = dropdown_tag( (protected_tag.create_access_levels.first.humanize || 'Select') ,
-                 options: { toggle_class: 'js-allowed-to-create', dropdown_class: 'dropdown-menu-selectable js-allowed-to-create-container',
+                 options: { toggle_class: 'js-allowed-to-create', dropdown_class: 'dropdown-menu-selectable capitalize-header js-allowed-to-create-container',
                  data: { field_name: "allowed_to_create_#{protected_tag.id}", access_level_id: protected_tag.create_access_levels.first.id }})
diff --git a/app/views/projects/settings/repository/show.html.haml b/app/views/projects/settings/repository/show.html.haml
index 3f50cc12137da..cfc3faa75b0dd 100644
--- a/app/views/projects/settings/repository/show.html.haml
+++ b/app/views/projects/settings/repository/show.html.haml
@@ -1,6 +1,10 @@
 - page_title "Repository"
 = render "projects/settings/head"
 
+- content_for :page_specific_javascripts do
+  = page_specific_javascript_bundle_tag('common_vue')
+  = page_specific_javascript_bundle_tag('deploy_keys')
+
 = render @deploy_keys
 = render "projects/push_rules/index"
 = render "projects/mirrors/show"
diff --git a/app/views/projects/tags/index.html.haml b/app/views/projects/tags/index.html.haml
index e64bf9bedcc98..1941bd0c08e91 100644
--- a/app/views/projects/tags/index.html.haml
+++ b/app/views/projects/tags/index.html.haml
@@ -1,4 +1,5 @@
 - @no_container = true
+- @sort ||= sort_value_recently_updated
 - page_title "Tags"
 = render "projects/commits/head"
 
@@ -14,16 +15,14 @@
       .dropdown
         %button.dropdown-toggle{ type: 'button', data: { toggle: 'dropdown'} }
           %span.light
-            = projects_sort_options_hash[@sort]
+            = tags_sort_options_hash[@sort]
           = icon('chevron-down')
-        %ul.dropdown-menu.dropdown-menu-align-right
-          %li
-            = link_to filter_tags_path(sort: sort_value_name) do
-              = sort_title_name
-            = link_to filter_tags_path(sort: sort_value_recently_updated) do
-              = sort_title_recently_updated
-            = link_to filter_tags_path(sort: sort_value_oldest_updated) do
-              = sort_title_oldest_updated
+        %ul.dropdown-menu.dropdown-menu-align-right.dropdown-menu-selectable
+          %li.dropdown-header
+            Sort by
+          - tags_sort_options_hash.each do |value, title|
+            %li
+              = link_to title, filter_tags_path(sort: value), class: ("is-active" if @sort == value)
       - if can?(current_user, :push_code, @project)
         = link_to new_namespace_project_tag_path(@project.namespace, @project), class: 'btn btn-create new-tag-btn' do
           New tag
diff --git a/app/views/projects/compare/_ref_dropdown.html.haml b/app/views/shared/_ref_dropdown.html.haml
similarity index 50%
rename from app/views/projects/compare/_ref_dropdown.html.haml
rename to app/views/shared/_ref_dropdown.html.haml
index 05fb37cdc0f52..96f68c80c48cd 100644
--- a/app/views/projects/compare/_ref_dropdown.html.haml
+++ b/app/views/shared/_ref_dropdown.html.haml
@@ -1,4 +1,6 @@
-.dropdown-menu.dropdown-menu-selectable
+- dropdown_class = local_assigns.fetch(:dropdown_class, '')
+
+.dropdown-menu.dropdown-menu-selectable{ class: dropdown_class }
   = dropdown_title "Select Git revision"
   = dropdown_filter "Filter by Git revision"
   = dropdown_content
diff --git a/app/views/shared/errors/_graphic_422.svg b/app/views/shared/errors/_graphic_422.svg
new file mode 100644
index 0000000000000..87128ecd69d54
--- /dev/null
+++ b/app/views/shared/errors/_graphic_422.svg
@@ -0,0 +1 @@
+<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 260 246" xmlns:xlink="http://www.w3.org/1999/xlink"><defs><rect id="0" width="178" height="136" rx="10"/><mask id="1" width="178" height="136" x="0" y="0" fill="#fff"><use xlink:href="#0"/></mask></defs><g fill="none" fill-rule="evenodd"><g fill="#e5e5e5" fill-rule="nonzero"><path d="m109.88 37.634c5.587-3.567 12.225-5.634 19.345-5.634 7.445 0 14.363 2.26 20.1 6.132l21.435-37.13c.554-.959 1.771-1.292 2.734-.736.957.552 1.284 1.777.73 2.736l-21.496 37.23c-.065.112-.138.215-.219.309 3.686 3.13 6.733 6.988 8.919 11.353l-3.393.002c-5.775-10.322-16.705-16.901-28.814-16.901-12.12 0-23.06 6.594-28.833 16.935l-3.393.002c2.32-4.646 5.616-8.72 9.618-11.954l-21.349-36.977c-.554-.959-.227-2.184.73-2.736.963-.556 2.181-.223 2.734.736l21.15 36.629"/><path d="m3 70v134c0 9.389 7.611 17 16.997 17h220.01c9.389 0 16.997-7.611 16.997-17v-134c0-9.389-7.611-17-16.997-17h-220.01c-9.389 0-16.997 7.611-16.997 17m-3 0c0-11.05 8.95-20 19.997-20h220.01c11.04 0 19.997 8.958 19.997 20v134c0 11.05-8.95 20-19.997 20h-220.01c-11.04 0-19.997-8.958-19.997-20v-134"/></g><ellipse cx="129" cy="241.5" fill="#f9f9f9" rx="89" ry="4.5"/><g fill-rule="nonzero" transform="translate(210 70)"><path fill="#eaeaea" d="m16 29c7.18 0 13-5.82 13-13 0-7.18-5.82-13-13-13-7.18 0-13 5.82-13 13 0 7.18 5.82 13 13 13m0 3c-8.837 0-16-7.163-16-16 0-8.837 7.163-16 16-16 8.837 0 16 7.163 16 16 0 8.837-7.163 16-16 16" id="2"/><path fill="#6b4fbb" d="m16 21c2.761 0 5-2.239 5-5 0-2.761-2.239-5-5-5-2.761 0-5 2.239-5 5 0 2.761 2.239 5 5 5m0 3c-4.418 0-8-3.582-8-8 0-4.418 3.582-8 8-8 4.418 0 8 3.582 8 8 0 4.418-3.582 8-8 8" id="3"/></g><g fill-rule="nonzero" transform="translate(210 109)"><use xlink:href="#2"/><use xlink:href="#3"/></g><g transform="translate(210 147)"><path fill="#e5e5e5" fill-rule="nonzero" d="m3 5.992v45.02c0 1.647 1.346 2.992 3 2.992h20c1.657 0 3-1.341 3-2.992v-45.02c0-1.647-1.346-2.992-3-2.992h-20c-1.657 0-3 1.341-3 2.992m-3 0c0-3.309 2.687-5.992 6-5.992h20c3.314 0 6 2.692 6 5.992v45.02c0 3.309-2.687 5.992-6 5.992h-20c-3.314 0-6-2.692-6-5.992v-45.02"/><rect width="16" height="4" x="8" y="27" fill="#fdb692" rx="2"/><rect width="16" height="4" x="8" y="19" fill="#fc9867" rx="2"/><rect width="16" height="4" x="8" y="11" fill="#fc6d26" rx="2"/><rect width="16" height="4" x="8" y="35" fill="#fed3bd" rx="2"/><rect width="16" height="4" x="8" y="43" fill="#fef0e9" rx="2"/></g><g transform="translate(16 69)"><use fill="#6b4fbb" fill-opacity=".1" stroke="#e5e5e5" stroke-width="6" mask="url(#1)" xlink:href="#0"/><g class="tv-screen" fill="#fff"><path opacity=".4" mix-blend-mode="overlay" d="m3 17h172v16h-172z"/><path opacity=".6" mix-blend-mode="overlay" d="m3 70h172v24h-172z"/><path opacity=".3" mix-blend-mode="overlay" d="m3 107h172v16h-172z"/><path opacity=".4" mix-blend-mode="overlay" d="m3 40h172v8h-172z"/><path opacity=".3" mix-blend-mode="overlay" d="m3 55h172v8h-172z"/></g></g><path class="text-422" d="m.693 19h5.808c.277 0 .498-.224.498-.5 0-.268-.223-.5-.498-.5h-5.808v-2.094l3.777-5.906h3.916l-4.124 6.454h6.259v-6.454h.978c.273 0 .5-.224.5-.5 0-.268-.224-.5-.5-.5h-.978v-2h4.698v6h-2.721c-.277 0-.498.224-.498.5 0 .268.223.5.498.5h2.721v2.454h2.723v4.2h-2.723v5.346h-4.698v-5.346h-9.828v-1.654m4.417-10l1.279-2h3.914l-1.278 2h-3.916m1.919-3l1.279-2h4.192c.27 0 .5-.224.5-.5 0-.268-.224-.5-.5-.5h-3.552l1.142-1.786h5.13v4.786h-8.191m31.09 19v1h-15.738v-2h5.118c.271 0 .503-.224.503-.5 0-.268-.225-.5-.503-.5h-5.118v-1.184l2.656-2.822c.682-.725 1.306-1.39 1.872-1.994h5.428c-.389.394-.808.815-1.256 1.264-1.428 1.428-2.562 2.568-3.403 3.42h10.442v2.316h-4.614c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h4.614m-6.674-13c.493-.631.87-1.208 1.129-1.73.365-.736.548-1.464.548-2.183 0-1.107-.335-1.962-1-2.565-.67-.603-1.619-.905-2.847-.905-.874 0-1.857.174-2.947.523-1.09.349-2.227.855-3.412 1.519v-2.659h3.589c.27 0 .5-.224.5-.5 0-.268-.224-.5-.5-.5h-3.589v-.906c1.184-.432 2.344-.761 3.478-.988 1.134-.227 2.222-.34 3.262-.34 2.623 0 4.684.611 6.184 1.834.157.128.307.262.448.4h-2.782c-.27 0-.5.224-.5.5 0 .268.224.5.5.5h3.602c.654 1.01.981 2.209.981 3.605 0 .974-.163 1.887-.49 2.739-.326.852-.888 1.798-1.685 2.839-.397.509-1.261 1.448-2.594 2.816h-5.474c1.34-1.436 2.261-2.436 2.763-3h4.396c.271 0 .499-.224.499-.5 0-.268-.223-.5-.499-.5h-3.557m28.14 12v2h-15.738v-4.184l2.651-2.816h5.313c-1.087 1.089-1.976 1.983-2.668 2.684h10.442v1.316h-4.083c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h4.083m-2.069-11c-.045.061-.092.122-.139.184-.567.727-2.089 2.333-4.568 4.816h-5.372c2.601-2.77 4.204-4.503 4.81-5.198.83-.952 1.428-1.796 1.793-2.532.365-.736.548-1.464.548-2.183 0-1.107-.335-1.962-1-2.565-.67-.603-1.619-.905-2.847-.905-.874 0-1.857.174-2.947.523-1.09.349-2.227.855-3.412 1.519v-2.659h3.117c.271 0 .503-.224.503-.5 0-.268-.225-.5-.503-.5h-3.117v-.906c1.184-.432 2.344-.761 3.478-.988 1.134-.227 2.222-.34 3.262-.34 2.623 0 4.684.611 6.184 1.834.157.128.307.262.448.4h-1.248c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h2.069c.654 1.01.981 2.209.981 3.605 0 .844-.123 1.642-.368 2.395h-2.683c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h2.272c-.159.321-.347.655-.566 1h-3.706c-.271 0-.503.224-.503.5 0 .268.225.5.503.5h3.01" transform="translate(75 124)" fill="#5c5c5c"/></g></svg>
diff --git a/app/views/shared/issuable/_assignees.html.haml b/app/views/shared/issuable/_assignees.html.haml
new file mode 100644
index 0000000000000..36bbb1148d49b
--- /dev/null
+++ b/app/views/shared/issuable/_assignees.html.haml
@@ -0,0 +1,15 @@
+- max_render = 3
+- max = [max_render, issue.assignees.length].min
+
+- issue.assignees.each_with_index do |assignee, index|
+  - if index < max
+    = link_to_member(@project, assignee, name: false, title: "Assigned to :name")
+
+- if issue.assignees.length > max_render
+  - counter = issue.assignees.length - max_render
+
+  %span{ class: 'avatar-counter has-tooltip', data: { container: 'body', placement: 'bottom', 'line-type' => 'old', 'original-title' => "+#{counter} more assignees" } }
+    - if counter < 99
+      = "+#{counter}"
+    - else
+      99+
diff --git a/app/views/shared/issuable/_participants.html.haml b/app/views/shared/issuable/_participants.html.haml
index 171da89993736..db407363a0929 100644
--- a/app/views/shared/issuable/_participants.html.haml
+++ b/app/views/shared/issuable/_participants.html.haml
@@ -12,9 +12,9 @@
     - participants.each do |participant|
       .participants-author.js-participants-author
         = link_to_member(@project, participant, name: false, size: 24)
-    - if participants_extra > 0
-      .participants-more
-        %a.js-participants-more{ href: "#", data: { original_text: "+ #{participants_size - 7} more", less_text: "- show less" } }
-          + #{participants_extra} more
+  - if participants_extra > 0
+    .hide-collapsed.participants-more
+      %a.js-participants-more{ href: "#", data: { original_text: "+ #{participants_size - 7} more", less_text: "- show less" } }
+        + #{participants_extra} more
 :javascript
   IssuableContext.prototype.PARTICIPANTS_ROW_COUNT = #{participants_row};
diff --git a/app/views/shared/issuable/_search_bar.html.haml b/app/views/shared/issuable/_search_bar.html.haml
index 3439775b10899..c99bec21774e0 100644
--- a/app/views/shared/issuable/_search_bar.html.haml
+++ b/app/views/shared/issuable/_search_bar.html.haml
@@ -147,8 +147,13 @@
                 %li
                   %a{ href: "#", data: { id: "close" } } Closed
           .filter-item.inline
+            - if type == :issues
+              - field_name = "update[assignee_ids][]"
+            - else
+              - field_name = "update[assignee_id]"
+
             = dropdown_tag("Assignee", options: { toggle_class: "js-user-search js-update-assignee js-filter-submit js-filter-bulk-update", title: "Assign to", filter: true, dropdown_class: "dropdown-menu-user dropdown-menu-selectable",
-              placeholder: "Search authors", data: { first_user: (current_user.username if current_user), null_user: true, current_user: true, project_id: @project.id, field_name: "update[assignee_id]", default_label: "Assignee" } })
+              placeholder: "Search authors", data: { first_user: (current_user.username if current_user), null_user: true, current_user: true, project_id: @project.id, field_name: field_name } })
           .filter-item.inline
             = dropdown_tag("Milestone", options: { title: "Assign milestone", toggle_class: 'js-milestone-select js-extra-options js-filter-submit js-filter-bulk-update', filter: true, dropdown_class: "dropdown-menu-selectable dropdown-menu-milestone", placeholder: "Search milestones", data: { show_no: true, field_name: "update[milestone_id]", project_id: @project.id, milestones: namespace_project_milestones_path(@project.namespace, @project, :json), use_id: true, default_label: "Milestone" } })
           .filter-item.inline.labels-filter
diff --git a/app/views/shared/issuable/_sidebar.html.haml b/app/views/shared/issuable/_sidebar.html.haml
index eec97e9000957..95d30831aaac0 100644
--- a/app/views/shared/issuable/_sidebar.html.haml
+++ b/app/views/shared/issuable/_sidebar.html.haml
@@ -1,10 +1,10 @@
 - todo = issuable_todo(issuable)
 - content_for :page_specific_javascripts do
   = page_specific_javascript_bundle_tag('common_vue')
-  = page_specific_javascript_bundle_tag('issuable')
+  = page_specific_javascript_bundle_tag('sidebar')
 
 %aside.right-sidebar.js-right-sidebar{ data: { "offset-top" => "102", "spy" => "affix" }, class: sidebar_gutter_collapsed_class, 'aria-live' => 'polite' }
-  .issuable-sidebar
+  .issuable-sidebar{ data: { endpoint: "#{issuable_json_path(issuable)}" } }
     - can_edit_issuable = can?(current_user, :"admin_#{issuable.to_ability_name}", @project)
     .block.issuable-sidebar-header
       - if current_user
@@ -20,36 +20,54 @@
         .block.todo.hide-expanded
           = render "shared/issuable/sidebar_todo", todo: todo, issuable: issuable, is_collapsed: true
       .block.assignee
-        .sidebar-collapsed-icon.sidebar-collapsed-user{ data: { toggle: "tooltip", placement: "left", container: "body" }, title: (issuable.assignee.name if issuable.assignee) }
-          - if issuable.assignee
-            = link_to_member(@project, issuable.assignee, size: 24)
-          - else
-            = icon('user', 'aria-hidden': 'true')
-        .title.hide-collapsed
-          Assignee
-          = icon('spinner spin', class: 'hidden block-loading', 'aria-hidden': 'true')
-          - if can_edit_issuable
-            = link_to 'Edit', '#', class: 'edit-link pull-right'
-        .value.hide-collapsed
-          - if issuable.assignee
-            = link_to_member(@project, issuable.assignee, size: 32, extra_class: 'bold') do
-              - if issuable.instance_of?(MergeRequest) && !issuable.can_be_merged_by?(issuable.assignee)
-                %span.pull-right.cannot-be-merged{ data: { toggle: 'tooltip', placement: 'left' }, title: 'Not allowed to merge' }
-                  = icon('exclamation-triangle', 'aria-hidden': 'true')
-              %span.username
-                = issuable.assignee.to_reference
-          - else
-            %span.assign-yourself.no-value
-              No assignee
-              - if can_edit_issuable
-                \-
-                %a.js-assign-yourself{ href: '#' }
-                  assign yourself
+        - if issuable.instance_of?(Issue)
+          #js-vue-sidebar-assignees{ data: { field: "#{issuable.to_ability_name}[assignee_ids]" } }
+        - else
+          .sidebar-collapsed-icon.sidebar-collapsed-user{ data: { toggle: "tooltip", placement: "left", container: "body" }, title: (issuable.assignee.name if issuable.assignee) }
+            - if issuable.assignee
+              = link_to_member(@project, issuable.assignee, size: 24)
+            - else
+              = icon('user', 'aria-hidden': 'true')
+          .title.hide-collapsed
+            Assignee
+            = icon('spinner spin', class: 'hidden block-loading', 'aria-hidden': 'true')
+            - if can_edit_issuable
+              = link_to 'Edit', '#', class: 'edit-link pull-right'
+          .value.hide-collapsed
+            - if issuable.assignee
+              = link_to_member(@project, issuable.assignee, size: 32, extra_class: 'bold') do
+                - if issuable.instance_of?(MergeRequest) && !issuable.can_be_merged_by?(issuable.assignee)
+                  %span.pull-right.cannot-be-merged{ data: { toggle: 'tooltip', placement: 'left' }, title: 'Not allowed to merge' }
+                    = icon('exclamation-triangle', 'aria-hidden': 'true')
+                %span.username
+                  = issuable.assignee.to_reference
+            - else
+              %span.assign-yourself.no-value
+                No assignee
+                - if can_edit_issuable
+                  \-
+                  %a.js-assign-yourself{ href: '#' }
+                    assign yourself
 
         .selectbox.hide-collapsed
-          = f.hidden_field 'assignee_id', value: issuable.assignee_id, id: 'issue_assignee_id'
-          = dropdown_tag('Select assignee', options: { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true, null_user_default: true, selected: issuable.assignee_id } })
+          - issuable.assignees.each do |assignee|
+            = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", assignee.id, id: nil
 
+          - options = { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true } }
+
+          - if issuable.instance_of?(Issue)
+            - if issuable.assignees.length == 0
+              = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", 0, id: nil
+            - title = 'Select assignee(s)'
+            - options[:toggle_class] += ' js-multiselect js-save-user-data'
+            - options[:data][:field_name] = "#{issuable.to_ability_name}[assignee_ids][]"
+            - options[:data][:multi_select] = true
+            - options[:data]['dropdown-title'] = title
+            - options[:data]['dropdown-header'] = 'Assignee(s)'
+          - else
+            - title = 'Select assignee'
+
+          = dropdown_tag(title, options: options)
       .block.milestone
         .sidebar-collapsed-icon
           = icon('clock-o', 'aria-hidden': 'true')
@@ -75,11 +93,10 @@
           = dropdown_tag('Milestone', options: { title: 'Assign milestone', toggle_class: 'js-milestone-select js-extra-options', filter: true, dropdown_class: 'dropdown-menu-selectable', placeholder: 'Search milestones', data: { show_no: true, field_name: "#{issuable.to_ability_name}[milestone_id]", project_id: @project.id, issuable_id: issuable.id, milestones: namespace_project_milestones_path(@project.namespace, @project, :json), ability_name: issuable.to_ability_name, issue_update: issuable_json_path(issuable), use_id: true }})
       - if issuable.has_attribute?(:time_estimate)
         #issuable-time-tracker.block
-          %issuable-time-tracker{ ':time_estimate' => 'issuable.time_estimate', ':time_spent' => 'issuable.total_time_spent', ':human_time_estimate' => 'issuable.human_time_estimate', ':human_time_spent' => 'issuable.human_total_time_spent', 'docs-url' => help_page_path('workflow/time_tracking.md') }
-            // Fallback while content is loading
-            .title.hide-collapsed
-              Time tracking
-              = icon('spinner spin', 'aria-hidden': 'true')
+          // Fallback while content is loading
+          .title.hide-collapsed
+            Time tracking
+            = icon('spinner spin', 'aria-hidden': 'true')
       - if issuable.has_attribute?(:due_date)
         .block.due_date
           .sidebar-collapsed-icon
@@ -196,8 +213,13 @@
           = clipboard_button(text: project_ref, title: "Copy reference to clipboard", placement: "left")
 
     :javascript
-      gl.IssuableResource = new gl.SubbableResource('#{issuable_json_path(issuable)}');
-      new gl.IssuableTimeTracking("#{escape_javascript(serialize_issuable(issuable))}");
+      gl.sidebarOptions = {
+        endpoint: "#{issuable_json_path(issuable)}",
+        editable: #{can_edit_issuable ? true : false},
+        currentUser: #{current_user.to_json(only: [:username, :id, :name], methods: :avatar_url)},
+        rootPath: "#{root_path}"
+      };
+
       new MilestoneSelect('{"full_path":"#{@project.full_path}"}');
       new LabelsSelect();
       new WeightSelect();
diff --git a/app/views/shared/issuable/form/_branch_chooser.html.haml b/app/views/shared/issuable/form/_branch_chooser.html.haml
index 2793e7bcff430..f57b4d899ce98 100644
--- a/app/views/shared/issuable/form/_branch_chooser.html.haml
+++ b/app/views/shared/issuable/form/_branch_chooser.html.haml
@@ -10,12 +10,16 @@
     = form.label :source_branch, class: 'control-label'
     .col-sm-10
       .issuable-form-select-holder
-        = form.select(:source_branch, [issuable.source_branch], {}, { class: 'source_branch select2 span2', disabled: true })
+        = form.select(:source_branch, [issuable.source_branch], {}, { class: 'source_branch select2', disabled: true })
 .form-group
   = form.label :target_branch, class: 'control-label'
-  .col-sm-10
+  .col-sm-10.target-branch-select-dropdown-container
     .issuable-form-select-holder
-      = form.select(:target_branch, issuable.target_branches, { include_blank: true }, { class: 'target_branch select2 span2', disabled: issuable.new_record?, data: { placeholder: "Select branch" }})
+      = form.select(:target_branch, issuable.target_branches,
+        { include_blank: true },
+        { class: 'target_branch js-target-branch-select',
+          disabled: issuable.new_record?,
+          data: { placeholder: "Select branch" }})
     - if issuable.new_record?
       &nbsp;
       = link_to 'Change branches', mr_change_branches_path(issuable)
diff --git a/app/views/shared/issuable/form/_issue_assignee.html.haml b/app/views/shared/issuable/form/_issue_assignee.html.haml
new file mode 100644
index 0000000000000..c33474ac3b4e7
--- /dev/null
+++ b/app/views/shared/issuable/form/_issue_assignee.html.haml
@@ -0,0 +1,30 @@
+- issue = issuable
+.block.assignee
+  .sidebar-collapsed-icon.sidebar-collapsed-user{ data: { toggle: "tooltip", placement: "left", container: "body" }, title: (issuable.assignee_list) }
+    - if issue.assignees.any?
+      - issue.assignees.each do |assignee|
+        = link_to_member(@project, assignee, size: 24)
+    - else
+      = icon('user', 'aria-hidden': 'true')
+  .title.hide-collapsed
+    Assignee
+    = icon('spinner spin', class: 'hidden block-loading', 'aria-hidden': 'true')
+    - if can_edit_issuable
+      = link_to 'Edit', '#', class: 'edit-link pull-right'
+  .value.hide-collapsed
+    - if issue.assignees.any?
+      - issue.assignees.each do |assignee|
+        = link_to_member(@project, assignee, size: 32, extra_class: 'bold') do
+          %span.username
+            = assignee.to_reference
+    - else
+      %span.assign-yourself.no-value
+        No assignee
+        - if can_edit_issuable
+          \-
+          %a.js-assign-yourself{ href: '#' }
+            assign yourself
+
+  .selectbox.hide-collapsed
+    = f.hidden_field 'assignee_ids', value: issuable.assignee_ids, id: 'issue_assignee_ids'
+    = dropdown_tag('Select assignee', options: { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: (@project.id if @project), author_id: issuable.author_id, field_name: "#{issuable.to_ability_name}[assignee_id]", issue_update: issuable_json_path(issuable), ability_name: issuable.to_ability_name, null_user: true } })
diff --git a/app/views/shared/issuable/form/_merge_request_assignee.html.haml b/app/views/shared/issuable/form/_merge_request_assignee.html.haml
new file mode 100644
index 0000000000000..18011d528a0be
--- /dev/null
+++ b/app/views/shared/issuable/form/_merge_request_assignee.html.haml
@@ -0,0 +1,31 @@
+- merge_request = issuable
+.block.assignee
+  .sidebar-collapsed-icon.sidebar-collapsed-user{ data: { toggle: "tooltip", placement: "left", container: "body" }, title: (merge_request.assignee.name if merge_request.assignee) }
+    - if merge_request.assignee
+      = link_to_member(@project, merge_request.assignee, size: 24)
+    - else
+      = icon('user', 'aria-hidden': 'true')
+  .title.hide-collapsed
+    Assignee
+    = icon('spinner spin', class: 'hidden block-loading', 'aria-hidden': 'true')
+    - if can_edit_issuable
+      = link_to 'Edit', '#', class: 'edit-link pull-right'
+  .value.hide-collapsed
+    - if merge_request.assignee
+      = link_to_member(@project, merge_request.assignee, size: 32, extra_class: 'bold') do
+        - unless merge_request.can_be_merged_by?(merge_request.assignee)
+          %span.pull-right.cannot-be-merged{ data: { toggle: 'tooltip', placement: 'left' }, title: 'Not allowed to merge' }
+            = icon('exclamation-triangle', 'aria-hidden': 'true')
+        %span.username
+          = merge_request.assignee.to_reference
+    - else
+      %span.assign-yourself.no-value
+        No assignee
+        - if can_edit_issuable
+          \-
+          %a.js-assign-yourself{ href: '#' }
+            assign yourself
+
+  .selectbox.hide-collapsed
+    = f.hidden_field 'assignee_id', value: merge_request.assignee_id, id: 'issue_assignee_id'
+    = dropdown_tag('Select assignee', options: { toggle_class: 'js-user-search js-author-search', title: 'Assign to', filter: true, dropdown_class: 'dropdown-menu-user dropdown-menu-selectable dropdown-menu-author', placeholder: 'Search users', data: { first_user: (current_user.username if current_user), current_user: true, project_id: @project&.id, author_id: merge_request.author_id, field_name: 'merge_request[assignee_id]', issue_update: issuable_json_path(merge_request), ability_name: 'merge_request', null_user: true } })
diff --git a/app/views/shared/issuable/form/_metadata.html.haml b/app/views/shared/issuable/form/_metadata.html.haml
index 5d5ba7752d1f4..651d19754f800 100644
--- a/app/views/shared/issuable/form/_metadata.html.haml
+++ b/app/views/shared/issuable/form/_metadata.html.haml
@@ -10,13 +10,27 @@
 .row
   %div{ class: (has_due_date ? "col-lg-6" : "col-sm-12") }
     .form-group.issue-assignee
-      = form.label :assignee_id, "Assignee", class: "control-label #{"col-lg-4" if has_due_date}"
-      .col-sm-10{ class: ("col-lg-8" if has_due_date) }
-        .issuable-form-select-holder
-          = form.hidden_field :assignee_id
-          = dropdown_tag(user_dropdown_label(issuable.assignee_id, "Assignee"), options: { toggle_class: "js-dropdown-keep-input js-user-search js-issuable-form-dropdown js-assignee-search", title: "Select assignee", filter: true, dropdown_class: "dropdown-menu-user dropdown-menu-selectable dropdown-menu-assignee js-filter-submit",
-            placeholder: "Search assignee", data: { first_user: current_user.try(:username), null_user: true, current_user: true, project_id: issuable.project.try(:id), selected: issuable.assignee_id, field_name: "#{issuable.class.model_name.param_key}[assignee_id]", default_label: "Assignee"} })
-        = link_to 'Assign to me', '#', class: "assign-to-me-link #{'hide' if issuable.assignee_id == current_user.id}"
+      - if issuable.is_a?(Issue)
+        = form.label :assignee_ids, "Assignee", class: "control-label #{"col-lg-4" if has_due_date}"
+        .col-sm-10{ class: ("col-lg-8" if has_due_date) }
+          .issuable-form-select-holder.selectbox
+            - issuable.assignees.each do |assignee|
+              = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", assignee.id, id: nil, data: { meta: assignee.name }
+
+            - if issuable.assignees.length === 0
+              = hidden_field_tag "#{issuable.to_ability_name}[assignee_ids][]", 0, id: nil, data: { meta: '' }
+
+            = dropdown_tag(users_dropdown_label(issuable.assignees), options: issue_dropdown_options(issuable, true))
+          = link_to 'Assign to me', '#', class: "assign-to-me-link #{'hide' if issuable.assignees.include?(current_user)}"
+      - else
+        = form.label :assignee_id, "Assignee", class: "control-label #{"col-lg-4" if has_due_date}"
+        .col-sm-10{ class: ("col-lg-8" if has_due_date) }
+          .issuable-form-select-holder
+            = form.hidden_field :assignee_id
+
+            = dropdown_tag(user_dropdown_label(issuable.assignee_id, "Assignee"), options: { toggle_class: "js-dropdown-keep-input js-user-search js-issuable-form-dropdown js-assignee-search", title: "Select assignee", filter: true, dropdown_class: "dropdown-menu-user dropdown-menu-selectable dropdown-menu-assignee js-filter-submit",
+              placeholder: "Search assignee", data: { first_user: current_user.try(:username), null_user: true, current_user: true, project_id: issuable.project.try(:id), selected: issuable.assignee_id, field_name: "#{issuable.class.model_name.param_key}[assignee_id]", default_label: "Assignee"} })
+          = link_to 'Assign to me', '#', class: "assign-to-me-link #{'hide' if issuable.assignee_id == current_user.id}"
     .form-group.issue-milestone
       = form.label :milestone_id, "Milestone", class: "control-label #{"col-lg-4" if has_due_date}"
       .col-sm-10{ class: ("col-lg-8" if has_due_date) }
diff --git a/app/views/shared/members/_requests.html.haml b/app/views/shared/members/_requests.html.haml
index 10050adfda5a4..92f6e7428ae2e 100644
--- a/app/views/shared/members/_requests.html.haml
+++ b/app/views/shared/members/_requests.html.haml
@@ -1,5 +1,5 @@
 - if requesters.any?
-  .panel.panel-default
+  .panel.panel-default.prepend-top-default
     .panel-heading
       Users requesting access to
       %strong= membership_source.name
diff --git a/app/views/shared/milestones/_issuable.html.haml b/app/views/shared/milestones/_issuable.html.haml
index 5247d6a51e642..22547a30cdfb1 100644
--- a/app/views/shared/milestones/_issuable.html.haml
+++ b/app/views/shared/milestones/_issuable.html.haml
@@ -1,7 +1,7 @@
 -# @project is present when viewing Project's milestone
 - project = @project || issuable.project
 - namespace = @project_namespace || project.namespace.becomes(Namespace)
-- assignee = issuable.assignee
+- assignees = issuable.assignees
 - issuable_type = issuable.class.table_name
 - base_url_args = [namespace, project]
 - issuable_type_args = base_url_args + [issuable_type]
@@ -26,7 +26,7 @@
         - render_colored_label(label)
 
     %span.assignee-icon
-      - if assignee
-        = link_to polymorphic_path(issuable_type_args, { milestone_title: @milestone.title, assignee_id: issuable.assignee_id, state: 'all' }),
+      - assignees.each do |assignee|
+        = link_to polymorphic_path(base_url_args, { milestone_title: @milestone.title, assignee_id: assignee.id, state: 'all' }),
                   class: 'has-tooltip', title: "Assigned to #{assignee.name}", data: { container: 'body' } do
-          - image_tag(avatar_icon(issuable.assignee, 16), class: "avatar s16", alt: '')
+          - image_tag(avatar_icon(assignee, 16), class: "avatar s16", alt: '')
diff --git a/app/views/shared/milestones/_labels_tab.html.haml b/app/views/shared/milestones/_labels_tab.html.haml
index 33f93dccd3c48..a26b3b8009e96 100644
--- a/app/views/shared/milestones/_labels_tab.html.haml
+++ b/app/views/shared/milestones/_labels_tab.html.haml
@@ -2,7 +2,7 @@
   - labels.each do |label|
     - options = { milestone_title: @milestone.title, label_name: label.title }
 
-    %li
+    %li.is-not-draggable
       %span.label-row
         %span.label-name
           = link_to milestones_label_path(options) do
@@ -10,10 +10,8 @@
         %span.prepend-description-left
           = markdown_field(label, :description)
 
-      .pull-info-right
-        %span.append-right-20
-          = link_to milestones_label_path(options.merge(state: 'opened')) do
-            - pluralize milestone_issues_by_label_count(@milestone, label, state: :opened), 'open issue'
-        %span.append-right-20
-          = link_to milestones_label_path(options.merge(state: 'closed')) do
-            - pluralize milestone_issues_by_label_count(@milestone, label, state: :closed), 'closed issue'
+      .pull-right.hidden-xs.hidden-sm.hidden-md
+        = link_to milestones_label_path(options.merge(state: 'opened')), class: 'btn btn-transparent btn-action' do
+          - pluralize milestone_issues_by_label_count(@milestone, label, state: :opened), 'open issue'
+        = link_to milestones_label_path(options.merge(state: 'closed')), class: 'btn btn-transparent btn-action' do
+          - pluralize milestone_issues_by_label_count(@milestone, label, state: :closed), 'closed issue'
diff --git a/app/views/shared/notes/_edit_form.html.haml b/app/views/shared/notes/_edit_form.html.haml
index fc57b92d4a6ca..8923e5602a41f 100644
--- a/app/views/shared/notes/_edit_form.html.haml
+++ b/app/views/shared/notes/_edit_form.html.haml
@@ -9,6 +9,6 @@
     .note-form-actions.clearfix
       .settings-message.note-edit-warning.js-finish-edit-warning
         Finish editing this message first!
-      = submit_tag 'Save comment', class: 'btn btn-nr btn-save js-comment-button'
+      = submit_tag 'Save comment', class: 'btn btn-nr btn-save js-comment-save-button'
       %button.btn.btn-nr.btn-cancel.note-edit-cancel{ type: 'button' }
         Cancel
diff --git a/app/views/shared/notes/_note.html.haml b/app/views/shared/notes/_note.html.haml
index 071c48fa2e494..5c1156b06fb16 100644
--- a/app/views/shared/notes/_note.html.haml
+++ b/app/views/shared/notes/_note.html.haml
@@ -40,7 +40,7 @@
       .note-body{ class: note_editable ? 'js-task-list-container' : '' }
         .note-text.md
           = note.redacted_note_html
-        = edited_time_ago_with_tooltip(note, placement: 'bottom', html_class: 'note_edited_ago', include_author: true)
+        = edited_time_ago_with_tooltip(note, placement: 'bottom', html_class: 'note_edited_ago')
         - if note_editable
           = render 'shared/notes/edit', note: note
         .note-awards
diff --git a/app/views/shared/snippets/_header.html.haml b/app/views/shared/snippets/_header.html.haml
index d084f5e968416..501c09d71d510 100644
--- a/app/views/shared/snippets/_header.html.haml
+++ b/app/views/shared/snippets/_header.html.haml
@@ -21,4 +21,4 @@
     = markdown_field(@snippet, :title)
 
   - if @snippet.updated_at != @snippet.created_at
-    = edited_time_ago_with_tooltip(@snippet, placement: 'bottom', html_class: 'snippet-edited-ago')
+    = edited_time_ago_with_tooltip(@snippet, placement: 'bottom', html_class: 'snippet-edited-ago', exclude_author: true)
diff --git a/app/views/snippets/notes/_actions.html.haml b/app/views/snippets/notes/_actions.html.haml
index dace11e547461..679a5e934da0e 100644
--- a/app/views/snippets/notes/_actions.html.haml
+++ b/app/views/snippets/notes/_actions.html.haml
@@ -1,13 +1,13 @@
 - if current_user
   - if note.emoji_awardable?
     - user_authored = note.user_authored?(current_user)
-    = link_to '#', title: 'Award Emoji', class: "note-action-button note-emoji-button js-add-award js-note-emoji #{'js-user-authored' if user_authored}", data: { position: 'right' } do
+    = link_to '#', title: 'Award Emoji', class: "note-action-button note-emoji-button js-add-award js-note-emoji #{'js-user-authored' if user_authored} has-tooltip", data: { position: 'right' } do
       = icon('spinner spin')
       %span{ class: 'link-highlight award-control-icon-neutral' }= custom_icon('emoji_slightly_smiling_face')
       %span{ class: 'link-highlight award-control-icon-positive' }= custom_icon('emoji_smiley')
       %span{ class: 'link-highlight award-control-icon-super-positive' }= custom_icon('emoji_smile')
   - if note_editable
-    = link_to '#', title: 'Edit comment', class: 'note-action-button js-note-edit' do
+    = link_to '#', title: 'Edit comment', class: 'note-action-button js-note-edit has-tooltip' do
       = icon('pencil', class: 'link-highlight')
-    = link_to snippet_note_path(note.noteable, note), title: 'Remove comment', method: :delete, data: { confirm: 'Are you sure you want to remove this comment?' }, remote: true, class: 'note-action-button js-note-delete danger' do
+    = link_to snippet_note_path(note.noteable, note), title: 'Remove comment', method: :delete, data: { confirm: 'Are you sure you want to remove this comment?' }, remote: true, class: 'note-action-button js-note-delete danger has-tooltip' do
       = icon('trash-o', class: 'danger-highlight')
diff --git a/app/workers/elastic_indexer_worker.rb b/app/workers/elastic_indexer_worker.rb
index 0db28180259b2..846c70647455f 100644
--- a/app/workers/elastic_indexer_worker.rb
+++ b/app/workers/elastic_indexer_worker.rb
@@ -6,7 +6,7 @@ class ElasticIndexerWorker
 
   sidekiq_options retry: 2
 
-  ISSUE_TRACKED_FIELDS = %w(assignee_id author_id confidential).freeze
+  ISSUE_TRACKED_FIELDS = %w(assignee_ids author_id confidential).freeze
 
   def perform(operation, class_name, record_id, options = {})
     return true unless current_application_settings.elasticsearch_indexing?
diff --git a/app/workers/propagate_service_template_worker.rb b/app/workers/propagate_service_template_worker.rb
new file mode 100644
index 0000000000000..5ce0e0405d0ac
--- /dev/null
+++ b/app/workers/propagate_service_template_worker.rb
@@ -0,0 +1,21 @@
+# Worker for updating any project specific caches.
+class PropagateServiceTemplateWorker
+  include Sidekiq::Worker
+  include DedicatedSidekiqQueue
+
+  LEASE_TIMEOUT = 4.hours.to_i
+
+  def perform(template_id)
+    return unless try_obtain_lease_for(template_id)
+
+    Projects::PropagateServiceTemplate.propagate(Service.find_by(id: template_id))
+  end
+
+  private
+
+  def try_obtain_lease_for(template_id)
+    Gitlab::ExclusiveLease.
+      new("propagate_service_template_worker:#{template_id}", timeout: LEASE_TIMEOUT).
+      try_obtain
+  end
+end
diff --git a/changelogs/unreleased/24883-build-failure-summary-page.yml b/changelogs/unreleased/24883-build-failure-summary-page.yml
new file mode 100644
index 0000000000000..214cd3e2bc71f
--- /dev/null
+++ b/changelogs/unreleased/24883-build-failure-summary-page.yml
@@ -0,0 +1,4 @@
+---
+title: Added build failures summary page for pipelines
+merge_request: 10719
+author:
diff --git a/changelogs/unreleased/27614-instant-comments.yml b/changelogs/unreleased/27614-instant-comments.yml
new file mode 100644
index 0000000000000..7b2592f46ede2
--- /dev/null
+++ b/changelogs/unreleased/27614-instant-comments.yml
@@ -0,0 +1,4 @@
+---
+title: Add support for instantly updating comments
+merge_request: 10760
+author:
diff --git a/changelogs/unreleased/29145-oauth-422.yml b/changelogs/unreleased/29145-oauth-422.yml
new file mode 100644
index 0000000000000..94e4cd84ad195
--- /dev/null
+++ b/changelogs/unreleased/29145-oauth-422.yml
@@ -0,0 +1,4 @@
+---
+title: Redesign auth 422 page
+merge_request:
+author:
diff --git a/changelogs/unreleased/30007-done-todo-hover-state.yml b/changelogs/unreleased/30007-done-todo-hover-state.yml
new file mode 100644
index 0000000000000..bfbde7a49c825
--- /dev/null
+++ b/changelogs/unreleased/30007-done-todo-hover-state.yml
@@ -0,0 +1,4 @@
+---
+title: Add transparent top-border to the hover state of done todos
+merge_request:
+author:
diff --git a/changelogs/unreleased/30903-vertically-align-mini-pipeline.yml b/changelogs/unreleased/30903-vertically-align-mini-pipeline.yml
new file mode 100644
index 0000000000000..af87e5ce39f17
--- /dev/null
+++ b/changelogs/unreleased/30903-vertically-align-mini-pipeline.yml
@@ -0,0 +1,4 @@
+---
+title: Vertically align mini pipeline stage container
+merge_request:
+author:
diff --git a/changelogs/unreleased/30973-network-graph-sorted-by-date-and-topo.yml b/changelogs/unreleased/30973-network-graph-sorted-by-date-and-topo.yml
new file mode 100644
index 0000000000000..42426c1865e72
--- /dev/null
+++ b/changelogs/unreleased/30973-network-graph-sorted-by-date-and-topo.yml
@@ -0,0 +1,4 @@
+---
+title: Sort the network graph both by commit date and topographically
+merge_request: 11057
+author:
diff --git a/changelogs/unreleased/31689-request-access-spacing.yml b/changelogs/unreleased/31689-request-access-spacing.yml
new file mode 100644
index 0000000000000..66076b44f466a
--- /dev/null
+++ b/changelogs/unreleased/31689-request-access-spacing.yml
@@ -0,0 +1,4 @@
+---
+title: Add default margin-top to user request table on project members page
+merge_request:
+author:
diff --git a/changelogs/unreleased/31760-add-tooltips-to-note-actions.yml b/changelogs/unreleased/31760-add-tooltips-to-note-actions.yml
new file mode 100644
index 0000000000000..9bbf43d652e37
--- /dev/null
+++ b/changelogs/unreleased/31760-add-tooltips-to-note-actions.yml
@@ -0,0 +1,4 @@
+---
+title: Add tooltips to note action buttons
+merge_request:
+author:
diff --git a/changelogs/unreleased/31810-commit-link.yml b/changelogs/unreleased/31810-commit-link.yml
new file mode 100644
index 0000000000000..857c9cb95c595
--- /dev/null
+++ b/changelogs/unreleased/31810-commit-link.yml
@@ -0,0 +1,4 @@
+---
+title: Remove `#` being added on commit sha in MR widget
+merge_request:
+author:
diff --git a/changelogs/unreleased/add_system_note_for_editing_issuable.yml b/changelogs/unreleased/add_system_note_for_editing_issuable.yml
new file mode 100644
index 0000000000000..3cbc7f91bf007
--- /dev/null
+++ b/changelogs/unreleased/add_system_note_for_editing_issuable.yml
@@ -0,0 +1,4 @@
+---
+title: Add system note on description change of issue/merge request
+merge_request: 10392
+author: blackst0ne
diff --git a/changelogs/unreleased/balsalmiq-support.yml b/changelogs/unreleased/balsalmiq-support.yml
new file mode 100644
index 0000000000000..56a0b4c83fae6
--- /dev/null
+++ b/changelogs/unreleased/balsalmiq-support.yml
@@ -0,0 +1,4 @@
+---
+title: Added balsamiq file viewer
+merge_request: 10564
+author:
diff --git a/changelogs/unreleased/deploy-keys-load-async.yml b/changelogs/unreleased/deploy-keys-load-async.yml
new file mode 100644
index 0000000000000..e90910278e807
--- /dev/null
+++ b/changelogs/unreleased/deploy-keys-load-async.yml
@@ -0,0 +1,4 @@
+---
+title: Deploy keys load are loaded async
+merge_request:
+author:
diff --git a/changelogs/unreleased/dm-artifact-blob-viewer.yml b/changelogs/unreleased/dm-artifact-blob-viewer.yml
new file mode 100644
index 0000000000000..38f5cbb73e18e
--- /dev/null
+++ b/changelogs/unreleased/dm-artifact-blob-viewer.yml
@@ -0,0 +1,4 @@
+---
+title: Add artifact file page that uses the blob viewer
+merge_request:
+author:
diff --git a/changelogs/unreleased/fix-admin-integrations.yml b/changelogs/unreleased/fix-admin-integrations.yml
new file mode 100644
index 0000000000000..7689623501ffc
--- /dev/null
+++ b/changelogs/unreleased/fix-admin-integrations.yml
@@ -0,0 +1,4 @@
+---
+title: Fix new admin integrations not taking effect on existing projects
+merge_request:
+author:
diff --git a/changelogs/unreleased/implement-i18n-support.yml b/changelogs/unreleased/implement-i18n-support.yml
new file mode 100644
index 0000000000000..d304fbecf9088
--- /dev/null
+++ b/changelogs/unreleased/implement-i18n-support.yml
@@ -0,0 +1,4 @@
+---
+title: Add support for i18n on Cycle Analytics page
+merge_request: 10669
+author:
diff --git a/changelogs/unreleased/issue-boards-no-avatar.yml b/changelogs/unreleased/issue-boards-no-avatar.yml
new file mode 100644
index 0000000000000..a2dd53b3f2fe8
--- /dev/null
+++ b/changelogs/unreleased/issue-boards-no-avatar.yml
@@ -0,0 +1,4 @@
+---
+title: Fixed avatar not display on issue boards when Gravatar is disabled
+merge_request:
+author:
diff --git a/changelogs/unreleased/issue-title-description-realtime.yml b/changelogs/unreleased/issue-title-description-realtime.yml
new file mode 100644
index 0000000000000..003e1a4ab334e
--- /dev/null
+++ b/changelogs/unreleased/issue-title-description-realtime.yml
@@ -0,0 +1,4 @@
+---
+title: Add realtime descriptions to issue show pages
+merge_request:
+author:
diff --git a/changelogs/unreleased/merge-request-poll-json-endpoint.yml b/changelogs/unreleased/merge-request-poll-json-endpoint.yml
new file mode 100644
index 0000000000000..6c41984e9b702
--- /dev/null
+++ b/changelogs/unreleased/merge-request-poll-json-endpoint.yml
@@ -0,0 +1,4 @@
+---
+title: Fixed bug where merge request JSON would be displayed
+merge_request:
+author:
diff --git a/changelogs/unreleased/mrchrisw-import-shell-timeout.yml b/changelogs/unreleased/mrchrisw-import-shell-timeout.yml
new file mode 100644
index 0000000000000..e43409109d6c5
--- /dev/null
+++ b/changelogs/unreleased/mrchrisw-import-shell-timeout.yml
@@ -0,0 +1,4 @@
+---
+title: Add configurable timeout for git fetch and clone operations
+merge_request: 10697
+author: 
diff --git a/changelogs/unreleased/prometheus-integration-test-setting-fix.yml b/changelogs/unreleased/prometheus-integration-test-setting-fix.yml
new file mode 100644
index 0000000000000..45b7c2263e694
--- /dev/null
+++ b/changelogs/unreleased/prometheus-integration-test-setting-fix.yml
@@ -0,0 +1,4 @@
+---
+title: Prevent 500 errors caused by testing the Prometheus service
+merge_request: 10994
+author:
diff --git a/changelogs/unreleased/tags-sort-default.yml b/changelogs/unreleased/tags-sort-default.yml
new file mode 100644
index 0000000000000..265b765d5400c
--- /dev/null
+++ b/changelogs/unreleased/tags-sort-default.yml
@@ -0,0 +1,4 @@
+---
+title: Fixed tags sort from defaulting to empty
+merge_request:
+author:
diff --git a/changelogs/unreleased/update-issue-board-cards-design.yml b/changelogs/unreleased/update-issue-board-cards-design.yml
new file mode 100644
index 0000000000000..5ef94a74e8a92
--- /dev/null
+++ b/changelogs/unreleased/update-issue-board-cards-design.yml
@@ -0,0 +1,4 @@
+---
+title: Update issue board cards design
+merge_request: 10353
+author:
diff --git a/changelogs/unreleased/winh-visual-token-labels.yml b/changelogs/unreleased/winh-visual-token-labels.yml
new file mode 100644
index 0000000000000..d4952e910b4c5
--- /dev/null
+++ b/changelogs/unreleased/winh-visual-token-labels.yml
@@ -0,0 +1,4 @@
+---
+title: Colorize labels in search field
+merge_request: 11047
+author:
diff --git a/config/application.rb b/config/application.rb
index cddfc892cdd93..35e1550ec9838 100644
--- a/config/application.rb
+++ b/config/application.rb
@@ -45,6 +45,9 @@ class Application < Rails::Application
     # config.i18n.default_locale = :de
     config.i18n.enforce_available_locales = false
 
+    # Translation for AR attrs is not working well for POROs like WikiPage
+    config.gettext_i18n_rails.use_for_active_record_attributes = false
+
     # Configure the default encoding used in templates for Ruby 1.9.
     config.encoding = "utf-8"
 
diff --git a/config/gitlab.yml.example b/config/gitlab.yml.example
index f52a10be069a4..d3002a0f82204 100644
--- a/config/gitlab.yml.example
+++ b/config/gitlab.yml.example
@@ -589,6 +589,9 @@ production: &base
     upload_pack: true
     receive_pack: true
 
+    # Git import/fetch timeout
+    # git_timeout: 800
+
     # If you use non-standard ssh port you need to specify it
     # ssh_port: 22
 
diff --git a/config/initializers/1_settings.rb b/config/initializers/1_settings.rb
index df26e44534577..7dfb8613098e2 100644
--- a/config/initializers/1_settings.rb
+++ b/config/initializers/1_settings.rb
@@ -462,6 +462,7 @@ def cron_random_weekly_time
 Settings.gitlab_shell['ssh_user']     ||= Settings.gitlab.user
 Settings.gitlab_shell['owner_group']  ||= Settings.gitlab.user
 Settings.gitlab_shell['ssh_path_prefix'] ||= Settings.__send__(:build_gitlab_shell_ssh_path_prefix)
+Settings.gitlab_shell['git_timeout'] ||= 800
 
 #
 # Workhorse
diff --git a/config/initializers/fast_gettext.rb b/config/initializers/fast_gettext.rb
new file mode 100644
index 0000000000000..a69fe0c902e80
--- /dev/null
+++ b/config/initializers/fast_gettext.rb
@@ -0,0 +1,5 @@
+FastGettext.add_text_domain 'gitlab', path: File.join(Rails.root, 'locale'), type: :po
+FastGettext.default_text_domain = 'gitlab'
+FastGettext.default_available_locales = Gitlab::I18n.available_locales
+
+I18n.available_locales = Gitlab::I18n.available_locales
diff --git a/config/initializers/gettext_rails_i18n_patch.rb b/config/initializers/gettext_rails_i18n_patch.rb
new file mode 100644
index 0000000000000..69118f464caa0
--- /dev/null
+++ b/config/initializers/gettext_rails_i18n_patch.rb
@@ -0,0 +1,42 @@
+require 'gettext_i18n_rails/haml_parser'
+require 'gettext_i18n_rails_js/parser/javascript'
+
+VUE_TRANSLATE_REGEX = /((%[\w.-]+)(?:\s))?{{ (N|n|s)?__\((.*)\) }}/
+
+module GettextI18nRails
+  class HamlParser
+    singleton_class.send(:alias_method, :old_convert_to_code, :convert_to_code)
+
+    # We need to convert text in Mustache format
+    # to a format that can be parsed by Gettext scripts.
+    # If we found a content like "{{ __('Stage') }}"
+    # in a HAML file we convert it to "= _('Stage')", that way
+    # it can be processed by the "rake gettext:find" script.
+    #
+    # Overwrites: https://github.com/grosser/gettext_i18n_rails/blob/8396387a431e0f8ead72fc1cd425cad2fa4992f2/lib/gettext_i18n_rails/haml_parser.rb#L9
+    def self.convert_to_code(text)
+      text.gsub!(VUE_TRANSLATE_REGEX, "\\2= \\3_(\\4)")
+
+      old_convert_to_code(text)
+    end
+  end
+end
+
+module GettextI18nRailsJs
+  module Parser
+    module Javascript
+      # This is required to tell the `rake gettext:find` script to use the Javascript
+      # parser for *.vue files.
+      #
+      # Overwrites: https://github.com/webhippie/gettext_i18n_rails_js/blob/46c58db6d2053a4f5f36a0eb024ea706ff5707cb/lib/gettext_i18n_rails_js/parser/javascript.rb#L36
+      def target?(file)
+        [
+          ".js",
+          ".jsx",
+          ".coffee",
+          ".vue"
+        ].include? ::File.extname(file)
+      end
+    end
+  end
+end
diff --git a/config/locales/de.yml b/config/locales/de.yml
new file mode 100644
index 0000000000000..533663a270495
--- /dev/null
+++ b/config/locales/de.yml
@@ -0,0 +1,219 @@
+---
+de:
+  activerecord:
+    errors:
+      messages:
+        record_invalid: 'Gültigkeitsprüfung ist fehlgeschlagen: %{errors}'
+        restrict_dependent_destroy:
+          has_one: Datensatz kann nicht gelöscht werden, da ein abhängiger %{record}-Datensatz
+            existiert.
+          has_many: Datensatz kann nicht gelöscht werden, da abhängige %{record} existieren.
+  date:
+    abbr_day_names:
+    - So
+    - Mo
+    - Di
+    - Mi
+    - Do
+    - Fr
+    - Sa
+    abbr_month_names:
+    -
+    - Jan
+    - Feb
+    - Mär
+    - Apr
+    - Mai
+    - Jun
+    - Jul
+    - Aug
+    - Sep
+    - Okt
+    - Nov
+    - Dez
+    day_names:
+    - Sonntag
+    - Montag
+    - Dienstag
+    - Mittwoch
+    - Donnerstag
+    - Freitag
+    - Samstag
+    formats:
+      default: "%d.%m.%Y"
+      long: "%e. %B %Y"
+      short: "%e. %b"
+    month_names:
+    -
+    - Januar
+    - Februar
+    - März
+    - April
+    - Mai
+    - Juni
+    - Juli
+    - August
+    - September
+    - Oktober
+    - November
+    - Dezember
+    order:
+    - :day
+    - :month
+    - :year
+  datetime:
+    distance_in_words:
+      about_x_hours:
+        one: etwa eine Stunde
+        other: etwa %{count} Stunden
+      about_x_months:
+        one: etwa ein Monat
+        other: etwa %{count} Monate
+      about_x_years:
+        one: etwa ein Jahr
+        other: etwa %{count} Jahre
+      almost_x_years:
+        one: fast ein Jahr
+        other: fast %{count} Jahre
+      half_a_minute: eine halbe Minute
+      less_than_x_minutes:
+        one: weniger als eine Minute
+        other: weniger als %{count} Minuten
+      less_than_x_seconds:
+        one: weniger als eine Sekunde
+        other: weniger als %{count} Sekunden
+      over_x_years:
+        one: mehr als ein Jahr
+        other: mehr als %{count} Jahre
+      x_days:
+        one: ein Tag
+        other: "%{count} Tage"
+      x_minutes:
+        one: eine Minute
+        other: "%{count} Minuten"
+      x_months:
+        one: ein Monat
+        other: "%{count} Monate"
+      x_seconds:
+        one: eine Sekunde
+        other: "%{count} Sekunden"
+    prompts:
+      day: Tag
+      hour: Stunden
+      minute: Minute
+      month: Monat
+      second: Sekunde
+      year: Jahr
+  errors:
+    format: "%{attribute} %{message}"
+    messages:
+      accepted: muss akzeptiert werden
+      blank: muss ausgefüllt werden
+      present: darf nicht ausgefüllt werden
+      confirmation: stimmt nicht mit %{attribute} überein
+      empty: muss ausgefüllt werden
+      equal_to: muss genau %{count} sein
+      even: muss gerade sein
+      exclusion: ist nicht verfügbar
+      greater_than: muss größer als %{count} sein
+      greater_than_or_equal_to: muss größer oder gleich %{count} sein
+      inclusion: ist kein gültiger Wert
+      invalid: ist nicht gültig
+      less_than: muss kleiner als %{count} sein
+      less_than_or_equal_to: muss kleiner oder gleich %{count} sein
+      model_invalid: 'Gültigkeitsprüfung ist fehlgeschlagen: %{errors}'
+      not_a_number: ist keine Zahl
+      not_an_integer: muss ganzzahlig sein
+      odd: muss ungerade sein
+      required: muss ausgefüllt werden
+      taken: ist bereits vergeben
+      too_long:
+        one: ist zu lang (mehr als 1 Zeichen)
+        other: ist zu lang (mehr als %{count} Zeichen)
+      too_short:
+        one: ist zu kurz (weniger als 1 Zeichen)
+        other: ist zu kurz (weniger als %{count} Zeichen)
+      wrong_length:
+        one: hat die falsche Länge (muss genau 1 Zeichen haben)
+        other: hat die falsche Länge (muss genau %{count} Zeichen haben)
+      other_than: darf nicht gleich %{count} sein
+    template:
+      body: 'Bitte überprüfen Sie die folgenden Felder:'
+      header:
+        one: 'Konnte %{model} nicht speichern: ein Fehler.'
+        other: 'Konnte %{model} nicht speichern: %{count} Fehler.'
+  helpers:
+    select:
+      prompt: Bitte wählen
+    submit:
+      create: "%{model} erstellen"
+      submit: "%{model} speichern"
+      update: "%{model} aktualisieren"
+  number:
+    currency:
+      format:
+        delimiter: "."
+        format: "%n %u"
+        precision: 2
+        separator: ","
+        significant: false
+        strip_insignificant_zeros: false
+        unit: "€"
+    format:
+      delimiter: "."
+      precision: 2
+      separator: ","
+      significant: false
+      strip_insignificant_zeros: false
+    human:
+      decimal_units:
+        format: "%n %u"
+        units:
+          billion:
+            one: Milliarde
+            other: Milliarden
+          million:
+            one: Million
+            other: Millionen
+          quadrillion:
+            one: Billiarde
+            other: Billiarden
+          thousand: Tausend
+          trillion:
+            one: Billion
+            other: Billionen
+          unit: ''
+      format:
+        delimiter: ''
+        precision: 3
+        significant: true
+        strip_insignificant_zeros: true
+      storage_units:
+        format: "%n %u"
+        units:
+          byte:
+            one: Byte
+            other: Bytes
+          gb: GB
+          kb: KB
+          mb: MB
+          tb: TB
+    percentage:
+      format:
+        delimiter: ''
+        format: "%n %"
+    precision:
+      format:
+        delimiter: ''
+  support:
+    array:
+      last_word_connector: " und "
+      two_words_connector: " und "
+      words_connector: ", "
+  time:
+    am: vormittags
+    formats:
+      default: "%A, %d. %B %Y, %H:%M Uhr"
+      long: "%A, %d. %B %Y, %H:%M Uhr"
+      short: "%d. %B, %H:%M Uhr"
+    pm: nachmittags
diff --git a/config/locales/es.yml b/config/locales/es.yml
new file mode 100644
index 0000000000000..87e79beee7460
--- /dev/null
+++ b/config/locales/es.yml
@@ -0,0 +1,217 @@
+---
+es:
+  activerecord:
+    errors:
+      messages:
+        record_invalid: "La validación falló: %{errors}"
+        restrict_dependent_destroy:
+          has_one: No se puede eliminar el registro porque existe un %{record} dependiente
+          has_many: No se puede eliminar el registro porque existen %{record} dependientes
+  date:
+    abbr_day_names:
+    - dom
+    - lun
+    - mar
+    - mié
+    - jue
+    - vie
+    - sáb
+    abbr_month_names:
+    -
+    - ene
+    - feb
+    - mar
+    - abr
+    - may
+    - jun
+    - jul
+    - ago
+    - sep
+    - oct
+    - nov
+    - dic
+    day_names:
+    - domingo
+    - lunes
+    - martes
+    - miércoles
+    - jueves
+    - viernes
+    - sábado
+    formats:
+      default: "%d/%m/%Y"
+      long: "%d de %B de %Y"
+      short: "%d de %b"
+    month_names:
+    -
+    - enero
+    - febrero
+    - marzo
+    - abril
+    - mayo
+    - junio
+    - julio
+    - agosto
+    - septiembre
+    - octubre
+    - noviembre
+    - diciembre
+    order:
+    - :day
+    - :month
+    - :year
+  datetime:
+    distance_in_words:
+      about_x_hours:
+        one: alrededor de 1 hora
+        other: alrededor de %{count} horas
+      about_x_months:
+        one: alrededor de 1 mes
+        other: alrededor de %{count} meses
+      about_x_years:
+        one: alrededor de 1 año
+        other: alrededor de %{count} años
+      almost_x_years:
+        one: casi 1 año
+        other: casi %{count} años
+      half_a_minute: medio minuto
+      less_than_x_minutes:
+        one: menos de 1 minuto
+        other: menos de %{count} minutos
+      less_than_x_seconds:
+        one: menos de 1 segundo
+        other: menos de %{count} segundos
+      over_x_years:
+        one: más de 1 año
+        other: más de %{count} años
+      x_days:
+        one: 1 día
+        other: "%{count} días"
+      x_minutes:
+        one: 1 minuto
+        other: "%{count} minutos"
+      x_months:
+        one: 1 mes
+        other: "%{count} meses"
+      x_years:
+        one: 1 año
+        other: "%{count} años"
+      x_seconds:
+        one: 1 segundo
+        other: "%{count} segundos"
+    prompts:
+      day: Día
+      hour: Hora
+      minute: Minutos
+      month: Mes
+      second: Segundos
+      year: Año
+  errors:
+    format: "%{attribute} %{message}"
+    messages:
+      accepted: debe ser aceptado
+      blank: no puede estar en blanco
+      present: debe estar en blanco
+      confirmation: no coincide
+      empty: no puede estar vacío
+      equal_to: debe ser igual a %{count}
+      even: debe ser par
+      exclusion: está reservado
+      greater_than: debe ser mayor que %{count}
+      greater_than_or_equal_to: debe ser mayor que o igual a %{count}
+      inclusion: no está incluido en la lista
+      invalid: no es válido
+      less_than: debe ser menor que %{count}
+      less_than_or_equal_to: debe ser menor que o igual a %{count}
+      model_invalid: "La validación falló: %{errors}"
+      not_a_number: no es un número
+      not_an_integer: debe ser un entero
+      odd: debe ser impar
+      required: debe existir
+      taken: ya está en uso
+      too_long:
+        one: "es demasiado largo (1 carácter máximo)"
+        other: "es demasiado largo (%{count} caracteres máximo)"
+      too_short:
+        one: "es demasiado corto (1 carácter mínimo)"
+        other: "es demasiado corto (%{count} caracteres mínimo)"
+      wrong_length:
+        one: "no tiene la longitud correcta (1 carácter exactos)"
+        other: "no tiene la longitud correcta (%{count} caracteres exactos)"
+      other_than: debe ser distinto de %{count}
+    template:
+      body: 'Se encontraron problemas con los siguientes campos:'
+      header:
+        one: No se pudo guardar este/a %{model} porque se encontró 1 error
+        other: No se pudo guardar este/a %{model} porque se encontraron %{count} errores
+  helpers:
+    select:
+      prompt: Por favor seleccione
+    submit:
+      create: Crear %{model}
+      submit: Guardar %{model}
+      update: Actualizar %{model}
+  number:
+    currency:
+      format:
+        delimiter: "."
+        format: "%n %u"
+        precision: 2
+        separator: ","
+        significant: false
+        strip_insignificant_zeros: false
+        unit: "€"
+    format:
+      delimiter: "."
+      precision: 3
+      separator: ","
+      significant: false
+      strip_insignificant_zeros: false
+    human:
+      decimal_units:
+        format: "%n %u"
+        units:
+          billion: mil millones
+          million:
+            one: millón
+            other: millones
+          quadrillion: mil billones
+          thousand: mil
+          trillion:
+            one: billón
+            other: billones
+          unit: ''
+      format:
+        delimiter: ''
+        precision: 1
+        significant: true
+        strip_insignificant_zeros: true
+      storage_units:
+        format: "%n %u"
+        units:
+          byte:
+            one: Byte
+            other: Bytes
+          gb: GB
+          kb: KB
+          mb: MB
+          tb: TB
+    percentage:
+      format:
+        delimiter: ''
+        format: "%n %"
+    precision:
+      format:
+        delimiter: ''
+  support:
+    array:
+      last_word_connector: " y "
+      two_words_connector: " y "
+      words_connector: ", "
+  time:
+    am: am
+    formats:
+      default: "%A, %d de %B de %Y %H:%M:%S %z"
+      long: "%d de %B de %Y %H:%M"
+      short: "%d de %b %H:%M"
+    pm: pm
diff --git a/config/routes/project.rb b/config/routes/project.rb
index 7cfd64fd8361e..7822c5d2bf3be 100644
--- a/config/routes/project.rb
+++ b/config/routes/project.rb
@@ -157,6 +157,7 @@
           post :cancel
           post :retry
           get :builds
+          get :failures
           get :status
         end
       end
@@ -218,6 +219,7 @@
           get :download
           get :browse, path: 'browse(/*path)', format: false
           get :file, path: 'file/*path', format: false
+          get :raw, path: 'raw/*path', format: false
           post :keep
         end
       end
diff --git a/config/sidekiq_queues.yml b/config/sidekiq_queues.yml
index eccd3999391f8..0d9a9d06aeb79 100644
--- a/config/sidekiq_queues.yml
+++ b/config/sidekiq_queues.yml
@@ -53,6 +53,7 @@
   - [pages, 1]
   - [system_hook_push, 1]
   - [update_user_activity, 1]
+  - [propagate_service_template, 1]
   # EE specific queues
   - [geo, 1]
   - [project_mirror, 1]
@@ -63,4 +64,3 @@
   - [elastic_indexer, 1]
   - [elastic_commit_indexer, 1]
   - [export_csv, 1]
-
diff --git a/config/webpack.config.js b/config/webpack.config.js
index 5bb636e7d99a1..772392833c20c 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -17,6 +17,10 @@ var DEV_SERVER_LIVERELOAD = process.env.DEV_SERVER_LIVERELOAD !== 'false';
 var WEBPACK_REPORT = process.env.WEBPACK_REPORT;
 
 var config = {
+  // because sqljs requires fs.
+  node: {
+    fs: "empty"
+  },
   context: path.join(ROOT_PATH, 'app/assets/javascripts'),
   entry: {
     blob:                 './blob_edit/blob_bundle.js',
@@ -27,6 +31,7 @@ var config = {
     common_d3:            ['d3'],
     cycle_analytics:      './cycle_analytics/cycle_analytics_bundle.js',
     commit_pipelines:     './commit/pipelines/pipelines_bundle.js',
+    deploy_keys:          './deploy_keys/index.js',
     diff_notes:           './diff_notes/diff_notes_bundle.js',
     environments:         './environments/environments_bundle.js',
     environments_folder:  './environments/folder/environments_folder_bundle.js',
@@ -35,8 +40,8 @@ var config = {
     group:                './group.js',
     groups_list:          './groups_list.js',
     issues:               './issues/issues_bundle.js',
-    issuable:             './issuable/issuable_bundle.js',
     issue_show:           './issue_show/index.js',
+    locale:               './locale/index.js',
     main:                 './main.js',
     merge_conflicts:      './merge_conflicts/merge_conflicts_bundle.js',
     merge_request_widget: './merge_request_widget/ci_bundle.js',
@@ -46,15 +51,18 @@ var config = {
     notebook_viewer:      './blob/notebook_viewer.js',
     pdf_viewer:           './blob/pdf_viewer.js',
     pipelines:            './pipelines/index.js',
+    balsamiq_viewer:      './blob/balsamiq_viewer.js',
     profile:              './profile/profile_bundle.js',
     protected_branches:   './protected_branches/protected_branches_bundle.js',
     protected_tags:       './protected_tags',
+    sidebar:              './sidebar/sidebar_bundle.js',
     snippet:              './snippet/snippet_bundle.js',
     sketch_viewer:        './blob/sketch_viewer.js',
     stl_viewer:           './blob/stl_viewer.js',
     terminal:             './terminal/terminal_bundle.js',
     u2f:                  ['vendor/u2f'],
     users:                './users/users_bundle.js',
+    raven:                './raven/index.js',
   },
 
   output: {
@@ -90,6 +98,10 @@ var config = {
         exclude: /node_modules/,
         loader: 'file-loader',
       },
+      {
+        test: /locale\/[a-z]+\/(.*)\.js$/,
+        loader: 'exports-loader?locales',
+      },
     ]
   },
 
@@ -125,10 +137,11 @@ var config = {
         'boards',
         'commit_pipelines',
         'cycle_analytics',
+        'deploy_keys',
         'diff_notes',
         'environments',
         'environments_folder',
-        'issuable',
+        'sidebar',
         'issue_show',
         'merge_conflicts',
         'notebook_viewer',
@@ -157,6 +170,14 @@ var config = {
     new webpack.optimize.CommonsChunkPlugin({
       names: ['main', 'common', 'runtime'],
     }),
+
+    // locale common library
+    new webpack.optimize.CommonsChunkPlugin({
+      name: 'locale',
+      chunks: [
+        'cycle_analytics',
+      ],
+    }),
   ],
 
   resolve: {
diff --git a/db/fixtures/development/09_issues.rb b/db/fixtures/development/09_issues.rb
index d93d133d15767..0b32a461d56b1 100644
--- a/db/fixtures/development/09_issues.rb
+++ b/db/fixtures/development/09_issues.rb
@@ -8,7 +8,7 @@
         description: FFaker::Lorem.sentence,
         state: ['opened', 'closed'].sample,
         milestone: project.milestones.sample,
-        assignee: project.team.users.sample
+        assignees: [project.team.users.sample]
       }
 
       Issues::CreateService.new(project, project.team.users.sample, issue_params).execute
diff --git a/db/fixtures/development/20_burndown.rb b/db/fixtures/development/20_burndown.rb
index f6a92d5275a55..c31327fc85729 100644
--- a/db/fixtures/development/20_burndown.rb
+++ b/db/fixtures/development/20_burndown.rb
@@ -49,7 +49,7 @@ def create_issues
         description: FFaker::Lorem.sentence,
         state: 'opened',
         milestone: @milestone,
-        assignee: @project.team.users.sample,
+        assignees: [@project.team.users.sample],
         weight: rand(1..9)
       }
 
diff --git a/db/migrate/20170320171632_create_issue_assignees_table.rb b/db/migrate/20170320171632_create_issue_assignees_table.rb
new file mode 100644
index 0000000000000..23b8da37b6d75
--- /dev/null
+++ b/db/migrate/20170320171632_create_issue_assignees_table.rb
@@ -0,0 +1,40 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class CreateIssueAssigneesTable < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  INDEX_NAME = 'index_issue_assignees_on_issue_id_and_user_id'
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using the methods "add_concurrent_index" or "add_column_with_default"
+  # you must disable the use of transactions as these methods can not run in an
+  # existing transaction. When using "add_concurrent_index" make sure that this
+  # method is the _only_ method called in the migration, any other changes
+  # should go in a separate migration. This ensures that upon failure _only_ the
+  # index creation fails and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  # disable_ddl_transaction!
+
+  def up
+    create_table :issue_assignees do |t|
+      t.references :user, foreign_key: { on_delete: :cascade }, index: true, null: false
+      t.references :issue, foreign_key: { on_delete: :cascade }, null: false
+    end
+
+    add_index :issue_assignees, [:issue_id, :user_id], unique: true, name: INDEX_NAME
+  end
+
+  def down
+    drop_table :issue_assignees
+  end
+end
diff --git a/db/migrate/20170320173259_migrate_assignees.rb b/db/migrate/20170320173259_migrate_assignees.rb
new file mode 100644
index 0000000000000..ba8edbd7d32e2
--- /dev/null
+++ b/db/migrate/20170320173259_migrate_assignees.rb
@@ -0,0 +1,52 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class MigrateAssignees < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using the methods "add_concurrent_index" or "add_column_with_default"
+  # you must disable the use of transactions as these methods can not run in an
+  # existing transaction. When using "add_concurrent_index" make sure that this
+  # method is the _only_ method called in the migration, any other changes
+  # should go in a separate migration. This ensures that upon failure _only_ the
+  # index creation fails and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  disable_ddl_transaction!
+
+  def up
+    # Optimisation: this accounts for most of the invalid assignee IDs on GitLab.com
+    update_column_in_batches(:issues, :assignee_id, nil) do |table, query|
+      query.where(table[:assignee_id].eq(0))
+    end
+
+    users = Arel::Table.new(:users)
+
+    update_column_in_batches(:issues, :assignee_id, nil) do |table, query|
+      query.where(table[:assignee_id].not_eq(nil)\
+        .and(
+          users.project("true").where(users[:id].eq(table[:assignee_id])).exists.not
+        ))
+    end
+
+    execute <<-EOF
+      INSERT INTO issue_assignees(issue_id, user_id)
+      SELECT id, assignee_id FROM issues WHERE assignee_id IS NOT NULL
+    EOF
+  end
+
+  def down
+    execute <<-EOF
+      DELETE FROM issue_assignees
+    EOF
+  end
+end
diff --git a/db/migrate/20170413035209_add_preferred_language_to_users.rb b/db/migrate/20170413035209_add_preferred_language_to_users.rb
new file mode 100644
index 0000000000000..92f1d6f243633
--- /dev/null
+++ b/db/migrate/20170413035209_add_preferred_language_to_users.rb
@@ -0,0 +1,16 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddPreferredLanguageToUsers < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  DOWNTIME = false
+
+  def up
+    add_column :users, :preferred_language, :string
+  end
+
+  def down
+    remove_column :users, :preferred_language
+  end
+end
diff --git a/db/migrate/20170503021915_add_last_edited_at_and_last_edited_by_id_to_issues.rb b/db/migrate/20170503021915_add_last_edited_at_and_last_edited_by_id_to_issues.rb
new file mode 100644
index 0000000000000..6ac10723c82f9
--- /dev/null
+++ b/db/migrate/20170503021915_add_last_edited_at_and_last_edited_by_id_to_issues.rb
@@ -0,0 +1,14 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddLastEditedAtAndLastEditedByIdToIssues < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  def change
+    add_column :issues, :last_edited_at, :timestamp
+    add_column :issues, :last_edited_by_id, :integer
+  end
+end
diff --git a/db/migrate/20170503022548_add_last_edited_at_and_last_edited_by_id_to_merge_requests.rb b/db/migrate/20170503022548_add_last_edited_at_and_last_edited_by_id_to_merge_requests.rb
new file mode 100644
index 0000000000000..7a1acdcbf69ef
--- /dev/null
+++ b/db/migrate/20170503022548_add_last_edited_at_and_last_edited_by_id_to_merge_requests.rb
@@ -0,0 +1,14 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddLastEditedAtAndLastEditedByIdToMergeRequests < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  def change
+    add_column :merge_requests, :last_edited_at, :timestamp
+    add_column :merge_requests, :last_edited_by_id, :integer
+  end
+end
diff --git a/db/migrate/20170504102911_add_clientside_sentry_to_application_settings.rb b/db/migrate/20170504102911_add_clientside_sentry_to_application_settings.rb
new file mode 100644
index 0000000000000..141112f8b5084
--- /dev/null
+++ b/db/migrate/20170504102911_add_clientside_sentry_to_application_settings.rb
@@ -0,0 +1,33 @@
+# See http://doc.gitlab.com/ce/development/migration_style_guide.html
+# for more information on how to write migrations for GitLab.
+
+class AddClientsideSentryToApplicationSettings < ActiveRecord::Migration
+  include Gitlab::Database::MigrationHelpers
+
+  # Set this constant to true if this migration requires downtime.
+  DOWNTIME = false
+
+  # When a migration requires downtime you **must** uncomment the following
+  # constant and define a short and easy to understand explanation as to why the
+  # migration requires downtime.
+  # DOWNTIME_REASON = ''
+
+  # When using "add_concurrent_index" or "remove_concurrent_index" methods make sure
+  # that either of them is the _only_ method called in the migration,
+  # any other changes should go in a separate migration.
+  # This ensures that upon failure _only_ the index creation or removing fails
+  # and can be retried or reverted easily.
+  #
+  # To disable transactions uncomment the following line and remove these
+  # comments:
+  disable_ddl_transaction!
+
+  def up
+    add_column_with_default :application_settings, :clientside_sentry_enabled, :boolean, default: false
+    add_column :application_settings, :clientside_sentry_dsn, :string
+  end
+
+  def down
+    remove_columns :application_settings, :clientside_sentry_enabled, :clientside_sentry_dsn
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 731320e4f6005..e6afe36bf02ac 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -11,7 +11,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema.define(version: 20170503185032) do
+ActiveRecord::Schema.define(version: 20170504102911) do
 
   # These are extensions that must be enabled in order to support this database
   enable_extension "plpgsql"
@@ -135,6 +135,8 @@
     t.decimal "polling_interval_multiplier", default: 1.0, null: false
     t.boolean "elasticsearch_experimental_indexer"
     t.integer "cached_markdown_version"
+    t.boolean "clientside_sentry_enabled", default: false, null: false
+    t.string "clientside_sentry_dsn"
   end
 
   create_table "approvals", force: :cascade do |t|
@@ -538,6 +540,14 @@
 
   add_index "index_statuses", ["project_id"], name: "index_index_statuses_on_project_id", unique: true, using: :btree
 
+  create_table "issue_assignees", force: :cascade do |t|
+    t.integer "user_id", null: false
+    t.integer "issue_id", null: false
+  end
+
+  add_index "issue_assignees", ["issue_id", "user_id"], name: "index_issue_assignees_on_issue_id_and_user_id", unique: true, using: :btree
+  add_index "issue_assignees", ["user_id"], name: "index_issue_assignees_on_user_id", using: :btree
+
   create_table "issue_metrics", force: :cascade do |t|
     t.integer "issue_id", null: false
     t.datetime "first_mentioned_in_commit_at"
@@ -576,6 +586,8 @@
     t.datetime "closed_at"
     t.string "service_desk_reply_to"
     t.integer "cached_markdown_version"
+    t.datetime "last_edited_at"
+    t.integer "last_edited_by_id"
   end
 
   add_index "issues", ["assignee_id"], name: "index_issues_on_assignee_id", using: :btree
@@ -782,6 +794,8 @@
     t.integer "time_estimate"
     t.boolean "squash", default: false, null: false
     t.integer "cached_markdown_version"
+    t.datetime "last_edited_at"
+    t.integer "last_edited_by_id"
   end
 
   add_index "merge_requests", ["assignee_id"], name: "index_merge_requests_on_assignee_id", using: :btree
@@ -1538,12 +1552,13 @@
     t.string "organization"
     t.boolean "authorized_projects_populated"
     t.boolean "auditor", default: false, null: false
+    t.boolean "ghost"
     t.boolean "require_two_factor_authentication_from_group", default: false, null: false
     t.integer "two_factor_grace_period", default: 48, null: false
-    t.boolean "ghost"
     t.date "last_activity_on"
     t.boolean "notified_of_own_activity"
     t.boolean "support_bot"
+    t.string "preferred_language"
   end
 
   add_index "users", ["admin"], name: "index_users_on_admin", using: :btree
@@ -1605,6 +1620,8 @@
   add_foreign_key "ci_trigger_schedules", "ci_triggers", column: "trigger_id", name: "fk_90a406cc94", on_delete: :cascade
   add_foreign_key "ci_triggers", "users", column: "owner_id", name: "fk_e8e10d1964", on_delete: :cascade
   add_foreign_key "container_repositories", "projects"
+  add_foreign_key "issue_assignees", "issues", on_delete: :cascade
+  add_foreign_key "issue_assignees", "users", on_delete: :cascade
   add_foreign_key "issue_metrics", "issues", on_delete: :cascade
   add_foreign_key "label_priorities", "labels", on_delete: :cascade
   add_foreign_key "label_priorities", "projects", on_delete: :cascade
diff --git a/doc/api/issues.md b/doc/api/issues.md
index da177ce6b75a3..75b1ab79f36ed 100644
--- a/doc/api/issues.md
+++ b/doc/api/issues.md
@@ -70,6 +70,14 @@ Example response:
          "updated_at" : "2016-01-04T15:31:39.996Z"
       },
       "project_id" : 1,
+      "assignees" : [{
+         "state" : "active",
+         "id" : 1,
+         "name" : "Administrator",
+         "web_url" : "https://gitlab.example.com/root",
+         "avatar_url" : null,
+         "username" : "root"
+      }],
       "assignee" : {
          "state" : "active",
          "id" : 1,
@@ -93,6 +101,8 @@ Example response:
 ]
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## List group issues
 
 Get a list of a group's issues.
@@ -154,6 +164,14 @@ Example response:
       "description" : "Omnis vero earum sunt corporis dolor et placeat.",
       "state" : "closed",
       "iid" : 1,
+      "assignees" : [{
+         "avatar_url" : null,
+         "web_url" : "https://gitlab.example.com/lennie",
+         "state" : "active",
+         "username" : "lennie",
+         "id" : 9,
+         "name" : "Dr. Luella Kovacek"
+      }],
       "assignee" : {
          "avatar_url" : null,
          "web_url" : "https://gitlab.example.com/lennie",
@@ -176,6 +194,8 @@ Example response:
 ]
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## List project issues
 
 Get a list of a project's issues.
@@ -237,6 +257,14 @@ Example response:
       "description" : "Omnis vero earum sunt corporis dolor et placeat.",
       "state" : "closed",
       "iid" : 1,
+      "assignees" : [{
+         "avatar_url" : null,
+         "web_url" : "https://gitlab.example.com/lennie",
+         "state" : "active",
+         "username" : "lennie",
+         "id" : 9,
+         "name" : "Dr. Luella Kovacek"
+      }],
       "assignee" : {
          "avatar_url" : null,
          "web_url" : "https://gitlab.example.com/lennie",
@@ -259,6 +287,8 @@ Example response:
 ]
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Single issue
 
 Get a single project issue.
@@ -303,6 +333,14 @@ Example response:
    "description" : "Omnis vero earum sunt corporis dolor et placeat.",
    "state" : "closed",
    "iid" : 1,
+   "assignees" : [{
+      "avatar_url" : null,
+      "web_url" : "https://gitlab.example.com/lennie",
+      "state" : "active",
+      "username" : "lennie",
+      "id" : 9,
+      "name" : "Dr. Luella Kovacek"
+   }],
    "assignee" : {
       "avatar_url" : null,
       "web_url" : "https://gitlab.example.com/lennie",
@@ -325,6 +363,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## New issue
 
 Creates a new project issue.
@@ -333,20 +373,20 @@ Creates a new project issue.
 POST /projects/:id/issues
 ```
 
-| Attribute                                 | Type    | Required | Description  |
-|-------------------------------------------|---------|----------|--------------|
+| Attribute                                 | Type           | Required | Description  |
+|-------------------------------------------|----------------|----------|--------------|
 | `id`                                      | integer/string | yes      | The ID or [URL-encoded path of the project](README.md#namespaced-path-encoding) owned by the authenticated user |
-| `title`                                   | string  | yes      | The title of an issue |
-| `description`                             | string  | no       | The description of an issue  |
-| `confidential`                            | boolean | no       | Set an issue to be confidential. Default is `false`.  |
-| `assignee_id`                             | integer | no       | The ID of a user to assign issue |
-| `milestone_id`                            | integer | no       | The ID of a milestone to assign issue  |
-| `labels`                                  | string  | no       | Comma-separated label names for an issue  |
-| `created_at`                              | string  | no       | Date time string, ISO 8601 formatted, e.g. `2016-03-11T03:45:40Z` (requires admin or project owner rights) |
-| `due_date`                                | string  | no       | Date time string in the format YEAR-MONTH-DAY, e.g. `2016-03-11` |
-| `merge_request_to_resolve_discussions_of` | integer | no       | The IID of a merge request in which to resolve all issues. This will fill the issue with a default description and mark all discussions as resolved. When passing a description or title, these values will take precedence over the default values.|
-| `discussion_to_resolve`                   | string  | no       | The ID of a discussion to resolve. This will fill in the issue with a default description and mark the discussion as resolved. Use in combination with `merge_request_to_resolve_discussions_of`. |
-| `weight` | integer | no | The weight of the issue in range 0 to 9 |
+| `title`                                   | string         | yes      | The title of an issue |
+| `description`                             | string         | no       | The description of an issue  |
+| `confidential`                            | boolean        | no       | Set an issue to be confidential. Default is `false`.  |
+| `assignee_ids`                            | Array[integer] | no       | The ID of a user to assign issue |
+| `milestone_id`                            | integer        | no       | The ID of a milestone to assign issue  |
+| `labels`                                  | string         | no       | Comma-separated label names for an issue  |
+| `created_at`                              | string         | no       | Date time string, ISO 8601 formatted, e.g. `2016-03-11T03:45:40Z` (requires admin or project owner rights) |
+| `due_date`                                | string         | no       | Date time string in the format YEAR-MONTH-DAY, e.g. `2016-03-11` |
+| `merge_request_to_resolve_discussions_of` | integer        | no       | The IID of a merge request in which to resolve all issues. This will fill the issue with a default description and mark all discussions as resolved. When passing a description or title, these values will take precedence over the default values.|
+| `discussion_to_resolve`                   | string         | no       | The ID of a discussion to resolve. This will fill in the issue with a default description and mark the discussion as resolved. Use in combination with `merge_request_to_resolve_discussions_of`. |
+| `weight` | integer                                         | no | The weight of the issue in range 0 to 9 |
 
 ```bash
 curl --request POST --header "PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK" https://gitlab.example.com/api/v4/projects/4/issues?title=Issues%20with%20auth&labels=bug
@@ -362,6 +402,7 @@ Example response:
    "iid" : 14,
    "title" : "Issues with auth",
    "state" : "opened",
+   "assignees" : [],
    "assignee" : null,
    "labels" : [
       "bug"
@@ -386,6 +427,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Edit issue
 
 Updates an existing project issue. This call is also used to mark an issue as
@@ -402,7 +445,7 @@ PUT /projects/:id/issues/:issue_iid
 | `title`        | string  | no       | The title of an issue                                                                                      |
 | `description`  | string  | no       | The description of an issue                                                                                |
 | `confidential` | boolean | no       | Updates an issue to be confidential                                                                        |
-| `assignee_id`  | integer | no       | The ID of a user to assign the issue to                                                                    |
+| `assignee_ids`  | Array[integer] | no       | The ID of a user to assign the issue to                                                                    |
 | `milestone_id` | integer | no       | The ID of a milestone to assign the issue to                                                               |
 | `labels`       | string  | no       | Comma-separated label names for an issue                                                                   |
 | `state_event`  | string  | no       | The state event of an issue. Set `close` to close the issue and `reopen` to reopen it                      |
@@ -437,6 +480,7 @@ Example response:
       "bug"
    ],
    "id" : 85,
+   "assignees" : [],
    "assignee" : null,
    "milestone" : null,
    "subscribed" : true,
@@ -448,6 +492,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Delete an issue
 
 Only for admins and project owners. Soft deletes the issue in question.
@@ -502,6 +548,14 @@ Example response:
   "updated_at": "2016-04-07T12:20:17.596Z",
   "labels": [],
   "milestone": null,
+  "assignees": [{
+    "name": "Miss Monserrate Beier",
+    "username": "axel.block",
+    "id": 12,
+    "state": "active",
+    "avatar_url": "http://www.gravatar.com/avatar/46f6f7dc858ada7be1853f7fb96e81da?s=80&d=identicon",
+    "web_url": "https://gitlab.example.com/axel.block"
+  }],
   "assignee": {
     "name": "Miss Monserrate Beier",
     "username": "axel.block",
@@ -525,6 +579,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Subscribe to an issue
 
 Subscribes the authenticated user to an issue to receive notifications.
@@ -558,6 +614,14 @@ Example response:
   "updated_at": "2016-04-07T12:20:17.596Z",
   "labels": [],
   "milestone": null,
+  "assignees": [{
+    "name": "Miss Monserrate Beier",
+    "username": "axel.block",
+    "id": 12,
+    "state": "active",
+    "avatar_url": "http://www.gravatar.com/avatar/46f6f7dc858ada7be1853f7fb96e81da?s=80&d=identicon",
+    "web_url": "https://gitlab.example.com/axel.block"
+  }],
   "assignee": {
     "name": "Miss Monserrate Beier",
     "username": "axel.block",
@@ -581,6 +645,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Unsubscribe from an issue
 
 Unsubscribes the authenticated user from the issue to not receive notifications
@@ -662,6 +728,14 @@ Example response:
       "updated_at": "2016-06-17T07:47:33.832Z",
       "due_date": null
     },
+    "assignees": [{
+      "name": "Jarret O'Keefe",
+      "username": "francisca",
+      "id": 14,
+      "state": "active",
+      "avatar_url": "http://www.gravatar.com/avatar/a7fa515d53450023c83d62986d0658a8?s=80&d=identicon",
+      "web_url": "https://gitlab.example.com/francisca"
+    }],
     "assignee": {
       "name": "Jarret O'Keefe",
       "username": "francisca",
@@ -694,6 +768,8 @@ Example response:
 }
 ```
 
+**Note**: `assignee` column is deprecated, it shows the first assignee only.
+
 ## Set a time estimate for an issue
 
 Sets an estimated time of work for this issue.
diff --git a/doc/development/README.md b/doc/development/README.md
index c86ce12403e89..bffd3fafdc9ca 100644
--- a/doc/development/README.md
+++ b/doc/development/README.md
@@ -42,6 +42,7 @@
 - [Shell commands](shell_commands.md) in the GitLab codebase
 - [Sidekiq debugging](sidekiq_debugging.md)
 - [Object state models](object_state_models.md)
+- [Building a package for testing purposes](build_test_package.md)
 
 ## Databases
 
diff --git a/doc/development/build_test_package.md b/doc/development/build_test_package.md
new file mode 100644
index 0000000000000..2bc1a7008444d
--- /dev/null
+++ b/doc/development/build_test_package.md
@@ -0,0 +1,35 @@
+# Building a package for testing
+
+While developing a new feature or modifying an existing one, it is helpful if an
+installable package (or a docker image) containing those changes is available
+for testing. For this very purpose, a manual job is provided in the GitLab CI/CD
+pipeline that can be used to trigger a pipeline in the omnibus-gitlab repository
+that will create
+1. A deb package for Ubuntu 16.04, available as a build artifact, and
+2. A docker image, which is pushed to [Omnibus GitLab's container 
+registry](https://gitlab.com/gitlab-org/omnibus-gitlab/container_registry)
+(images titled `gitlab-ce` and `gitlab-ee` respectively and image tag is the
+commit which triggered the pipeline).
+
+When you push a commit to either the gitlab-ce or gitlab-ee project, the
+pipeline for that commit will have a `build-package` manual action you can
+trigger.
+
+## Specifying versions of components
+
+If you want to create a package from a specific branch, commit or tag of any of
+the GitLab components (like GitLab Workhorse, Gitaly, GitLab Pages, etc.), you
+can specify the branch name, commit sha or tag in the component's respective
+`*_VERSION` file. For example, if you want to build a package that uses the
+branch `0-1-stable`, modify the content of `GITALY_SERVER_VERSION` to
+`0-1-stable` and push the commit. This will create a manual job that can be
+used to trigger the build.
+
+## Specifying the branch in omnibus-gitlab repository
+
+In scenarios where a configuration change is to be introduced and omnibus-gitlab
+repository already has the necessary changes in a specific branch, you can build
+a package against that branch through an environment variable named
+`OMNIBUS_BRANCH`. To do this, specify that environment variable with the name of
+the branch as value in `.gitlab-ci.yml` and push a commit. This will create a
+manual job that can be used to trigger the build.
diff --git a/doc/development/code_review.md b/doc/development/code_review.md
index 819578404b6fb..be3dd1e2cc69b 100644
--- a/doc/development/code_review.md
+++ b/doc/development/code_review.md
@@ -1,5 +1,25 @@
 # Code Review Guidelines
 
+## Getting your merge request reviewed, approved, and merged
+
+There are a few rules to get your merge request accepted:
+
+1. Your merge request should only be **merged by a [maintainer][team]**.
+  1. If your merge request includes only backend changes [^1], it must be
+    **approved by a [backend maintainer][team]**.
+  1. If your merge request includes only frontend changes [^1], it must be
+    **approved by a [frontend maintainer][team]**.
+  1. If your merge request includes frontend and backend changes [^1], it must
+    be **approved by a [frontend and a backend maintainer][team]**.
+1. To lower the amount of merge requests maintainers need to review, you can
+  ask or assign any [reviewers][team] for a first review.
+  1. If you need some guidance (e.g. it's your first merge request), feel free
+    to ask one of the [Merge request coaches][team].
+  1. The reviewer will assign the merge request to a maintainer once the
+    reviewer is satisfied with the state of the merge request.
+
+## Best practices
+
 This guide contains advice and best practices for performing code review, and
 having your code reviewed.
 
@@ -12,7 +32,7 @@ of colleagues and contributors. However, the final decision to accept a merge
 request is up to one the project's maintainers, denoted on the
 [team page](https://about.gitlab.com/team).
 
-## Everyone
+### Everyone
 
 - Accept that many programming decisions are opinions. Discuss tradeoffs, which
   you prefer, and reach a resolution quickly.
@@ -31,8 +51,11 @@ request is up to one the project's maintainers, denoted on the
 - Consider one-on-one chats or video calls if there are too many "I didn't
   understand" or "Alternative solution:" comments. Post a follow-up comment
   summarizing one-on-one discussion.
+- If you ask a question to a specific person, always start the comment by
+  mentioning them; this will ensure they see it if their notification level is
+  set to "mentioned" and other people will understand they don't have to respond.
 
-## Having your code reviewed
+### Having your code reviewed
 
 Please keep in mind that code review is a process that can take multiple
 iterations, and reviewers may spot things later that they may not have seen the
@@ -50,11 +73,12 @@ first time.
 - Extract unrelated changes and refactorings into future merge requests/issues.
 - Seek to understand the reviewer's perspective.
 - Try to respond to every comment.
+- Let the reviewer select the "Resolve discussion" buttons.
 - Push commits based on earlier rounds of feedback as isolated commits to the
   branch. Do not squash until the branch is ready to merge. Reviewers should be
   able to read individual updates based on their earlier feedback.
 
-## Reviewing code
+### Reviewing code
 
 Understand why the change is necessary (fixes a bug, improves the user
 experience, refactors the existing code). Then:
@@ -69,12 +93,19 @@ experience, refactors the existing code). Then:
   someone else would be confused by it as well.
 - After a round of line notes, it can be helpful to post a summary note such as
   "LGTM :thumbsup:", or "Just a couple things to address."
+- Assign the merge request to the author if changes are required following your
+  review.
+- Set the milestone before merging a merge request.
 - Avoid accepting a merge request before the job succeeds. Of course, "Merge
   When Pipeline Succeeds" (MWPS) is fine.
 - If you set the MR to "Merge When Pipeline Succeeds", you should take over
   subsequent revisions for anything that would be spotted after that.
+- Consider using the [Squash and
+  merge][squash-and-merge] feature when the merge request has a lot of commits.
+
+[squash-and-merge]: https://docs.gitlab.com/ee/user/project/merge_requests/squash_and_merge.html#squash-and-merge
 
-## The right balance
+### The right balance
 
 One of the most difficult things during code review is finding the right
 balance in how deep the reviewer can interfere with the code created by a
@@ -100,7 +131,7 @@ reviewee.
   tomorrow. When you are not able to find the right balance, ask other people
   about their opinion.
 
-## Credits
+### Credits
 
 Largely based on the [thoughtbot code review guide].
 
diff --git a/doc/development/fe_guide/droplab/droplab.md b/doc/development/fe_guide/droplab/droplab.md
index 8f0b6b21953a0..112ff3419d975 100644
--- a/doc/development/fe_guide/droplab/droplab.md
+++ b/doc/development/fe_guide/droplab/droplab.md
@@ -183,6 +183,8 @@ For example,
 either by a mouse click or by enter key selection.
 * The `droplab-item-active` css class is added to items that have been selected
 using arrow key navigation.
+* You can add the `droplab-item-ignore` css class to any item that you do not want to be selectable. For example,
+an `<li class="divider"></li>` list divider element that should not be interactive.
 
 ## Internal events
 
diff --git a/doc/development/fe_guide/style_guide_js.md b/doc/development/fe_guide/style_guide_js.md
index 1d2b055894830..d2d895172410f 100644
--- a/doc/development/fe_guide/style_guide_js.md
+++ b/doc/development/fe_guide/style_guide_js.md
@@ -11,207 +11,205 @@ See [our current .eslintrc][eslintrc] for specific rules and patterns.
 
 #### ESlint
 
-- **Never** disable eslint rules unless you have a good reason.  You may see a lot of legacy files with `/* eslint-disable some-rule, some-other-rule */` at the top, but legacy files are a special case.  Any time you develop a new feature or refactor an existing one, you should abide by the eslint rules.
-
-- **Never Ever EVER** disable eslint globally for a file
+1. **Never** disable eslint rules unless you have a good reason.  
+You may see a lot of legacy files with `/* eslint-disable some-rule, some-other-rule */`
+at the top, but legacy files are a special case.  Any time you develop a new feature or
+refactor an existing one, you should abide by the eslint rules.
 
+1. **Never Ever EVER** disable eslint globally for a file
   ```javascript
-  // bad
-  /* eslint-disable */
+    // bad
+    /* eslint-disable */
 
-  // better
-  /* eslint-disable some-rule, some-other-rule */
+    // better
+    /* eslint-disable some-rule, some-other-rule */
 
-  // best
-  // nothing :)
+    // best
+    // nothing :)
   ```
 
-- If you do need to disable a rule for a single violation, try to do it as locally as possible
-
+1. If you do need to disable a rule for a single violation, try to do it as locally as possible
   ```javascript
-  // bad
-  /* eslint-disable no-new */
+    // bad
+    /* eslint-disable no-new */
 
-  import Foo from 'foo';
+    import Foo from 'foo';
 
-  new Foo();
+    new Foo();
 
-  // better
-  import Foo from 'foo';
+    // better
+    import Foo from 'foo';
 
-  // eslint-disable-next-line no-new
-  new Foo();
+    // eslint-disable-next-line no-new
+    new Foo();
   ```
+1. There are few rules that we need to disable due to technical debt. Which are:
+  1. [no-new][eslint-new]
+  1. [class-methods-use-this][eslint-this]
 
-- When they are needed _always_ place ESlint directive comment blocks on the first line of a script, followed by any global declarations, then a blank newline prior to any imports or code.
-
+1. When they are needed _always_ place ESlint directive comment blocks on the first line of a script,
+followed by any global declarations, then a blank newline prior to any imports or code.
   ```javascript
-  // bad
-  /* global Foo */
-  /* eslint-disable no-new */
-  import Bar from './bar';
+    // bad
+    /* global Foo */
+    /* eslint-disable no-new */
+    import Bar from './bar';
 
-  // good
-  /* eslint-disable no-new */
-  /* global Foo */
+    // good
+    /* eslint-disable no-new */
+    /* global Foo */
 
-  import Bar from './bar';
+    import Bar from './bar';
   ```
 
-- **Never** disable the `no-undef` rule. Declare globals with `/* global Foo */` instead.
-
-- When declaring multiple globals, always use one `/* global [name] */` line per variable.
+1. **Never** disable the `no-undef` rule. Declare globals with `/* global Foo */` instead.
 
+1. When declaring multiple globals, always use one `/* global [name] */` line per variable.
   ```javascript
-  // bad
-  /* globals Flash, Cookies, jQuery */
+    // bad
+    /* globals Flash, Cookies, jQuery */
 
-  // good
-  /* global Flash */
-  /* global Cookies */
-  /* global jQuery */
+    // good
+    /* global Flash */
+    /* global Cookies */
+    /* global jQuery */
   ```
-  
-- Use up to 3 parameters for a function or class. If you need more accept an Object instead.
 
+1. Use up to 3 parameters for a function or class. If you need more accept an Object instead.
   ```javascript
-  // bad
-  fn(p1, p2, p3, p4) {}
+    // bad
+    fn(p1, p2, p3, p4) {}
 
-  // good
-  fn(options) {}
+    // good
+    fn(options) {}
   ```
 
 #### Modules, Imports, and Exports
-- Use ES module syntax to import modules
-
+1. Use ES module syntax to import modules
   ```javascript
-  // bad
-  require('foo');
+    // bad
+    require('foo');
 
-  // good
-  import Foo from 'foo';
+    // good
+    import Foo from 'foo';
 
-  // bad
-  module.exports = Foo;
+    // bad
+    module.exports = Foo;
 
-  // good
-  export default Foo;
+    // good
+    export default Foo;
   ```
 
-- Relative paths
-
-  Unless you are writing a test, always reference other scripts using relative paths instead of `~`
+1. Relative paths: Unless you are writing a test, always reference other scripts using
+relative paths instead of `~`
+  * In **app/assets/javascripts**:
 
-  In **app/assets/javascripts**:
-  ```javascript
-  // bad
-  import Foo from '~/foo'
-
-  // good
-  import Foo from '../foo';
-  ```
+    ```javascript
+      // bad
+      import Foo from '~/foo'
 
-  In **spec/javascripts**:
-  ```javascript
-  // bad
-  import Foo from '../../app/assets/javascripts/foo'
+      // good
+      import Foo from '../foo';
+    ```
+  * In **spec/javascripts**:
 
-  // good
-  import Foo from '~/foo';
-  ```
+    ```javascript
+      // bad
+      import Foo from '../../app/assets/javascripts/foo'
 
-- Avoid using IIFE. Although we have a lot of examples of files which wrap their contents in IIFEs (immediately-invoked function expressions), this is no longer necessary after the transition from Sprockets to webpack. Do not use them anymore and feel free to remove them when refactoring legacy code.
+      // good
+      import Foo from '~/foo';
+    ```
 
-- Avoid adding to the global namespace.
+1. Avoid using IIFE. Although we have a lot of examples of files which wrap their
+contents in IIFEs (immediately-invoked function expressions),
+this is no longer necessary after the transition from Sprockets to webpack.
+Do not use them anymore and feel free to remove them when refactoring legacy code.
 
+1. Avoid adding to the global namespace.
   ```javascript
-  // bad
-  window.MyClass = class { /* ... */ };
+    // bad
+    window.MyClass = class { /* ... */ };
 
-  // good
-  export default class MyClass { /* ... */ }
+    // good
+    export default class MyClass { /* ... */ }
   ```
 
-- Side effects are forbidden in any script which contains exports
-
+1. Side effects are forbidden in any script which contains exports
   ```javascript
-  // bad
-  export default class MyClass { /* ... */ }
+    // bad
+    export default class MyClass { /* ... */ }
 
-  document.addEventListener("DOMContentLoaded", function(event) {
-    new MyClass();
-  }
+    document.addEventListener("DOMContentLoaded", function(event) {
+      new MyClass();
+    }
   ```
 
 
 #### Data Mutation and Pure functions
-- Strive to write many small pure functions, and minimize where mutations occur.
-
+1. Strive to write many small pure functions, and minimize where mutations occur.
   ```javascript
-  // bad
-  const values = {foo: 1};
+    // bad
+    const values = {foo: 1};
 
-  function impureFunction(items) {
-    const bar = 1;
+    function impureFunction(items) {
+      const bar = 1;
 
-    items.foo = items.a * bar + 2;
+      items.foo = items.a * bar + 2;
 
-    return items.a;
-  }
+      return items.a;
+    }
 
-  const c = impureFunction(values);
+    const c = impureFunction(values);
 
-  // good
-  var values = {foo: 1};
+    // good
+    var values = {foo: 1};
 
-  function pureFunction (foo) {
-    var bar = 1;
+    function pureFunction (foo) {
+      var bar = 1;
 
-    foo = foo * bar + 2;
+      foo = foo * bar + 2;
 
-    return foo;
-  }
+      return foo;
+    }
 
-  var c = pureFunction(values.foo);
+    var c = pureFunction(values.foo);
   ```
 
-- Avoid constructors with side-effects
+1. Avoid constructors with side-effects
 
-- Prefer `.map`, `.reduce` or `.filter` over `.forEach`
+1. Prefer `.map`, `.reduce` or `.filter` over `.forEach`
 A forEach will cause side effects, it will be mutating the array being iterated. Prefer using `.map`,
 `.reduce` or `.filter`
-
   ```javascript
-  const users = [ { name: 'Foo' }, { name: 'Bar' } ];
+    const users = [ { name: 'Foo' }, { name: 'Bar' } ];
 
-  // bad
-  users.forEach((user, index) => {
-    user.id = index;
-  });
+    // bad
+    users.forEach((user, index) => {
+      user.id = index;
+    });
 
-  // good
-  const usersWithId = users.map((user, index) => {
-    return Object.assign({}, user, { id: index });
-  });
+    // good
+    const usersWithId = users.map((user, index) => {
+      return Object.assign({}, user, { id: index });
+    });
   ```
 
 #### Parse Strings into Numbers
-- `parseInt()` is preferable over `Number()` or `+`
-
+1. `parseInt()` is preferable over `Number()` or `+`
   ```javascript
-  // bad
-  +'10' // 10
+    // bad
+    +'10' // 10
 
-  // good
-  Number('10') // 10
+    // good
+    Number('10') // 10
 
-  // better
-  parseInt('10', 10);
+    // better
+    parseInt('10', 10);
   ```
 
 #### CSS classes used for JavaScript
-- If the class is being used in Javascript it needs to be prepend with `js-`
+1. If the class is being used in Javascript it needs to be prepend with `js-`
   ```html
     // bad
     <button class="add-user">
@@ -226,234 +224,270 @@ A forEach will cause side effects, it will be mutating the array being iterated.
 
 ### Vue.js
 
-
 #### Basic Rules
-- Only include one Vue.js component per file.
-- Export components as plain objects:
-
+1. The service has it's own file
+1. The store has it's own file
+1. Use a function in the bundle file to instantiate the Vue component:
   ```javascript
-  export default {
-    template: `<h1>I'm a component</h1>
-  }
-  ```
+    // bad
+    class {
+      init() {
+        new Component({})
+      }
+    }
 
-#### Naming
-- **Extensions**: Use `.vue` extension for Vue components.
-- **Reference Naming**: Use PascalCase for Vue components and camelCase for their instances:
+    // good
+    document.addEventListener('DOMContentLoaded', () => new Vue({
+      el: '#element',
+      components: {
+        componentName
+      },
+      render: createElement => createElement('component-name'),
+    }));
+  ```
 
+1. Don not use a singleton for the service or the store
   ```javascript
-  // bad
-  import cardBoard from 'cardBoard';
+    // bad
+    class Store {
+      constructor() {
+        if (!this.prototype.singleton) {
+          // do something
+        }
+      }
+    }
 
-  // good
-  import CardBoard from 'cardBoard'
+    // good
+    class Store {
+      constructor() {
+        // do something
+      }
+    }
+  ```
 
-  // bad
-  components: {
-    CardBoard: CardBoard
-  };
+#### Naming
+1. **Extensions**: Use `.vue` extension for Vue components.
+1. **Reference Naming**: Use camelCase for their instances:
+  ```javascript
+    // good
+    import cardBoard from 'cardBoard'
 
-  // good
-  components: {
-    cardBoard: CardBoard
-  };
+    components: {
+      cardBoard:
+    };
   ```
 
-- **Props Naming:**
-- Avoid using DOM component prop names.
-- Use kebab-case instead of camelCase to provide props in templates.
-
+1. **Props Naming:**  Avoid using DOM component prop names.
+1. **Props Naming:** Use kebab-case instead of camelCase to provide props in templates.
   ```javascript
-  // bad
-  <component class="btn">
+    // bad
+    <component class="btn">
 
-  // good
-  <component css-class="btn">
+    // good
+    <component css-class="btn">
 
-  // bad
-  <component myProp="prop" />
+    // bad
+    <component myProp="prop" />
 
-  // good
-  <component my-prop="prop" />
-```
+    // good
+    <component my-prop="prop" />
+  ```
 
 #### Alignment
-- Follow these alignment styles for the template method:
-
+1. Follow these alignment styles for the template method:
   ```javascript
-  // bad
-  <component v-if="bar"
-      param="baz" />
+    // bad
+    <component v-if="bar"
+        param="baz" />
 
-  <button class="btn">Click me</button>
+    <button class="btn">Click me</button>
 
-  // good
-  <component
-    v-if="bar"
-    param="baz"
-  />
+    // good
+    <component
+      v-if="bar"
+      param="baz"
+    />
 
-  <button class="btn">
-    Click me
-  </button>
+    <button class="btn">
+      Click me
+    </button>
 
-  // if props fit in one line then keep it on the same line
-  <component bar="bar" />
+    // if props fit in one line then keep it on the same line
+    <component bar="bar" />
   ```
 
 #### Quotes
-- Always use double quotes `"` inside templates and single quotes `'` for all other JS.
-
+1. Always use double quotes `"` inside templates and single quotes `'` for all other JS.
   ```javascript
-  // bad
-  template: `
-    <button :class='style'>Button</button>
-  `
-
-  // good
-  template: `
-    <button :class="style">Button</button>
-  `
+    // bad
+    template: `
+      <button :class='style'>Button</button>
+    `
+
+    // good
+    template: `
+      <button :class="style">Button</button>
+    `
   ```
 
 #### Props
-- Props should be declared as an object
-
+1. Props should be declared as an object
   ```javascript
-  // bad
-  props: ['foo']
-
-  // good
-  props: {
-    foo: {
-      type: String,
-      required: false,
-      default: 'bar'
+    // bad
+    props: ['foo']
+
+    // good
+    props: {
+      foo: {
+        type: String,
+        required: false,
+        default: 'bar'
+      }
     }
-  }
   ```
 
-- Required key should always be provided when declaring a prop
-
+1. Required key should always be provided when declaring a prop
   ```javascript
-  // bad
-  props: {
-    foo: {
-      type: String,
+    // bad
+    props: {
+      foo: {
+        type: String,
+      }
     }
-  }
-
-  // good
-  props: {
-    foo: {
-      type: String,
-      required: false,
-      default: 'bar'
+
+    // good
+    props: {
+      foo: {
+        type: String,
+        required: false,
+        default: 'bar'
+      }
     }
-  }
   ```
 
-- Default key should always be provided if the prop is not required:
-
+1. Default key should always be provided if the prop is not required:
   ```javascript
-  // bad
-  props: {
-    foo: {
-      type: String,
-      required: false,
+    // bad
+    props: {
+      foo: {
+        type: String,
+        required: false,
+      }
     }
-  }
-
-  // good
-  props: {
-    foo: {
-      type: String,
-      required: false,
-      default: 'bar'
+
+    // good
+    props: {
+      foo: {
+        type: String,
+        required: false,
+        default: 'bar'
+      }
     }
-  }
 
-  // good
-  props: {
-    foo: {
-      type: String,
-      required: true
+    // good
+    props: {
+      foo: {
+        type: String,
+        required: true
+      }
     }
-  }
   ```
 
 #### Data
-- `data` method should always be a function
+1. `data` method should always be a function
 
   ```javascript
-  // bad
-  data: {
-    foo: 'foo'
-  }
-
-  // good
-  data() {
-    return {
+    // bad
+    data: {
       foo: 'foo'
-    };
-  }
+    }
+
+    // good
+    data() {
+      return {
+        foo: 'foo'
+      };
+    }
   ```
 
 #### Directives
 
-- Shorthand `@` is preferable over `v-on`
-
+1. Shorthand `@` is preferable over `v-on`
   ```javascript
-  // bad
-  <component v-on:click="eventHandler"/>
+    // bad
+    <component v-on:click="eventHandler"/>
 
 
-  // good
-  <component @click="eventHandler"/>
+    // good
+    <component @click="eventHandler"/>
   ```
 
-- Shorthand `:` is preferable over `v-bind`
-
+1. Shorthand `:` is preferable over `v-bind`
   ```javascript
-  // bad
-  <component v-bind:class="btn"/>
+    // bad
+    <component v-bind:class="btn"/>
 
 
-  // good
-  <component :class="btn"/>
+    // good
+    <component :class="btn"/>
   ```
 
 #### Closing tags
-- Prefer self closing component tags
-
+1. Prefer self closing component tags
   ```javascript
-  // bad
-  <component></component>
+    // bad
+    <component></component>
 
-  // good
-  <component />
+    // good
+    <component />
   ```
 
 #### Ordering
-- Order for a Vue Component:
+1. Order for a Vue Component:
   1. `name`
-  2. `props`
-  3. `data`
-  4. `components`
-  5. `computedProps`
-  6. `methods`
-  7. lifecycle methods
-    1. `beforeCreate`
-    2. `created`
-    3. `beforeMount`
-    4. `mounted`
-    5. `beforeUpdate`
-    6. `updated`
-    7. `activated`
-    8. `deactivated`
-    9. `beforeDestroy`
-    10. `destroyed`
-  8. `template`
+  1. `props`
+  1. `mixins`
+  1. `data`
+  1. `components`
+  1. `computedProps`
+  1. `methods`
+  1. `beforeCreate`
+  1. `created`
+  1. `beforeMount`
+  1. `mounted`
+  1. `beforeUpdate`
+  1. `updated`
+  1. `activated`
+  1. `deactivated`
+  1. `beforeDestroy`
+  1. `destroyed`
+
+#### Vue and Boostrap
+1. Tooltips: Do not rely on `has-tooltip` class name for vue components
+  ```javascript
+    // bad
+    <span class="has-tooltip">
+      Text
+    </span>
+
+    // good
+    <span data-toggle="tooltip">
+      Text
+    </span>
+  ```
+
+1. Tooltips: When using a tooltip, include the tooltip mixin
+
+1. Don't change `data-original-title`.
+  ```javascript
+    // bad
+    <span data-original-title="tooltip text">Foo</span>
+
+    // good
+    <span title="tooltip text">Foo</span>
+
+    $('span').tooltip('fixTitle');
+  ```
 
 
 ## SCSS
@@ -461,3 +495,5 @@ A forEach will cause side effects, it will be mutating the array being iterated.
 
 [airbnb-js-style-guide]: https://github.com/airbnb/javascript
 [eslintrc]: https://gitlab.com/gitlab-org/gitlab-ce/blob/master/.eslintrc
+[eslint-this]: http://eslint.org/docs/rules/class-methods-use-this
+[eslint-new]: http://eslint.org/docs/rules/no-new
diff --git a/doc/development/fe_guide/vue.md b/doc/development/fe_guide/vue.md
index 73d2ffc1bdc7b..a984bb6c94ce6 100644
--- a/doc/development/fe_guide/vue.md
+++ b/doc/development/fe_guide/vue.md
@@ -387,6 +387,10 @@ describe('Todos App', () => {
   });
 });
 ```
+#### Test the component's output
+The main return value of a Vue component is the rendered output. In order to test the component we
+need to test the rendered output. [Vue][vue-test] guide's to unit test show us exactly that:
+
 
 ### Stubbing API responses
 [Vue Resource Interceptors][vue-resource-interceptor] allow us to add a interceptor with
@@ -419,6 +423,16 @@ the response we need:
   });
 ```
 
+1. Use `$.mount()` to mount the component
+```javascript
+  // bad
+  new Component({
+    el: document.createElement('div')
+  });
+
+  // good
+  new Component().$mount();
+```
 
 [vue-docs]: http://vuejs.org/guide/index.html
 [issue-boards]: https://gitlab.com/gitlab-org/gitlab-ce/tree/master/app/assets/javascripts/boards
@@ -429,5 +443,6 @@ the response we need:
 [one-way-data-flow]: https://vuejs.org/v2/guide/components.html#One-Way-Data-Flow
 [vue-resource-repo]: https://github.com/pagekit/vue-resource
 [vue-resource-interceptor]: https://github.com/pagekit/vue-resource/blob/develop/docs/http.md#interceptors
+[vue-test]: https://vuejs.org/v2/guide/unit-testing.html
 [issue-boards-service]: https://gitlab.com/gitlab-org/gitlab-ce/blob/master/app/assets/javascripts/boards/services/board_service.js.es6
 [flux]: https://facebook.github.io/flux
diff --git a/doc/user/project/integrations/webhooks.md b/doc/user/project/integrations/webhooks.md
index dbdc93a77a8eb..e15daa2feae1b 100644
--- a/doc/user/project/integrations/webhooks.md
+++ b/doc/user/project/integrations/webhooks.md
@@ -232,6 +232,7 @@ X-Gitlab-Event: Issue Hook
   "object_attributes": {
     "id": 301,
     "title": "New API: create/update/delete file",
+    "assignee_ids": [51],
     "assignee_id": 51,
     "author_id": 51,
     "project_id": 14,
@@ -246,6 +247,11 @@ X-Gitlab-Event: Issue Hook
     "url": "http://example.com/diaspora/issues/23",
     "action": "open"
   },
+  "assignees": [{
+    "name": "User1",
+    "username": "user1",
+    "avatar_url": "http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=40\u0026d=identicon"
+  }],
   "assignee": {
     "name": "User1",
     "username": "user1",
@@ -265,6 +271,9 @@ X-Gitlab-Event: Issue Hook
   }]
 }
 ```
+
+**Note**: `assignee` and `assignee_id` keys are deprecated and now show the first assignee only.
+
 ### Comment events
 
 Triggered when a new comment is made on commits, merge requests, issues, and code snippets.
@@ -544,6 +553,7 @@ X-Gitlab-Event: Note Hook
   "issue": {
     "id": 92,
     "title": "test",
+    "assignee_ids": [],
     "assignee_id": null,
     "author_id": 1,
     "project_id": 5,
@@ -559,6 +569,8 @@ X-Gitlab-Event: Note Hook
 }
 ```
 
+**Note**: `assignee_id` field is deprecated and now shows the first assignee only.
+
 #### Comment on code snippet
 
 **Request header**:
diff --git a/features/project/builds/artifacts.feature b/features/project/builds/artifacts.feature
index 09094d638c966..5abc24949cf5c 100644
--- a/features/project/builds/artifacts.feature
+++ b/features/project/builds/artifacts.feature
@@ -46,13 +46,14 @@ Feature: Project Builds Artifacts
     And I navigate to parent directory of directory with invalid name
     Then I should not see directory with invalid name on the list
 
+  @javascript
   Scenario: I download a single file from build artifacts
     Given recent build has artifacts available
     And recent build has artifacts metadata available
     When I visit recent build details page
     And I click artifacts browse button
     And I click a link to file within build artifacts
-    Then download of a file extracted from build artifacts should start
+    Then I see a download link
 
   @javascript
   Scenario: I click on a row in an artifacts table
diff --git a/features/project/deploy_keys.feature b/features/project/deploy_keys.feature
index 960b4100ee506..6f1ed9ff5b6b6 100644
--- a/features/project/deploy_keys.feature
+++ b/features/project/deploy_keys.feature
@@ -3,28 +3,33 @@ Feature: Project Deploy Keys
     Given I sign in as a user
     And I own project "Shop"
 
+  @javascript
   Scenario: I should see deploy keys list
     Given project has deploy key
     When I visit project deploy keys page
     Then I should see project deploy key
 
+  @javascript
   Scenario: I should see project deploy keys
     Given other projects have deploy keys
     When I visit project deploy keys page
     Then I should see other project deploy key
     And I should only see the same deploy key once
 
+  @javascript
   Scenario: I should see public deploy keys
     Given public deploy key exists
     When I visit project deploy keys page
     Then I should see public deploy key
 
+  @javascript
   Scenario: I add new deploy key
     Given I visit project deploy keys page
     And I submit new deploy key
     Then I should be on deploy keys page
     And I should see newly created deploy key
 
+  @javascript
   Scenario: I attach other project deploy key to project
     Given other projects have deploy keys
     And I visit project deploy keys page
@@ -32,6 +37,7 @@ Feature: Project Deploy Keys
     Then I should be on deploy keys page
     And I should see newly created deploy key
 
+  @javascript
   Scenario: I attach public deploy key to project
     Given public deploy key exists
     And I visit project deploy keys page
diff --git a/features/steps/dashboard/dashboard.rb b/features/steps/dashboard/dashboard.rb
index c715c85c43c01..bf09d7b7114be 100644
--- a/features/steps/dashboard/dashboard.rb
+++ b/features/steps/dashboard/dashboard.rb
@@ -77,7 +77,7 @@ class Spinach::Features::Dashboard < Spinach::FeatureSteps
 
   step 'project "Shop" has issue "Bugfix1" with label "feature"' do
     project = Project.find_by(name: "Shop")
-    issue = create(:issue, title: "Bugfix1", project: project, assignee: current_user)
+    issue = create(:issue, title: "Bugfix1", project: project, assignees: [current_user])
     issue.labels << project.labels.find_by(title: 'feature')
   end
 end
diff --git a/features/steps/dashboard/todos.rb b/features/steps/dashboard/todos.rb
index 3225e19995bbe..b56558ba0d2ae 100644
--- a/features/steps/dashboard/todos.rb
+++ b/features/steps/dashboard/todos.rb
@@ -182,7 +182,7 @@ def enterprise
   end
 
   def issue
-    @issue ||= create(:issue, assignee: current_user, project: project)
+    @issue ||= create(:issue, assignees: [current_user], project: project)
   end
 
   def merge_request
diff --git a/features/steps/group/milestones.rb b/features/steps/group/milestones.rb
index 49fcd6f120143..0b0983f0d0653 100644
--- a/features/steps/group/milestones.rb
+++ b/features/steps/group/milestones.rb
@@ -113,7 +113,7 @@ def group_milestone
 
       create :issue,
         project: project,
-        assignee: current_user,
+        assignees: [current_user],
         author: current_user,
         milestone: milestone
 
@@ -125,7 +125,7 @@ def group_milestone
 
       issue = create :issue,
         project: project,
-        assignee: current_user,
+        assignees: [current_user],
         author: current_user,
         milestone: milestone
 
diff --git a/features/steps/groups.rb b/features/steps/groups.rb
index 7bb12374e5fbe..24d91c9dfbb1b 100644
--- a/features/steps/groups.rb
+++ b/features/steps/groups.rb
@@ -61,7 +61,7 @@ class Spinach::Features::Groups < Spinach::FeatureSteps
   step 'project from group "Owned" has issues assigned to me' do
     create :issue,
       project: project,
-      assignee: current_user,
+      assignees: [current_user],
       author: current_user
   end
 
@@ -168,7 +168,7 @@ class Spinach::Features::Groups < Spinach::FeatureSteps
   step 'the archived project have some issues' do
     create :issue,
       project: @archived_project,
-      assignee: current_user,
+      assignees: [current_user],
       author: current_user
   end
 
diff --git a/features/steps/project/builds/artifacts.rb b/features/steps/project/builds/artifacts.rb
index 3ec5c8e2f4ffc..eec375b0532fc 100644
--- a/features/steps/project/builds/artifacts.rb
+++ b/features/steps/project/builds/artifacts.rb
@@ -3,6 +3,7 @@ class Spinach::Features::ProjectBuildsArtifacts < Spinach::FeatureSteps
   include SharedProject
   include SharedBuilds
   include RepoHelpers
+  include WaitForAjax
 
   step 'I click artifacts download button' do
     click_link 'Download'
@@ -78,19 +79,11 @@ class Spinach::Features::ProjectBuildsArtifacts < Spinach::FeatureSteps
 
   step 'I click a link to file within build artifacts' do
     page.within('.tree-table') { find_link('ci_artifacts.txt').click }
+    wait_for_ajax
   end
 
-  step 'download of a file extracted from build artifacts should start' do
-    send_data = response_headers[Gitlab::Workhorse::SEND_DATA_HEADER]
-
-    expect(send_data).to start_with('artifacts-entry:')
-
-    base64_params = send_data.sub(/\Aartifacts\-entry:/, '')
-    params = JSON.parse(Base64.urlsafe_decode64(base64_params))
-
-    expect(params.keys).to eq(%w(Archive Entry))
-    expect(params['Archive']).to end_with('build_artifacts.zip')
-    expect(params['Entry']).to eq(Base64.encode64('ci_artifacts.txt'))
+  step 'I see a download link' do
+    expect(page).to have_link 'download it'
   end
 
   step 'I click a first row within build artifacts table' do
diff --git a/features/steps/project/deploy_keys.rb b/features/steps/project/deploy_keys.rb
index ec59a2c094e79..8ad9d4a47419c 100644
--- a/features/steps/project/deploy_keys.rb
+++ b/features/steps/project/deploy_keys.rb
@@ -8,19 +8,19 @@ class Spinach::Features::ProjectDeployKeys < Spinach::FeatureSteps
   end
 
   step 'I should see project deploy key' do
-    page.within '.deploy-keys' do
+    page.within(find('.deploy-keys')) do
       expect(page).to have_content deploy_key.title
     end
   end
 
   step 'I should see other project deploy key' do
-    page.within '.deploy-keys' do
+    page.within(find('.deploy-keys')) do
       expect(page).to have_content other_deploy_key.title
     end
   end
 
   step 'I should see public deploy key' do
-    page.within '.deploy-keys' do
+    page.within(find('.deploy-keys')) do
       expect(page).to have_content public_deploy_key.title
     end
   end
@@ -40,7 +40,8 @@ class Spinach::Features::ProjectDeployKeys < Spinach::FeatureSteps
   end
 
   step 'I should see newly created deploy key' do
-    page.within '.deploy-keys' do
+    @project.reload
+    page.within(find('.deploy-keys')) do
       expect(page).to have_content(deploy_key.title)
     end
   end
@@ -56,7 +57,7 @@ class Spinach::Features::ProjectDeployKeys < Spinach::FeatureSteps
   end
 
   step 'I should only see the same deploy key once' do
-    page.within '.deploy-keys' do
+    page.within(find('.deploy-keys')) do
       expect(page).to have_selector('ul li', count: 1)
     end
   end
@@ -66,8 +67,9 @@ class Spinach::Features::ProjectDeployKeys < Spinach::FeatureSteps
   end
 
   step 'I click attach deploy key' do
-    page.within '.deploy-keys' do
-      click_link 'Enable'
+    page.within(find('.deploy-keys')) do
+      click_button 'Enable'
+      expect(page).not_to have_selector('.fa-spinner')
     end
   end
 
diff --git a/features/steps/project/merge_requests.rb b/features/steps/project/merge_requests.rb
index ce7674a931460..08d5d6972ffa9 100644
--- a/features/steps/project/merge_requests.rb
+++ b/features/steps/project/merge_requests.rb
@@ -458,6 +458,8 @@ class Spinach::Features::ProjectMergeRequests < Spinach::FeatureSteps
       click_button "Comment"
     end
 
+    wait_for_ajax
+
     page.within ".files>div:nth-child(2) .note-body > .note-text" do
       expect(page).to have_content "Line is correct"
     end
@@ -470,6 +472,8 @@ class Spinach::Features::ProjectMergeRequests < Spinach::FeatureSteps
       fill_in "note_note", with: "Line is wrong on here"
       click_button "Comment"
     end
+
+    wait_for_ajax
   end
 
   step 'I should still see a comment like "Line is correct" in the second file' do
@@ -715,6 +719,9 @@ def leave_comment(message)
       fill_in "note_note", with: message
       click_button "Comment"
     end
+
+    wait_for_ajax
+
     page.within(".notes_holder", visible: true) do
       expect(page).to have_content message
     end
diff --git a/features/steps/shared/note.rb b/features/steps/shared/note.rb
index 7885cc7ab7729..7d260025052cf 100644
--- a/features/steps/shared/note.rb
+++ b/features/steps/shared/note.rb
@@ -24,6 +24,8 @@ module SharedNote
       fill_in "note[note]", with: "XML attached"
       click_button "Comment"
     end
+
+    wait_for_ajax
   end
 
   step 'I preview a comment text like "Bug fixed :smile:"' do
@@ -37,6 +39,8 @@ module SharedNote
     page.within(".js-main-target-form") do
       click_button "Comment"
     end
+
+    wait_for_ajax
   end
 
   step 'I write a comment like ":+1: Nice"' do
diff --git a/lib/api/api.rb b/lib/api/api.rb
index 5cf69d7212cf7..244c5992e80d8 100644
--- a/lib/api/api.rb
+++ b/lib/api/api.rb
@@ -6,6 +6,7 @@ class API < Grape::API
 
     version 'v3', using: :path do
       helpers ::API::V3::Helpers
+      helpers ::API::Helpers::CommonHelpers
 
       mount ::API::V3::AwardEmoji
       mount ::API::V3::Boards
@@ -48,6 +49,9 @@ class API < Grape::API
     end
 
     before { allow_access_with_scope :api }
+    before { Gitlab::I18n.set_locale(current_user) }
+
+    after { Gitlab::I18n.reset_locale }
 
     rescue_from Gitlab::Access::AccessDeniedError do
       rack_response({ 'message' => '403 Forbidden' }.to_json, 403)
@@ -81,6 +85,7 @@ class API < Grape::API
     # Ensure the namespace is right, otherwise we might load Grape::API::Helpers
     helpers ::SentryHelper
     helpers ::API::Helpers
+    helpers ::API::Helpers::CommonHelpers
 
     # Keep in alphabetical order
     mount ::API::AccessRequests
diff --git a/lib/api/entities.rb b/lib/api/entities.rb
index 9735e40030f23..92f2865053bf4 100644
--- a/lib/api/entities.rb
+++ b/lib/api/entities.rb
@@ -277,7 +277,11 @@ class Milestone < ProjectEntity
     class IssueBasic < ProjectEntity
       expose :label_names, as: :labels
       expose :milestone, using: Entities::Milestone
-      expose :assignee, :author, using: Entities::UserBasic
+      expose :assignees, :author, using: Entities::UserBasic
+
+      expose :assignee, using: ::API::Entities::UserBasic do |issue, options|
+        issue.assignees.first
+      end
 
       expose :user_notes_count
       expose :upvotes, :downvotes
diff --git a/lib/api/helpers/common_helpers.rb b/lib/api/helpers/common_helpers.rb
new file mode 100644
index 0000000000000..6236fdd43ca1e
--- /dev/null
+++ b/lib/api/helpers/common_helpers.rb
@@ -0,0 +1,13 @@
+module API
+  module Helpers
+    module CommonHelpers
+      def convert_parameters_from_legacy_format(params)
+        if params[:assignee_id].present?
+          params[:assignee_ids] = [params.delete(:assignee_id)]
+        end
+
+        params
+      end
+    end
+  end
+end
diff --git a/lib/api/issues.rb b/lib/api/issues.rb
index 3af3181142009..4e3132f4dca78 100644
--- a/lib/api/issues.rb
+++ b/lib/api/issues.rb
@@ -32,7 +32,8 @@ def find_issues(args = {})
 
       params :issue_params_ce do
         optional :description, type: String, desc: 'The description of an issue'
-        optional :assignee_id, type: Integer, desc: 'The ID of a user to assign issue'
+        optional :assignee_ids, type: Array[Integer], desc: 'The array of user IDs to assign issue'
+        optional :assignee_id,  type: Integer, desc: '[Deprecated] The ID of a user to assign issue'
         optional :milestone_id, type: Integer, desc: 'The ID of a milestone to assign issue'
         optional :labels, type: String, desc: 'Comma-separated list of label names'
         optional :due_date, type: String, desc: 'Date string in the format YEAR-MONTH-DAY'
@@ -140,6 +141,8 @@ def find_issues(args = {})
 
         issue_params = declared_params(include_missing: false)
 
+        issue_params = convert_parameters_from_legacy_format(issue_params)
+
         issue = ::Issues::CreateService.new(user_project,
                                             current_user,
                                             issue_params.merge(request: request, api: true)).execute
@@ -164,7 +167,7 @@ def find_issues(args = {})
                               desc: 'Date time when the issue was updated. Available only for admins and project owners.'
         optional :state_event, type: String, values: %w[reopen close], desc: 'State of the issue'
         use :issue_params
-        at_least_one_of :title, :description, :assignee_id, :milestone_id,
+        at_least_one_of :title, :description, :assignee_ids, :assignee_id, :milestone_id,
                         :labels, :created_at, :due_date, :confidential, :state_event,
                         :weight
       end
@@ -179,6 +182,8 @@ def find_issues(args = {})
 
         update_params = declared_params(include_missing: false).merge(request: request, api: true)
 
+        update_params = convert_parameters_from_legacy_format(update_params)
+
         issue = ::Issues::UpdateService.new(user_project,
                                             current_user,
                                             update_params).execute(issue)
diff --git a/lib/api/settings.rb b/lib/api/settings.rb
index f71be7a11c32d..1d8aebe11231a 100644
--- a/lib/api/settings.rb
+++ b/lib/api/settings.rb
@@ -58,6 +58,7 @@ def current_settings
         :restricted_visibility_levels,
         :send_user_confirmation_email,
         :sentry_enabled,
+        :clientside_sentry_enabled,
         :session_expire_delay,
         :shared_runners_enabled,
         :sidekiq_throttling_enabled,
@@ -138,6 +139,10 @@ def current_settings
       given sentry_enabled: ->(val) { val } do
         requires :sentry_dsn, type: String, desc: 'Sentry Data Source Name'
       end
+      optional :clientside_sentry_enabled, type: Boolean, desc: 'Sentry can also be used for reporting and logging clientside exceptions. https://sentry.io/for/javascript/'
+      given clientside_sentry_enabled: ->(val) { val } do
+        requires :clientside_sentry_dsn, type: String, desc: 'Clientside Sentry Data Source Name'
+      end
       optional :repository_storage, type: String, desc: 'Storage paths for new projects'
       optional :repository_checks_enabled, type: Boolean, desc: "GitLab will periodically run 'git fsck' in all project and wiki repositories to look for silent disk corruption issues."
       optional :koding_enabled, type: Boolean, desc: 'Enable Koding'
diff --git a/lib/api/v3/entities.rb b/lib/api/v3/entities.rb
index de87986c18b43..739a9219147d1 100644
--- a/lib/api/v3/entities.rb
+++ b/lib/api/v3/entities.rb
@@ -295,6 +295,13 @@ class ProjectHook < ::API::Entities::Hook
         expose :project_id, :issues_events, :merge_requests_events
         expose :note_events, :build_events, :pipeline_events, :wiki_page_events
       end
+
+      class Issue < ::API::Entities::Issue
+        unexpose :assignees
+        expose :assignee do |issue, options|
+          ::API::Entities::UserBasic.represent(issue.assignees.first, options)
+        end
+      end
     end
   end
 end
diff --git a/lib/api/v3/issues.rb b/lib/api/v3/issues.rb
index 6c0705820d121..be9d1feb84d5e 100644
--- a/lib/api/v3/issues.rb
+++ b/lib/api/v3/issues.rb
@@ -8,6 +8,7 @@ class Issues < Grape::API
       helpers do
         def find_issues(args = {})
           args = params.merge(args)
+          args = convert_parameters_from_legacy_format(args)
 
           args.delete(:id)
           args[:milestone_title] = args.delete(:milestone)
@@ -53,7 +54,7 @@ def find_issues(args = {})
 
       resource :issues do
         desc "Get currently authenticated user's issues" do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           optional :state, type: String, values: %w[opened closed all], default: 'all',
@@ -62,7 +63,7 @@ def find_issues(args = {})
         end
         get do
           issues = find_issues(scope: 'authored')
-          present paginate(issues), with: ::API::Entities::Issue, current_user: current_user
+          present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user
         end
       end
 
@@ -71,7 +72,7 @@ def find_issues(args = {})
       end
       resource :groups, requirements: { id: %r{[^/]+} } do
         desc 'Get a list of group issues' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           optional :state, type: String, values: %w[opened closed all], default: 'all',
@@ -83,7 +84,7 @@ def find_issues(args = {})
 
           issues = find_issues(group_id: group.id, match_all_labels: true)
 
-          present paginate(issues), with: ::API::Entities::Issue, current_user: current_user
+          present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user
         end
       end
 
@@ -95,7 +96,7 @@ def find_issues(args = {})
 
         desc 'Get a list of project issues' do
           detail 'iid filter is deprecated have been removed on V4'
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           optional :state, type: String, values: %w[opened closed all], default: 'all',
@@ -108,22 +109,22 @@ def find_issues(args = {})
 
           issues = find_issues(project_id: project.id)
 
-          present paginate(issues), with: ::API::Entities::Issue, current_user: current_user, project: user_project
+          present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
         end
 
         desc 'Get a single project issue' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :issue_id, type: Integer, desc: 'The ID of a project issue'
         end
         get ":id/issues/:issue_id" do
           issue = find_project_issue(params[:issue_id])
-          present issue, with: ::API::Entities::Issue, current_user: current_user, project: user_project
+          present issue, with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
         end
 
         desc 'Create a new project issue' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :title, type: String, desc: 'The title of an issue'
@@ -141,6 +142,7 @@ def find_issues(args = {})
 
           issue_params = declared_params(include_missing: false)
           issue_params = issue_params.merge(merge_request_to_resolve_discussions_of: issue_params.delete(:merge_request_for_resolving_discussions))
+          issue_params = convert_parameters_from_legacy_format(issue_params)
 
           issue = ::Issues::CreateService.new(user_project,
                                               current_user,
@@ -148,14 +150,14 @@ def find_issues(args = {})
           render_spam_error! if issue.spam?
 
           if issue.valid?
-            present issue, with: ::API::Entities::Issue, current_user: current_user, project: user_project
+            present issue, with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
           else
             render_validation_error!(issue)
           end
         end
 
         desc 'Update an existing issue' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :issue_id, type: Integer, desc: 'The ID of a project issue'
@@ -178,6 +180,7 @@ def find_issues(args = {})
           end
 
           update_params = declared_params(include_missing: false).merge(request: request, api: true)
+          update_params = convert_parameters_from_legacy_format(update_params)
 
           issue = ::Issues::UpdateService.new(user_project,
                                               current_user,
@@ -186,14 +189,14 @@ def find_issues(args = {})
           render_spam_error! if issue.spam?
 
           if issue.valid?
-            present issue, with: ::API::Entities::Issue, current_user: current_user, project: user_project
+            present issue, with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
           else
             render_validation_error!(issue)
           end
         end
 
         desc 'Move an existing issue' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :issue_id, type: Integer, desc: 'The ID of a project issue'
@@ -208,7 +211,7 @@ def find_issues(args = {})
 
           begin
             issue = ::Issues::MoveService.new(user_project, current_user).execute(issue, new_project)
-            present issue, with: ::API::Entities::Issue, current_user: current_user, project: user_project
+            present issue, with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
           rescue ::Issues::MoveService::MoveError => error
             render_api_error!(error.message, 400)
           end
diff --git a/lib/api/v3/merge_requests.rb b/lib/api/v3/merge_requests.rb
index 534c7992d6bf9..a70a272b5c47f 100644
--- a/lib/api/v3/merge_requests.rb
+++ b/lib/api/v3/merge_requests.rb
@@ -34,7 +34,7 @@ def issue_entity(project)
             if project.has_external_issue_tracker?
               ::API::Entities::ExternalIssue
             else
-              ::API::Entities::Issue
+              ::API::V3::Entities::Issue
             end
           end
 
diff --git a/lib/api/v3/milestones.rb b/lib/api/v3/milestones.rb
index be90cec4afcbc..4c7061d493948 100644
--- a/lib/api/v3/milestones.rb
+++ b/lib/api/v3/milestones.rb
@@ -39,7 +39,7 @@ def filter_milestones_state(milestones, state)
         end
 
         desc 'Get all issues for a single project milestone' do
-          success ::API::Entities::Issue
+          success ::API::V3::Entities::Issue
         end
         params do
           requires :milestone_id, type: Integer, desc: 'The ID of a project milestone'
@@ -56,7 +56,7 @@ def filter_milestones_state(milestones, state)
           }
 
           issues = IssuesFinder.new(current_user, finder_params).execute
-          present paginate(issues), with: ::API::Entities::Issue, current_user: current_user, project: user_project
+          present paginate(issues), with: ::API::V3::Entities::Issue, current_user: current_user, project: user_project
         end
       end
     end
diff --git a/lib/banzai/reference_parser/issue_parser.rb b/lib/banzai/reference_parser/issue_parser.rb
index e02b360924ae2..89ec715ddf6d9 100644
--- a/lib/banzai/reference_parser/issue_parser.rb
+++ b/lib/banzai/reference_parser/issue_parser.rb
@@ -28,7 +28,7 @@ def issues_for_nodes(nodes)
           nodes,
           Issue.all.includes(
             :author,
-            :assignee,
+            :assignees,
             {
               # These associations are primarily used for checking permissions.
               # Eager loading these ensures we don't end up running dozens of
diff --git a/lib/github/import.rb b/lib/github/import.rb
index d49761fd6c608..06beb607a3eca 100644
--- a/lib/github/import.rb
+++ b/lib/github/import.rb
@@ -245,7 +245,7 @@ def fetch_issues
               issue.label_ids    = label_ids(representation.labels)
               issue.milestone_id = milestone_id(representation.milestone)
               issue.author_id    = author_id
-              issue.assignee_id  = user_id(representation.assignee)
+              issue.assignee_ids = [user_id(representation.assignee)]
               issue.created_at   = representation.created_at
               issue.updated_at   = representation.updated_at
               issue.save!(validate: false)
diff --git a/lib/gitlab/chat_commands/presenters/issue_base.rb b/lib/gitlab/chat_commands/presenters/issue_base.rb
index da097a7039127..6385b5eea7494 100644
--- a/lib/gitlab/chat_commands/presenters/issue_base.rb
+++ b/lib/gitlab/chat_commands/presenters/issue_base.rb
@@ -22,7 +22,7 @@ def fields
           [
             {
               title: "Assignee",
-              value: @resource.assignee ? @resource.assignee.name : "_None_",
+              value: @resource.assignees.any? ? @resource.assignees.first.name : "_None_",
               short: true
             },
             {
diff --git a/lib/gitlab/ci/build/artifacts/metadata/entry.rb b/lib/gitlab/ci/build/artifacts/metadata/entry.rb
index 6f799c2f031b0..2e073334abca9 100644
--- a/lib/gitlab/ci/build/artifacts/metadata/entry.rb
+++ b/lib/gitlab/ci/build/artifacts/metadata/entry.rb
@@ -37,6 +37,12 @@ def file?
           !directory?
         end
 
+        def blob
+          return unless file?
+
+          @blob ||= Blob.decorate(::Ci::ArtifactBlob.new(self), nil)
+        end
+
         def has_parent?
           nodes > 0
         end
diff --git a/lib/gitlab/cycle_analytics/base_stage.rb b/lib/gitlab/cycle_analytics/base_stage.rb
index 559e3939da641..cac31ea8cff42 100644
--- a/lib/gitlab/cycle_analytics/base_stage.rb
+++ b/lib/gitlab/cycle_analytics/base_stage.rb
@@ -17,7 +17,7 @@ def as_json
       end
 
       def title
-        name.to_s.capitalize
+        raise NotImplementedError.new("Expected #{self.name} to implement title")
       end
 
       def median
diff --git a/lib/gitlab/cycle_analytics/code_stage.rb b/lib/gitlab/cycle_analytics/code_stage.rb
index 1e52b6614a152..5f9dc9a430376 100644
--- a/lib/gitlab/cycle_analytics/code_stage.rb
+++ b/lib/gitlab/cycle_analytics/code_stage.rb
@@ -13,12 +13,16 @@ def name
         :code
       end
 
+      def title
+        s_('CycleAnalyticsStage|Code')
+      end
+
       def legend
-        "Related Merge Requests"
+        _("Related Merge Requests")
       end
 
       def description
-        "Time until first merge request"
+        _("Time until first merge request")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/issue_stage.rb b/lib/gitlab/cycle_analytics/issue_stage.rb
index 213994988a5a5..7b03811efb21e 100644
--- a/lib/gitlab/cycle_analytics/issue_stage.rb
+++ b/lib/gitlab/cycle_analytics/issue_stage.rb
@@ -14,12 +14,16 @@ def name
         :issue
       end
 
+      def title
+        s_('CycleAnalyticsStage|Issue')
+      end
+
       def legend
-        "Related Issues"
+        _("Related Issues")
       end
 
       def description
-        "Time before an issue gets scheduled"
+        _("Time before an issue gets scheduled")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/plan_stage.rb b/lib/gitlab/cycle_analytics/plan_stage.rb
index 45d51d30ccc6b..1a0afb56b4fb0 100644
--- a/lib/gitlab/cycle_analytics/plan_stage.rb
+++ b/lib/gitlab/cycle_analytics/plan_stage.rb
@@ -14,12 +14,16 @@ def name
         :plan
       end
 
+      def title
+        s_('CycleAnalyticsStage|Plan')
+      end
+
       def legend
-        "Related Commits"
+        _("Related Commits")
       end
 
       def description
-        "Time before an issue starts implementation"
+        _("Time before an issue starts implementation")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/production_stage.rb b/lib/gitlab/cycle_analytics/production_stage.rb
index 9f387a0294533..0fa8a65cb9911 100644
--- a/lib/gitlab/cycle_analytics/production_stage.rb
+++ b/lib/gitlab/cycle_analytics/production_stage.rb
@@ -15,12 +15,16 @@ def name
         :production
       end
 
+      def title
+        s_('CycleAnalyticsStage|Production')
+      end
+
       def legend
-        "Related Issues"
+        _("Related Issues")
       end
 
       def description
-        "From issue creation until deploy to production"
+        _("From issue creation until deploy to production")
       end
 
       def query
diff --git a/lib/gitlab/cycle_analytics/review_stage.rb b/lib/gitlab/cycle_analytics/review_stage.rb
index 4744be834de27..cfbbdc43fd93c 100644
--- a/lib/gitlab/cycle_analytics/review_stage.rb
+++ b/lib/gitlab/cycle_analytics/review_stage.rb
@@ -13,12 +13,16 @@ def name
         :review
       end
 
+      def title
+        s_('CycleAnalyticsStage|Review')
+      end
+
       def legend
-        "Relative Merged Requests"
+        _("Related Merged Requests")
       end
 
       def description
-        "Time between merge request creation and merge/close"
+        _("Time between merge request creation and merge/close")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/staging_stage.rb b/lib/gitlab/cycle_analytics/staging_stage.rb
index 3cdbe04fbaf1d..d5684bb920178 100644
--- a/lib/gitlab/cycle_analytics/staging_stage.rb
+++ b/lib/gitlab/cycle_analytics/staging_stage.rb
@@ -14,12 +14,16 @@ def name
         :staging
       end
 
+      def title
+        s_('CycleAnalyticsStage|Staging')
+      end
+
       def legend
-        "Relative Deployed Builds"
+        _("Related Deployed Jobs")
       end
 
       def description
-        "From merge request merge until deploy to production"
+        _("From merge request merge until deploy to production")
       end
     end
   end
diff --git a/lib/gitlab/cycle_analytics/summary/base.rb b/lib/gitlab/cycle_analytics/summary/base.rb
index 43fa3795e5ca9..a917ddccac732 100644
--- a/lib/gitlab/cycle_analytics/summary/base.rb
+++ b/lib/gitlab/cycle_analytics/summary/base.rb
@@ -8,7 +8,7 @@ def initialize(project:, from:)
         end
 
         def title
-          self.class.name.demodulize
+          raise NotImplementedError.new("Expected #{self.name} to implement title")
         end
 
         def value
diff --git a/lib/gitlab/cycle_analytics/summary/commit.rb b/lib/gitlab/cycle_analytics/summary/commit.rb
index 7b8faa4d854ed..bea7886275718 100644
--- a/lib/gitlab/cycle_analytics/summary/commit.rb
+++ b/lib/gitlab/cycle_analytics/summary/commit.rb
@@ -2,6 +2,10 @@ module Gitlab
   module CycleAnalytics
     module Summary
       class Commit < Base
+        def title
+          n_('Commit', 'Commits', value)
+        end
+
         def value
           @value ||= count_commits
         end
diff --git a/lib/gitlab/cycle_analytics/summary/deploy.rb b/lib/gitlab/cycle_analytics/summary/deploy.rb
index 06032e9200edf..099d798aac691 100644
--- a/lib/gitlab/cycle_analytics/summary/deploy.rb
+++ b/lib/gitlab/cycle_analytics/summary/deploy.rb
@@ -2,6 +2,10 @@ module Gitlab
   module CycleAnalytics
     module Summary
       class Deploy < Base
+        def title
+          n_('Deploy', 'Deploys', value)
+        end
+
         def value
           @value ||= @project.deployments.where("created_at > ?", @from).count
         end
diff --git a/lib/gitlab/cycle_analytics/summary/issue.rb b/lib/gitlab/cycle_analytics/summary/issue.rb
index 008468f24b90a..9bbf7a2685f52 100644
--- a/lib/gitlab/cycle_analytics/summary/issue.rb
+++ b/lib/gitlab/cycle_analytics/summary/issue.rb
@@ -9,7 +9,7 @@ def initialize(project:, from:, current_user:)
         end
 
         def title
-          'New Issue'
+          n_('New Issue', 'New Issues', value)
         end
 
         def value
diff --git a/lib/gitlab/cycle_analytics/test_stage.rb b/lib/gitlab/cycle_analytics/test_stage.rb
index e96943833bc46..2b5f72bef8930 100644
--- a/lib/gitlab/cycle_analytics/test_stage.rb
+++ b/lib/gitlab/cycle_analytics/test_stage.rb
@@ -13,12 +13,16 @@ def name
         :test
       end
 
+      def title
+        s_('CycleAnalyticsStage|Test')
+      end
+
       def legend
-        "Relative Builds Trigger by Commits"
+        _("Related Jobs")
       end
 
       def description
-        "Total test time for all commits/merges"
+        _("Total test time for all commits/merges")
       end
 
       def stage_query
diff --git a/lib/gitlab/fogbugz_import/importer.rb b/lib/gitlab/fogbugz_import/importer.rb
index 222bcdcbf9c8a..3dcee681c72f4 100644
--- a/lib/gitlab/fogbugz_import/importer.rb
+++ b/lib/gitlab/fogbugz_import/importer.rb
@@ -122,15 +122,15 @@ def import_cases
           author_id = user_info(bug['ixPersonOpenedBy'])[:gitlab_id] || project.creator_id
 
           issue = Issue.create!(
-            iid:         bug['ixBug'],
-            project_id:  project.id,
-            title:       bug['sTitle'],
-            description: body,
-            author_id:   author_id,
-            assignee_id: assignee_id,
-            state:       bug['fOpen'] == 'true' ? 'opened' : 'closed',
-            created_at:  date,
-            updated_at:  DateTime.parse(bug['dtLastUpdated'])
+            iid:          bug['ixBug'],
+            project_id:   project.id,
+            title:        bug['sTitle'],
+            description:  body,
+            author_id:    author_id,
+            assignee_ids: [assignee_id],
+            state:        bug['fOpen'] == 'true' ? 'opened' : 'closed',
+            created_at:   date,
+            updated_at:   DateTime.parse(bug['dtLastUpdated'])
           )
 
           issue_labels = ::LabelsFinder.new(nil, project_id: project.id, title: labels).execute(skip_authorization: true)
diff --git a/lib/gitlab/git/repository.rb b/lib/gitlab/git/repository.rb
index e7485c220399d..6a0f12b7e5027 100644
--- a/lib/gitlab/git/repository.rb
+++ b/lib/gitlab/git/repository.rb
@@ -499,8 +499,9 @@ def ref_name_for_sha(ref_path, sha)
       #     :contains is the commit contained by the refs from which to begin (SHA1 or name)
       #     :max_count is the maximum number of commits to fetch
       #     :skip is the number of commits to skip
-      #     :order is the commits order and allowed value is :none (default), :date, or :topo
-      #        commit ordering types are documented here:
+      #     :order is the commits order and allowed value is :none (default), :date,
+      #        :topo, or any combination of them (in an array). Commit ordering types
+      #        are documented here:
       #        http://www.rubydoc.info/github/libgit2/rugged/Rugged#SORT_NONE-constant)
       #
       def find_commits(options = {})
@@ -1269,16 +1270,18 @@ def gitaly_migrate(method, &block)
         raise CommandError.new(e)
       end
 
-      # Returns the `Rugged` sorting type constant for a given
-      # sort type key. Valid keys are `:none`, `:topo`, and `:date`
-      def rugged_sort_type(key)
+      # Returns the `Rugged` sorting type constant for one or more given
+      # sort types. Valid keys are `:none`, `:topo`, and `:date`, or an array
+      # containing more than one of them. `:date` uses a combination of date and
+      # topological sorting to closer mimic git's native ordering.
+      def rugged_sort_type(sort_type)
         @rugged_sort_types ||= {
           none: Rugged::SORT_NONE,
           topo: Rugged::SORT_TOPO,
-          date: Rugged::SORT_DATE
+          date: Rugged::SORT_DATE | Rugged::SORT_TOPO
         }
 
-        @rugged_sort_types.fetch(key, Rugged::SORT_NONE)
+        @rugged_sort_types.fetch(sort_type, Rugged::SORT_NONE)
       end
     end
   end
diff --git a/lib/gitlab/github_import/issue_formatter.rb b/lib/gitlab/github_import/issue_formatter.rb
index 6f5ac4dac0d38..977cd0423ba53 100644
--- a/lib/gitlab/github_import/issue_formatter.rb
+++ b/lib/gitlab/github_import/issue_formatter.rb
@@ -10,7 +10,7 @@ def attributes
           description: description,
           state: state,
           author_id: author_id,
-          assignee_id: assignee_id,
+          assignee_ids: Array(assignee_id),
           created_at: raw_data.created_at,
           updated_at: raw_data.updated_at
         }
diff --git a/lib/gitlab/gon_helper.rb b/lib/gitlab/gon_helper.rb
index 5ab84266b7d51..26473f99bc351 100644
--- a/lib/gitlab/gon_helper.rb
+++ b/lib/gitlab/gon_helper.rb
@@ -10,6 +10,8 @@ def add_gon_variables
       gon.user_color_scheme      = Gitlab::ColorSchemes.for_user(current_user).css_class
       gon.katex_css_url          = ActionController::Base.helpers.asset_path('katex.css')
       gon.katex_js_url           = ActionController::Base.helpers.asset_path('katex.js')
+      gon.sentry_dsn             = current_application_settings.clientside_sentry_dsn if current_application_settings.clientside_sentry_enabled
+      gon.gitlab_url             = Gitlab.config.gitlab.url
 
       if current_user
         gon.current_user_id = current_user.id
diff --git a/lib/gitlab/google_code_import/importer.rb b/lib/gitlab/google_code_import/importer.rb
index 5ca3e6a95cab8..1b43440673c11 100644
--- a/lib/gitlab/google_code_import/importer.rb
+++ b/lib/gitlab/google_code_import/importer.rb
@@ -108,13 +108,13 @@ def import_issues
           end
 
           issue = Issue.create!(
-            iid:         raw_issue['id'],
-            project_id:  project.id,
-            title:       raw_issue['title'],
-            description: body,
-            author_id:   project.creator_id,
-            assignee_id: assignee_id,
-            state:       raw_issue['state'] == 'closed' ? 'closed' : 'opened'
+            iid:          raw_issue['id'],
+            project_id:   project.id,
+            title:        raw_issue['title'],
+            description:  body,
+            author_id:    project.creator_id,
+            assignee_ids: [assignee_id],
+            state:        raw_issue['state'] == 'closed' ? 'closed' : 'opened'
           )
 
           issue_labels = ::LabelsFinder.new(nil, project_id: project.id, title: labels).execute(skip_authorization: true)
diff --git a/lib/gitlab/i18n.rb b/lib/gitlab/i18n.rb
new file mode 100644
index 0000000000000..3411516319f88
--- /dev/null
+++ b/lib/gitlab/i18n.rb
@@ -0,0 +1,26 @@
+module Gitlab
+  module I18n
+    extend self
+
+    AVAILABLE_LANGUAGES = {
+      'en' => 'English',
+      'es' => 'Español',
+      'de' => 'Deutsch'
+    }.freeze
+
+    def available_locales
+      AVAILABLE_LANGUAGES.keys
+    end
+
+    def set_locale(current_user)
+      requested_locale = current_user&.preferred_language || ::I18n.default_locale
+      locale = FastGettext.set_locale(requested_locale)
+      ::I18n.locale = locale
+    end
+
+    def reset_locale
+      FastGettext.set_locale(::I18n.default_locale)
+      ::I18n.locale = ::I18n.default_locale
+    end
+  end
+end
diff --git a/lib/gitlab/import_export/relation_factory.rb b/lib/gitlab/import_export/relation_factory.rb
index 4a54e7ef2e736..956763fa39927 100644
--- a/lib/gitlab/import_export/relation_factory.rb
+++ b/lib/gitlab/import_export/relation_factory.rb
@@ -15,7 +15,7 @@ class RelationFactory
                     priorities: :label_priorities,
                     label: :project_label }.freeze
 
-      USER_REFERENCES = %w[author_id assignee_id updated_by_id user_id created_by_id merge_user_id resolved_by_id].freeze
+      USER_REFERENCES = %w[author_id assignee_id updated_by_id user_id created_by_id last_edited_by_id merge_user_id resolved_by_id].freeze
 
       PROJECT_REFERENCES = %w[project_id source_project_id target_project_id].freeze
 
diff --git a/lib/gitlab/prometheus.rb b/lib/gitlab/prometheus.rb
index 62239779454f6..8827507955dd7 100644
--- a/lib/gitlab/prometheus.rb
+++ b/lib/gitlab/prometheus.rb
@@ -50,6 +50,12 @@ def join_api_url(type, args = {})
 
     def get(url)
       handle_response(HTTParty.get(url))
+    rescue SocketError
+      raise PrometheusError, "Can't connect to #{url}"
+    rescue OpenSSL::SSL::SSLError
+      raise PrometheusError, "#{url} contains invalid SSL data"
+    rescue HTTParty::Error
+      raise PrometheusError, "Network connection error"
     end
 
     def handle_response(response)
diff --git a/lib/gitlab/shell.rb b/lib/gitlab/shell.rb
index 0309877e4689b..4a01ceaf275a5 100644
--- a/lib/gitlab/shell.rb
+++ b/lib/gitlab/shell.rb
@@ -83,7 +83,7 @@ def import_repository(storage, name, url)
       # Timeout should be less than 900 ideally, to prevent the memory killer
       # to silently kill the process without knowing we are timing out here.
       output, status = Popen.popen([gitlab_shell_projects_path, 'import-project',
-                                    storage, "#{name}.git", url, '800'])
+                                    storage, "#{name}.git", url, "#{Gitlab.config.gitlab_shell.git_timeout}"])
       raise Error, output unless status.zero?
       true
     end
@@ -127,7 +127,7 @@ def list_remote_tags(storage, name, remote)
     #   fetch_remote("gitlab/gitlab-ci", "upstream")
     #
     def fetch_remote(storage, name, remote, forced: false, no_tags: false)
-      args = [gitlab_shell_projects_path, 'fetch-remote', storage, "#{name}.git", remote, '800']
+      args = [gitlab_shell_projects_path, 'fetch-remote', storage, "#{name}.git", remote, "#{Gitlab.config.gitlab_shell.git_timeout}"]
       args << '--force' if forced
       args << '--no-tags' if no_tags
 
diff --git a/lib/tasks/gettext.rake b/lib/tasks/gettext.rake
new file mode 100644
index 0000000000000..0aa21a4bd13ba
--- /dev/null
+++ b/lib/tasks/gettext.rake
@@ -0,0 +1,14 @@
+require "gettext_i18n_rails/tasks"
+
+namespace :gettext do
+  # Customize list of translatable files
+  # See: https://github.com/grosser/gettext_i18n_rails#customizing-list-of-translatable-files
+  def files_to_translate
+    folders = %W(app lib config #{locale_path}).join(',')
+    exts = %w(rb erb haml slim rhtml js jsx vue coffee handlebars hbs mustache).join(',')
+
+    Dir.glob(
+      "{#{folders}}/**/*.{#{exts}}"
+    )
+  end
+end
diff --git a/locale/de/gitlab.po b/locale/de/gitlab.po
new file mode 100644
index 0000000000000..b804dc0436f4f
--- /dev/null
+++ b/locale/de/gitlab.po
@@ -0,0 +1,207 @@
+# German translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"PO-Revision-Date: 2017-04-12 22:37-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: German\n"
+"Language: de\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"\n"
+
+msgid "ByAuthor|by"
+msgstr ""
+
+msgid "Commit"
+msgid_plural "Commits"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgstr ""
+
+msgid "CycleAnalyticsStage|Code"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Issue"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Plan"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Production"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Review"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Staging"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Test"
+msgstr ""
+
+msgid "Deploy"
+msgid_plural "Deploys"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "FirstPushedBy|First"
+msgstr ""
+
+msgid "FirstPushedBy|pushed by"
+msgstr ""
+
+msgid "From issue creation until deploy to production"
+msgstr ""
+
+msgid "From merge request merge until deploy to production"
+msgstr ""
+
+msgid "Introducing Cycle Analytics"
+msgstr ""
+
+msgid "Last %d day"
+msgid_plural "Last %d days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Limited to showing %d event at most"
+msgid_plural "Limited to showing %d events at most"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Median"
+msgstr ""
+
+msgid "New Issue"
+msgid_plural "New Issues"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Not available"
+msgstr ""
+
+msgid "Not enough data"
+msgstr ""
+
+msgid "OpenedNDaysAgo|Opened"
+msgstr ""
+
+msgid "Pipeline Health"
+msgstr ""
+
+msgid "ProjectLifecycle|Stage"
+msgstr ""
+
+msgid "Read more"
+msgstr ""
+
+msgid "Related Commits"
+msgstr ""
+
+msgid "Related Deployed Jobs"
+msgstr ""
+
+msgid "Related Issues"
+msgstr ""
+
+msgid "Related Jobs"
+msgstr ""
+
+msgid "Related Merge Requests"
+msgstr ""
+
+msgid "Related Merged Requests"
+msgstr ""
+
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
+msgstr ""
+
+msgid "The collection of events added to the data gathered for that stage."
+msgstr ""
+
+msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
+msgstr ""
+
+msgid "The phase of the development lifecycle."
+msgstr ""
+
+msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
+msgstr ""
+
+msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
+msgstr ""
+
+msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
+msgstr ""
+
+msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
+msgstr ""
+
+msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
+msgstr ""
+
+msgid "The time taken by each data entry gathered by that stage."
+msgstr ""
+
+msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
+msgstr ""
+
+msgid "Time before an issue gets scheduled"
+msgstr ""
+
+msgid "Time before an issue starts implementation"
+msgstr ""
+
+msgid "Time between merge request creation and merge/close"
+msgstr ""
+
+msgid "Time until first merge request"
+msgstr ""
+
+msgid "Time|hr"
+msgid_plural "Time|hrs"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|min"
+msgid_plural "Time|mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|s"
+msgstr ""
+
+msgid "Total Time"
+msgstr ""
+
+msgid "Total test time for all commits/merges"
+msgstr ""
+
+msgid "Want to see the data? Please ask an administrator for access."
+msgstr ""
+
+msgid "We don't have enough data to show this stage."
+msgstr ""
+
+msgid "You need permission."
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
diff --git a/locale/de/gitlab.po.time_stamp b/locale/de/gitlab.po.time_stamp
new file mode 100644
index 0000000000000..e69de29bb2d1d
diff --git a/locale/en/gitlab.po b/locale/en/gitlab.po
new file mode 100644
index 0000000000000..a43bafbbe2872
--- /dev/null
+++ b/locale/en/gitlab.po
@@ -0,0 +1,207 @@
+# English translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"PO-Revision-Date: 2017-04-12 22:36-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: English\n"
+"Language: en\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"\n"
+
+msgid "ByAuthor|by"
+msgstr ""
+
+msgid "Commit"
+msgid_plural "Commits"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgstr ""
+
+msgid "CycleAnalyticsStage|Code"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Issue"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Plan"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Production"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Review"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Staging"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Test"
+msgstr ""
+
+msgid "Deploy"
+msgid_plural "Deploys"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "FirstPushedBy|First"
+msgstr ""
+
+msgid "FirstPushedBy|pushed by"
+msgstr ""
+
+msgid "From issue creation until deploy to production"
+msgstr ""
+
+msgid "From merge request merge until deploy to production"
+msgstr ""
+
+msgid "Introducing Cycle Analytics"
+msgstr ""
+
+msgid "Last %d day"
+msgid_plural "Last %d days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Limited to showing %d event at most"
+msgid_plural "Limited to showing %d events at most"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Median"
+msgstr ""
+
+msgid "New Issue"
+msgid_plural "New Issues"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Not available"
+msgstr ""
+
+msgid "Not enough data"
+msgstr ""
+
+msgid "OpenedNDaysAgo|Opened"
+msgstr ""
+
+msgid "Pipeline Health"
+msgstr ""
+
+msgid "ProjectLifecycle|Stage"
+msgstr ""
+
+msgid "Read more"
+msgstr ""
+
+msgid "Related Commits"
+msgstr ""
+
+msgid "Related Deployed Jobs"
+msgstr ""
+
+msgid "Related Issues"
+msgstr ""
+
+msgid "Related Jobs"
+msgstr ""
+
+msgid "Related Merge Requests"
+msgstr ""
+
+msgid "Related Merged Requests"
+msgstr ""
+
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
+msgstr ""
+
+msgid "The collection of events added to the data gathered for that stage."
+msgstr ""
+
+msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
+msgstr ""
+
+msgid "The phase of the development lifecycle."
+msgstr ""
+
+msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
+msgstr ""
+
+msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
+msgstr ""
+
+msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
+msgstr ""
+
+msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
+msgstr ""
+
+msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
+msgstr ""
+
+msgid "The time taken by each data entry gathered by that stage."
+msgstr ""
+
+msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
+msgstr ""
+
+msgid "Time before an issue gets scheduled"
+msgstr ""
+
+msgid "Time before an issue starts implementation"
+msgstr ""
+
+msgid "Time between merge request creation and merge/close"
+msgstr ""
+
+msgid "Time until first merge request"
+msgstr ""
+
+msgid "Time|hr"
+msgid_plural "Time|hrs"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|min"
+msgid_plural "Time|mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|s"
+msgstr ""
+
+msgid "Total Time"
+msgstr ""
+
+msgid "Total test time for all commits/merges"
+msgstr ""
+
+msgid "Want to see the data? Please ask an administrator for access."
+msgstr ""
+
+msgid "We don't have enough data to show this stage."
+msgstr ""
+
+msgid "You need permission."
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
diff --git a/locale/en/gitlab.po.time_stamp b/locale/en/gitlab.po.time_stamp
new file mode 100644
index 0000000000000..e69de29bb2d1d
diff --git a/locale/es/gitlab.po b/locale/es/gitlab.po
new file mode 100644
index 0000000000000..c14ddd3b94c6c
--- /dev/null
+++ b/locale/es/gitlab.po
@@ -0,0 +1,208 @@
+# Spanish translations for gitlab package.
+# Copyright (C) 2017 THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, 2017.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"PO-Revision-Date: 2017-05-04 19:24-0500\n"
+"Language-Team: Spanish\n"
+"Language: es\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"Last-Translator: \n"
+"X-Generator: Poedit 2.0.1\n"
+
+msgid "ByAuthor|by"
+msgstr "por"
+
+msgid "Commit"
+msgid_plural "Commits"
+msgstr[0] "Cambio"
+msgstr[1] "Cambios"
+
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgstr "Cycle Analytics ofrece una visión general de cuánto tiempo tarda en pasar de idea a producción en su proyecto."
+
+msgid "CycleAnalyticsStage|Code"
+msgstr "Código"
+
+msgid "CycleAnalyticsStage|Issue"
+msgstr "Incidencia"
+
+msgid "CycleAnalyticsStage|Plan"
+msgstr "Planificación"
+
+msgid "CycleAnalyticsStage|Production"
+msgstr "Producción"
+
+msgid "CycleAnalyticsStage|Review"
+msgstr "Revisión"
+
+#, fuzzy
+msgid "CycleAnalyticsStage|Staging"
+msgstr "Puesta en escena"
+
+msgid "CycleAnalyticsStage|Test"
+msgstr "Pruebas"
+
+msgid "Deploy"
+msgid_plural "Deploys"
+msgstr[0] "Despliegue"
+msgstr[1] "Despliegues"
+
+msgid "FirstPushedBy|First"
+msgstr "Primer"
+
+msgid "FirstPushedBy|pushed by"
+msgstr "enviado por"
+
+msgid "From issue creation until deploy to production"
+msgstr "Desde la creación de la incidencia hasta el despliegue a producción"
+
+msgid "From merge request merge until deploy to production"
+msgstr "Desde la integración de la solicitud de fusión hasta el despliegue a producción"
+
+msgid "Introducing Cycle Analytics"
+msgstr "Introducción a Cycle Analytics"
+
+msgid "Last %d day"
+msgid_plural "Last %d days"
+msgstr[0] "Último %d día"
+msgstr[1] "Últimos %d días"
+
+msgid "Limited to showing %d event at most"
+msgid_plural "Limited to showing %d events at most"
+msgstr[0] "Limitado a mostrar máximo %d evento"
+msgstr[1] "Limitado a mostrar máximo %d eventos"
+
+msgid "Median"
+msgstr "Mediana"
+
+msgid "New Issue"
+msgid_plural "New Issues"
+msgstr[0] "Nueva incidencia"
+msgstr[1] "Nuevas incidencias"
+
+msgid "Not available"
+msgstr "No disponible"
+
+msgid "Not enough data"
+msgstr "No hay suficientes datos"
+
+msgid "OpenedNDaysAgo|Opened"
+msgstr "Abierto"
+
+msgid "Pipeline Health"
+msgstr "Estado del Pipeline"
+
+msgid "ProjectLifecycle|Stage"
+msgstr "Etapa"
+
+msgid "Read more"
+msgstr "Leer más"
+
+msgid "Related Commits"
+msgstr "Cambios Relacionados"
+
+msgid "Related Deployed Jobs"
+msgstr "Trabajos Desplegados Relacionados"
+
+msgid "Related Issues"
+msgstr "Incidencias Relacionadas"
+
+msgid "Related Jobs"
+msgstr "Trabajos Relacionados"
+
+msgid "Related Merge Requests"
+msgstr "Solicitudes de fusión Relacionadas"
+
+msgid "Related Merged Requests"
+msgstr "Solicitudes de fusión Relacionadas"
+
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] "Mostrando %d evento"
+msgstr[1] "Mostrando %d eventos"
+
+msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
+msgstr "La etapa de codificación muestra el tiempo desde el primer cambio hasta la creación de la solicitud de fusión. Los datos serán automáticamente incorporados aquí una vez creada tu primera solicitud de fusión."
+
+msgid "The collection of events added to the data gathered for that stage."
+msgstr "La colección de eventos agregados a los datos recopilados para esa etapa."
+
+msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
+msgstr "La etapa de incidencia muestra el tiempo que toma desde la creación de un tema hasta asignar el tema a un hito, o añadir el tema a una lista en el panel de temas. Empieza a crear temas para ver los datos de esta etapa."
+
+msgid "The phase of the development lifecycle."
+msgstr "La etapa del ciclo de vida de desarrollo."
+
+msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
+msgstr "La etapa de planificación muestra el tiempo desde el paso anterior hasta el envío de tu primera confirmación. Este tiempo se añadirá automáticamente una vez que usted envíe el primer cambio."
+
+msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
+msgstr "La etapa de producción muestra el tiempo total que tarda entre la creación de una incidencia y el despliegue del código a producción. Los datos se añadirán automáticamente una vez haya finalizado por completo el ciclo de idea a producción."
+
+msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
+msgstr "La etapa de revisión muestra el tiempo desde la creación de la solicitud de fusión hasta que los cambios se fusionaron. Los datos se añadirán automáticamente después de fusionar su primera solicitud de fusión."
+
+msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
+msgstr "La etapa de puesta en escena muestra el tiempo entre la fusión y el despliegue de código en el entorno de producción. Los datos se añadirán automáticamente una vez que se despliega a producción por primera vez."
+
+msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
+msgstr "La etapa de pruebas muestra el tiempo que GitLab CI toma para ejecutar cada pipeline para la solicitud de fusión relacionada. Los datos se añadirán automáticamente luego de que el primer pipeline termine de ejecutarse."
+
+msgid "The time taken by each data entry gathered by that stage."
+msgstr "El tiempo utilizado por cada entrada de datos obtenido por esa etapa."
+
+msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
+msgstr "El valor en el punto medio de una serie de valores observados. Por ejemplo, entre 3, 5, 9, la mediana es 5. Entre 3, 5, 7, 8, la mediana es (5 + 7) / 2 = 6."
+
+msgid "Time before an issue gets scheduled"
+msgstr "Tiempo antes de que una incidencia sea programada"
+
+msgid "Time before an issue starts implementation"
+msgstr "Tiempo antes de que empieze la implementación de una incidencia"
+
+msgid "Time between merge request creation and merge/close"
+msgstr "Tiempo entre la creación de la solicitud de fusión y la integración o cierre de ésta"
+
+msgid "Time until first merge request"
+msgstr "Tiempo hasta la primera solicitud de fusión"
+
+msgid "Time|hr"
+msgid_plural "Time|hrs"
+msgstr[0] "hr"
+msgstr[1] "hrs"
+
+msgid "Time|min"
+msgid_plural "Time|mins"
+msgstr[0] "min"
+msgstr[1] "mins"
+
+msgid "Time|s"
+msgstr "s"
+
+msgid "Total Time"
+msgstr "Tiempo Total"
+
+msgid "Total test time for all commits/merges"
+msgstr "Tiempo total de pruebas para todos los cambios o integraciones"
+
+msgid "Want to see the data? Please ask an administrator for access."
+msgstr "¿Quieres ver los datos? Por favor pide acceso al administrador."
+
+msgid "We don't have enough data to show this stage."
+msgstr "No hay suficientes datos para mostrar en esta etapa."
+
+msgid "You need permission."
+msgstr "Necesitas permisos."
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] "día"
+msgstr[1] "días"
diff --git a/locale/es/gitlab.po.time_stamp b/locale/es/gitlab.po.time_stamp
new file mode 100644
index 0000000000000..e69de29bb2d1d
diff --git a/locale/gitlab.pot b/locale/gitlab.pot
new file mode 100644
index 0000000000000..3967d40ea9e50
--- /dev/null
+++ b/locale/gitlab.pot
@@ -0,0 +1,208 @@
+# SOME DESCRIPTIVE TITLE.
+# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the gitlab package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
+#
+#, fuzzy
+msgid ""
+msgstr ""
+"Project-Id-Version: gitlab 1.0.0\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2017-05-04 19:24-0500\n"
+"PO-Revision-Date: 2017-05-04 19:24-0500\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: LANGUAGE <LL@li.org>\n"
+"Language: \n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
+
+msgid "ByAuthor|by"
+msgstr ""
+
+msgid "Commit"
+msgid_plural "Commits"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Cycle Analytics gives an overview of how much time it takes to go from idea to production in your project."
+msgstr ""
+
+msgid "CycleAnalyticsStage|Code"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Issue"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Plan"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Production"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Review"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Staging"
+msgstr ""
+
+msgid "CycleAnalyticsStage|Test"
+msgstr ""
+
+msgid "Deploy"
+msgid_plural "Deploys"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "FirstPushedBy|First"
+msgstr ""
+
+msgid "FirstPushedBy|pushed by"
+msgstr ""
+
+msgid "From issue creation until deploy to production"
+msgstr ""
+
+msgid "From merge request merge until deploy to production"
+msgstr ""
+
+msgid "Introducing Cycle Analytics"
+msgstr ""
+
+msgid "Last %d day"
+msgid_plural "Last %d days"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Limited to showing %d event at most"
+msgid_plural "Limited to showing %d events at most"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Median"
+msgstr ""
+
+msgid "New Issue"
+msgid_plural "New Issues"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Not available"
+msgstr ""
+
+msgid "Not enough data"
+msgstr ""
+
+msgid "OpenedNDaysAgo|Opened"
+msgstr ""
+
+msgid "Pipeline Health"
+msgstr ""
+
+msgid "ProjectLifecycle|Stage"
+msgstr ""
+
+msgid "Read more"
+msgstr ""
+
+msgid "Related Commits"
+msgstr ""
+
+msgid "Related Deployed Jobs"
+msgstr ""
+
+msgid "Related Issues"
+msgstr ""
+
+msgid "Related Jobs"
+msgstr ""
+
+msgid "Related Merge Requests"
+msgstr ""
+
+msgid "Related Merged Requests"
+msgstr ""
+
+msgid "Showing %d event"
+msgid_plural "Showing %d events"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "The coding stage shows the time from the first commit to creating the merge request. The data will automatically be added here once you create your first merge request."
+msgstr ""
+
+msgid "The collection of events added to the data gathered for that stage."
+msgstr ""
+
+msgid "The issue stage shows the time it takes from creating an issue to assigning the issue to a milestone, or add the issue to a list on your Issue Board. Begin creating issues to see data for this stage."
+msgstr ""
+
+msgid "The phase of the development lifecycle."
+msgstr ""
+
+msgid "The planning stage shows the time from the previous step to pushing your first commit. This time will be added automatically once you push your first commit."
+msgstr ""
+
+msgid "The production stage shows the total time it takes between creating an issue and deploying the code to production. The data will be automatically added once you have completed the full idea to production cycle."
+msgstr ""
+
+msgid "The review stage shows the time from creating the merge request to merging it. The data will automatically be added after you merge your first merge request."
+msgstr ""
+
+msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
+msgstr ""
+
+msgid "The testing stage shows the time GitLab CI takes to run every pipeline for the related merge request. The data will automatically be added after your first pipeline finishes running."
+msgstr ""
+
+msgid "The time taken by each data entry gathered by that stage."
+msgstr ""
+
+msgid "The value lying at the midpoint of a series of observed values. E.g., between 3, 5, 9, the median is 5. Between 3, 5, 7, 8, the median is (5+7)/2 = 6."
+msgstr ""
+
+msgid "Time before an issue gets scheduled"
+msgstr ""
+
+msgid "Time before an issue starts implementation"
+msgstr ""
+
+msgid "Time between merge request creation and merge/close"
+msgstr ""
+
+msgid "Time until first merge request"
+msgstr ""
+
+msgid "Time|hr"
+msgid_plural "Time|hrs"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|min"
+msgid_plural "Time|mins"
+msgstr[0] ""
+msgstr[1] ""
+
+msgid "Time|s"
+msgstr ""
+
+msgid "Total Time"
+msgstr ""
+
+msgid "Total test time for all commits/merges"
+msgstr ""
+
+msgid "Want to see the data? Please ask an administrator for access."
+msgstr ""
+
+msgid "We don't have enough data to show this stage."
+msgstr ""
+
+msgid "You need permission."
+msgstr ""
+
+msgid "day"
+msgid_plural "days"
+msgstr[0] ""
+msgstr[1] ""
diff --git a/package.json b/package.json
index 3d260a66a53f4..22f9a6833aa65 100644
--- a/package.json
+++ b/package.json
@@ -27,7 +27,9 @@
     "dropzone": "^4.2.0",
     "emoji-unicode-version": "^0.2.1",
     "eslint-plugin-html": "^2.0.1",
+    "exports-loader": "^0.6.4",
     "file-loader": "^0.11.1",
+    "jed": "^1.1.1",
     "jquery": "^2.2.1",
     "jquery-ujs": "^1.2.1",
     "js-cookie": "^2.1.3",
@@ -39,9 +41,11 @@
     "pikaday": "^1.5.1",
     "prismjs": "^1.6.0",
     "raphael": "^2.2.7",
+    "raven-js": "^3.14.0",
     "raw-loader": "^0.5.1",
     "react-dev-utils": "^0.5.2",
     "select2": "3.5.2-browserify",
+    "sql.js": "^0.4.0",
     "stats-webpack-plugin": "^0.4.3",
     "three": "^0.84.0",
     "three-orbit-controls": "^82.1.0",
diff --git a/scripts/static-analysis b/scripts/static-analysis
index 1bd6b33983090..7dc8f67903640 100755
--- a/scripts/static-analysis
+++ b/scripts/static-analysis
@@ -3,6 +3,7 @@
 require ::File.expand_path('../lib/gitlab/popen', __dir__)
 
 tasks = [
+  %w[bundle exec bundle-audit check --update --ignore CVE-2016-4658],
   %w[bundle exec rake config_lint],
   %w[bundle exec rake flay],
   %w[bundle exec rake haml_lint],
diff --git a/spec/controllers/admin/services_controller_spec.rb b/spec/controllers/admin/services_controller_spec.rb
index e5cdd52307ee4..c94616d8508f4 100644
--- a/spec/controllers/admin/services_controller_spec.rb
+++ b/spec/controllers/admin/services_controller_spec.rb
@@ -23,4 +23,36 @@
       end
     end
   end
+
+  describe "#update" do
+    let(:project) { create(:empty_project) }
+    let!(:service) do
+      RedmineService.create(
+        project: project,
+        active: false,
+        template: true,
+        properties: {
+          project_url: 'http://abc',
+          issues_url: 'http://abc',
+          new_issue_url: 'http://abc'
+        }
+      )
+    end
+
+    it 'calls the propagation worker when service is active' do
+      expect(PropagateServiceTemplateWorker).to receive(:perform_async).with(service.id)
+
+      put :update, id: service.id, service: { active: true }
+
+      expect(response).to have_http_status(302)
+    end
+
+    it 'does not call the propagation worker when service is not active' do
+      expect(PropagateServiceTemplateWorker).not_to receive(:perform_async)
+
+      put :update, id: service.id, service: { properties: {} }
+
+      expect(response).to have_http_status(302)
+    end
+  end
 end
diff --git a/spec/controllers/dashboard/todos_controller_spec.rb b/spec/controllers/dashboard/todos_controller_spec.rb
index 762e90f4a1672..085f3fd8543d5 100644
--- a/spec/controllers/dashboard/todos_controller_spec.rb
+++ b/spec/controllers/dashboard/todos_controller_spec.rb
@@ -14,7 +14,7 @@
   describe 'GET #index' do
     context 'when using pagination' do
       let(:last_page) { user.todos.page.total_pages }
-      let!(:issues) { create_list(:issue, 2, project: project, assignee: user) }
+      let!(:issues) { create_list(:issue, 2, project: project, assignees: [user]) }
 
       before do
         issues.each { |issue| todo_service.new_issue(issue, user) }
diff --git a/spec/controllers/projects/artifacts_controller_spec.rb b/spec/controllers/projects/artifacts_controller_spec.rb
new file mode 100644
index 0000000000000..eff9fab8da2a5
--- /dev/null
+++ b/spec/controllers/projects/artifacts_controller_spec.rb
@@ -0,0 +1,188 @@
+require 'spec_helper'
+
+describe Projects::ArtifactsController do
+  let(:user) { create(:user) }
+  let(:project) { create(:project, :repository) }
+
+  let(:pipeline) do
+    create(:ci_pipeline,
+            project: project,
+            sha: project.commit.sha,
+            ref: project.default_branch,
+            status: 'success')
+  end
+
+  let(:build) { create(:ci_build, :success, :artifacts, pipeline: pipeline) }
+
+  before do
+    project.team << [user, :developer]
+
+    sign_in(user)
+  end
+
+  describe 'GET download' do
+    it 'sends the artifacts file' do
+      expect(controller).to receive(:send_file).with(build.artifacts_file.path, disposition: 'attachment').and_call_original
+
+      get :download, namespace_id: project.namespace, project_id: project, build_id: build
+    end
+  end
+
+  describe 'GET browse' do
+    context 'when the directory exists' do
+      it 'renders the browse view' do
+        get :browse, namespace_id: project.namespace, project_id: project, build_id: build, path: 'other_artifacts_0.1.2'
+
+        expect(response).to render_template('projects/artifacts/browse')
+      end
+    end
+
+    context 'when the directory does not exist' do
+      it 'responds Not Found' do
+        get :browse, namespace_id: project.namespace, project_id: project, build_id: build, path: 'unknown'
+
+        expect(response).to be_not_found
+      end
+    end
+  end
+
+  describe 'GET file' do
+    context 'when the file exists' do
+      it 'renders the file view' do
+        get :file, namespace_id: project.namespace, project_id: project, build_id: build, path: 'ci_artifacts.txt'
+
+        expect(response).to render_template('projects/artifacts/file')
+      end
+    end
+
+    context 'when the file does not exist' do
+      it 'responds Not Found' do
+        get :file, namespace_id: project.namespace, project_id: project, build_id: build, path: 'unknown'
+
+        expect(response).to be_not_found
+      end
+    end
+  end
+
+  describe 'GET raw' do
+    context 'when the file exists' do
+      it 'serves the file using workhorse' do
+        get :raw, namespace_id: project.namespace, project_id: project, build_id: build, path: 'ci_artifacts.txt'
+
+        send_data = response.headers[Gitlab::Workhorse::SEND_DATA_HEADER]
+
+        expect(send_data).to start_with('artifacts-entry:')
+
+        base64_params = send_data.sub(/\Aartifacts\-entry:/, '')
+        params = JSON.parse(Base64.urlsafe_decode64(base64_params))
+
+        expect(params.keys).to eq(%w(Archive Entry))
+        expect(params['Archive']).to end_with('build_artifacts.zip')
+        expect(params['Entry']).to eq(Base64.encode64('ci_artifacts.txt'))
+      end
+    end
+
+    context 'when the file does not exist' do
+      it 'responds Not Found' do
+        get :raw, namespace_id: project.namespace, project_id: project, build_id: build, path: 'unknown'
+
+        expect(response).to be_not_found
+      end
+    end
+  end
+
+  describe 'GET latest_succeeded' do
+    def params_from_ref(ref = pipeline.ref, job = build.name, path = 'browse')
+      {
+        namespace_id: project.namespace,
+        project_id: project,
+        ref_name_and_path: File.join(ref, path),
+        job: job
+      }
+    end
+
+    context 'cannot find the build' do
+      shared_examples 'not found' do
+        it { expect(response).to have_http_status(:not_found) }
+      end
+
+      context 'has no such ref' do
+        before do
+          get :latest_succeeded, params_from_ref('TAIL', build.name)
+        end
+
+        it_behaves_like 'not found'
+      end
+
+      context 'has no such build' do
+        before do
+          get :latest_succeeded, params_from_ref(pipeline.ref, 'NOBUILD')
+        end
+
+        it_behaves_like 'not found'
+      end
+
+      context 'has no path' do
+        before do
+          get :latest_succeeded, params_from_ref(pipeline.sha, build.name, '')
+        end
+
+        it_behaves_like 'not found'
+      end
+    end
+
+    context 'found the build and redirect' do
+      shared_examples 'redirect to the build' do
+        it 'redirects' do
+          path = browse_namespace_project_build_artifacts_path(
+            project.namespace,
+            project,
+            build)
+
+          expect(response).to redirect_to(path)
+        end
+      end
+
+      context 'with regular branch' do
+        before do
+          pipeline.update(ref: 'master',
+                          sha: project.commit('master').sha)
+
+          get :latest_succeeded, params_from_ref('master')
+        end
+
+        it_behaves_like 'redirect to the build'
+      end
+
+      context 'with branch name containing slash' do
+        before do
+          pipeline.update(ref: 'improve/awesome',
+                          sha: project.commit('improve/awesome').sha)
+
+          get :latest_succeeded, params_from_ref('improve/awesome')
+        end
+
+        it_behaves_like 'redirect to the build'
+      end
+
+      context 'with branch name and path containing slashes' do
+        before do
+          pipeline.update(ref: 'improve/awesome',
+                          sha: project.commit('improve/awesome').sha)
+
+          get :latest_succeeded, params_from_ref('improve/awesome', build.name, 'file/README.md')
+        end
+
+        it 'redirects' do
+          path = file_namespace_project_build_artifacts_path(
+            project.namespace,
+            project,
+            build,
+            'README.md')
+
+          expect(response).to redirect_to(path)
+        end
+      end
+    end
+  end
+end
diff --git a/spec/controllers/projects/boards/issues_controller_spec.rb b/spec/controllers/projects/boards/issues_controller_spec.rb
index 817c2285672c1..f5d5204c24563 100644
--- a/spec/controllers/projects/boards/issues_controller_spec.rb
+++ b/spec/controllers/projects/boards/issues_controller_spec.rb
@@ -34,7 +34,7 @@
           issue = create(:labeled_issue, project: project, labels: [planning])
           create(:labeled_issue, project: project, labels: [planning])
           create(:labeled_issue, project: project, labels: [development], due_date: Date.tomorrow)
-          create(:labeled_issue, project: project, labels: [development], assignee: johndoe)
+          create(:labeled_issue, project: project, labels: [development], assignees: [johndoe])
           issue.subscribe(johndoe, project)
 
           list_issues user: user, board: board, list: list2
diff --git a/spec/controllers/projects/deploy_keys_controller_spec.rb b/spec/controllers/projects/deploy_keys_controller_spec.rb
new file mode 100644
index 0000000000000..efe1a78415b7d
--- /dev/null
+++ b/spec/controllers/projects/deploy_keys_controller_spec.rb
@@ -0,0 +1,66 @@
+require 'spec_helper'
+
+describe Projects::DeployKeysController do
+  let(:project) { create(:project, :repository) }
+  let(:user) { create(:user) }
+
+  before do
+    project.team << [user, :master]
+
+    sign_in(user)
+  end
+
+  describe 'GET index' do
+    let(:params) do
+      { namespace_id: project.namespace, project_id: project }
+    end
+
+    context 'when html requested' do
+      it 'redirects to blob' do
+        get :index, params
+
+        expect(response).to redirect_to(namespace_project_settings_repository_path(params))
+      end
+    end
+
+    context 'when json requested' do
+      let(:project2) { create(:empty_project, :internal)}
+      let(:project_private) { create(:empty_project, :private)}
+
+      let(:deploy_key_internal) do
+        create(:deploy_key, key: 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQCdMHEHyhRjbhEZVddFn6lTWdgEy5Q6Bz4nwGB76xWZI5YT/1WJOMEW+sL5zYd31kk7sd3FJ5L9ft8zWMWrr/iWXQikC2cqZK24H1xy+ZUmrRuJD4qGAaIVoyyzBL+avL+lF8J5lg6YSw8gwJY/lX64/vnJHUlWw2n5BF8IFOWhiw== dummy@gitlab.com')
+      end
+      let(:deploy_key_actual) do
+        create(:deploy_key, key: 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQDNd/UJWhPrpb+b/G5oL109y57yKuCxE+WUGJGYaj7WQKsYRJmLYh1mgjrl+KVyfsWpq4ylOxIfFSnN9xBBFN8mlb0Fma5DC7YsSsibJr3MZ19ZNBprwNcdogET7aW9I0In7Wu5f2KqI6e5W/spJHCy4JVxzVMUvk6Myab0LnJ2iQ== dummy@gitlab.com')
+      end
+      let!(:deploy_key_public) { create(:deploy_key, public: true) }
+
+      let!(:deploy_keys_project_internal) do
+        create(:deploy_keys_project, project: project2, deploy_key: deploy_key_internal)
+      end
+
+      let!(:deploy_keys_actual_project) do
+        create(:deploy_keys_project, project: project, deploy_key: deploy_key_actual)
+      end
+
+      let!(:deploy_keys_project_private) do
+        create(:deploy_keys_project, project: project_private, deploy_key: create(:another_deploy_key))
+      end
+
+      before do
+        project2.team << [user, :developer]
+      end
+
+      it 'returns json in a correct format' do
+        get :index, params.merge(format: :json)
+
+        json = JSON.parse(response.body)
+
+        expect(json.keys).to match_array(%w(enabled_keys available_project_keys public_keys))
+        expect(json['enabled_keys'].count).to eq(1)
+        expect(json['available_project_keys'].count).to eq(1)
+        expect(json['public_keys'].count).to eq(1)
+      end
+    end
+  end
+end
diff --git a/spec/controllers/projects/issues_controller_spec.rb b/spec/controllers/projects/issues_controller_spec.rb
index 5f1f892821a05..1f79e72495ac7 100644
--- a/spec/controllers/projects/issues_controller_spec.rb
+++ b/spec/controllers/projects/issues_controller_spec.rb
@@ -173,12 +173,12 @@
           namespace_id: project.namespace.to_param,
           project_id: project,
           id: issue.iid,
-          issue: { assignee_id: assignee.id },
+          issue: { assignee_ids: [assignee.id] },
           format: :json
         body = JSON.parse(response.body)
 
-        expect(body['assignee'].keys)
-          .to match_array(%w(name username avatar_url))
+        expect(body['assignees'].first.keys)
+          .to match_array(%w(id name username avatar_url))
       end
     end
 
@@ -348,7 +348,7 @@ def move_issue
     let(:admin) { create(:admin) }
     let!(:issue) { create(:issue, project: project) }
     let!(:unescaped_parameter_value) { create(:issue, :confidential, project: project, author: author) }
-    let!(:request_forgery_timing_attack) { create(:issue, :confidential, project: project, assignee: assignee) }
+    let!(:request_forgery_timing_attack) { create(:issue, :confidential, project: project, assignees: [assignee]) }
 
     describe 'GET #index' do
       it 'does not list confidential issues for guests' do
diff --git a/spec/controllers/projects/merge_requests_controller_spec.rb b/spec/controllers/projects/merge_requests_controller_spec.rb
index df54e21c8083c..730d2156b1dfd 100644
--- a/spec/controllers/projects/merge_requests_controller_spec.rb
+++ b/spec/controllers/projects/merge_requests_controller_spec.rb
@@ -1356,7 +1356,7 @@ def post_assign_issues
     end
 
     it 'correctly pluralizes flash message on success' do
-      issue2.update!(assignee: user)
+      issue2.assignees = [user]
 
       post_assign_issues
 
diff --git a/spec/controllers/projects/pipelines_controller_spec.rb b/spec/controllers/projects/pipelines_controller_spec.rb
index b9bacc5a64a25..1b47d163c0b3d 100644
--- a/spec/controllers/projects/pipelines_controller_spec.rb
+++ b/spec/controllers/projects/pipelines_controller_spec.rb
@@ -5,6 +5,8 @@
   let(:project) { create(:empty_project, :public) }
 
   before do
+    project.add_developer(user)
+
     sign_in(user)
   end
 
@@ -87,4 +89,38 @@ def get_stage(name)
       expect(json_response['favicon']).to eq "/assets/ci_favicons/#{status.favicon}.ico"
     end
   end
+
+  describe 'POST retry.json' do
+    let!(:pipeline) { create(:ci_pipeline, :failed, project: project) }
+    let!(:build) { create(:ci_build, :failed, pipeline: pipeline) }
+
+    before do
+      post :retry, namespace_id: project.namespace,
+                   project_id: project,
+                   id: pipeline.id,
+                   format: :json
+    end
+
+    it 'retries a pipeline without returning any content' do
+      expect(response).to have_http_status(:no_content)
+      expect(build.reload).to be_retried
+    end
+  end
+
+  describe 'POST cancel.json' do
+    let!(:pipeline) { create(:ci_pipeline, project: project) }
+    let!(:build) { create(:ci_build, :running, pipeline: pipeline) }
+
+    before do
+      post :cancel, namespace_id: project.namespace,
+                    project_id: project,
+                    id: pipeline.id,
+                    format: :json
+    end
+
+    it 'cancels a pipeline without returning any content' do
+      expect(response).to have_http_status(:no_content)
+      expect(pipeline.reload).to be_canceled
+    end
+  end
 end
diff --git a/spec/features/atom/dashboard_issues_spec.rb b/spec/features/atom/dashboard_issues_spec.rb
index 58b14e09740fa..9ea325ab41b05 100644
--- a/spec/features/atom/dashboard_issues_spec.rb
+++ b/spec/features/atom/dashboard_issues_spec.rb
@@ -32,7 +32,7 @@
       end
 
       context "issue with basic fields" do
-        let!(:issue2) { create(:issue, author: user, assignee: assignee, project: project2, description: 'test desc') }
+        let!(:issue2) { create(:issue, author: user, assignees: [assignee], project: project2, description: 'test desc') }
 
         it "renders issue fields" do
           visit issues_dashboard_path(:atom, private_token: user.private_token)
@@ -41,7 +41,7 @@
 
           expect(entry).to be_present
           expect(entry).to have_selector('author email', text: issue2.author_public_email)
-          expect(entry).to have_selector('assignee email', text: issue2.assignee_public_email)
+          expect(entry).to have_selector('assignees email', text: assignee.public_email)
           expect(entry).not_to have_selector('labels')
           expect(entry).not_to have_selector('milestone')
           expect(entry).to have_selector('description', text: issue2.description)
@@ -51,7 +51,7 @@
       context "issue with label and milestone" do
         let!(:milestone1) { create(:milestone, project: project1, title: 'v1') }
         let!(:label1)     { create(:label, project: project1, title: 'label1') }
-        let!(:issue1)     { create(:issue, author: user, assignee: assignee, project: project1, milestone: milestone1) }
+        let!(:issue1)     { create(:issue, author: user, assignees: [assignee], project: project1, milestone: milestone1) }
 
         before do
           issue1.labels << label1
@@ -64,7 +64,7 @@
 
           expect(entry).to be_present
           expect(entry).to have_selector('author email', text: issue1.author_public_email)
-          expect(entry).to have_selector('assignee email', text: issue1.assignee_public_email)
+          expect(entry).to have_selector('assignees email', text: assignee.public_email)
           expect(entry).to have_selector('labels label', text: label1.title)
           expect(entry).to have_selector('milestone', text: milestone1.title)
           expect(entry).not_to have_selector('description')
diff --git a/spec/features/atom/issues_spec.rb b/spec/features/atom/issues_spec.rb
index b3903ec2faf92..4f6754ad54105 100644
--- a/spec/features/atom/issues_spec.rb
+++ b/spec/features/atom/issues_spec.rb
@@ -6,7 +6,7 @@
     let!(:assignee) { create(:user, email: 'private2@example.com', public_email: 'public2@example.com') }
     let!(:group)    { create(:group) }
     let!(:project)  { create(:project) }
-    let!(:issue)    { create(:issue, author: user, assignee: assignee, project: project) }
+    let!(:issue)    { create(:issue, author: user, assignees: [assignee], project: project) }
 
     before do
       project.team << [user, :developer]
@@ -22,7 +22,8 @@
           to have_content('application/atom+xml')
         expect(body).to have_selector('title', text: "#{project.name} issues")
         expect(body).to have_selector('author email', text: issue.author_public_email)
-        expect(body).to have_selector('assignee email', text: issue.author_public_email)
+        expect(body).to have_selector('assignees assignee email', text: issue.assignees.first.public_email)
+        expect(body).to have_selector('assignee email', text: issue.assignees.first.public_email)
         expect(body).to have_selector('entry summary', text: issue.title)
       end
     end
@@ -36,7 +37,8 @@
           to have_content('application/atom+xml')
         expect(body).to have_selector('title', text: "#{project.name} issues")
         expect(body).to have_selector('author email', text: issue.author_public_email)
-        expect(body).to have_selector('assignee email', text: issue.author_public_email)
+        expect(body).to have_selector('assignees assignee email', text: issue.assignees.first.public_email)
+        expect(body).to have_selector('assignee email', text: issue.assignees.first.public_email)
         expect(body).to have_selector('entry summary', text: issue.title)
       end
     end
diff --git a/spec/features/boards/board_with_milestone_spec.rb b/spec/features/boards/board_with_milestone_spec.rb
index 845921ad0686c..2b11ec1cf6409 100644
--- a/spec/features/boards/board_with_milestone_spec.rb
+++ b/spec/features/boards/board_with_milestone_spec.rb
@@ -109,7 +109,7 @@
     it 'removes issues milestone when removing from the board' do
       wait_for_vue_resource
 
-      first('.card').click
+      first('.card .card-number').click
 
       click_button('Remove from board')
 
diff --git a/spec/features/boards/boards_spec.rb b/spec/features/boards/boards_spec.rb
index f962cd6d0b6eb..83e03f539bbe2 100644
--- a/spec/features/boards/boards_spec.rb
+++ b/spec/features/boards/boards_spec.rb
@@ -71,7 +71,7 @@
     let!(:list2) { create(:list, board: board, label: development, position: 1) }
 
     let!(:confidential_issue) { create(:labeled_issue, :confidential, project: project, author: user, labels: [planning], relative_position: 9) }
-    let!(:issue1) { create(:labeled_issue, project: project, assignee: user, labels: [planning], relative_position: 8) }
+    let!(:issue1) { create(:labeled_issue, project: project, assignees: [user], labels: [planning], relative_position: 8) }
     let!(:issue2) { create(:labeled_issue, project: project, author: user2, labels: [planning], relative_position: 7) }
     let!(:issue3) { create(:labeled_issue, project: project, labels: [planning], relative_position: 6) }
     let!(:issue4) { create(:labeled_issue, project: project, labels: [planning], relative_position: 5) }
diff --git a/spec/features/boards/modal_filter_spec.rb b/spec/features/boards/modal_filter_spec.rb
index 4a4c13e79c8a9..e1367c675e58b 100644
--- a/spec/features/boards/modal_filter_spec.rb
+++ b/spec/features/boards/modal_filter_spec.rb
@@ -98,7 +98,7 @@
   end
 
   context 'assignee' do
-    let!(:issue) { create(:issue, project: project, assignee: user2) }
+    let!(:issue) { create(:issue, project: project, assignees: [user2]) }
 
     before do
       project.team << [user2, :developer]
diff --git a/spec/features/boards/sidebar_spec.rb b/spec/features/boards/sidebar_spec.rb
index bafa4f0593710..02b6b5dc88865 100644
--- a/spec/features/boards/sidebar_spec.rb
+++ b/spec/features/boards/sidebar_spec.rb
@@ -4,13 +4,14 @@
   include WaitForVueResource
 
   let(:user)         { create(:user) }
+  let(:user2)        { create(:user) }
   let(:project)      { create(:empty_project, :public) }
   let!(:milestone)   { create(:milestone, project: project) }
   let!(:development) { create(:label, project: project, name: 'Development') }
   let!(:bug)         { create(:label, project: project, name: 'Bug') }
   let!(:regression)  { create(:label, project: project, name: 'Regression') }
   let!(:stretch)     { create(:label, project: project, name: 'Stretch') }
-  let!(:issue1)      { create(:labeled_issue, project: project, assignee: user, milestone: milestone, labels: [development], relative_position: 2) }
+  let!(:issue1)      { create(:labeled_issue, project: project, assignees: [user], milestone: milestone, labels: [development], relative_position: 2) }
   let!(:issue2)      { create(:labeled_issue, project: project, labels: [development, stretch], relative_position: 1) }
   let(:board)        { create(:board, project: project) }
   let!(:list)        { create(:list, board: board, label: development, position: 0) }
@@ -20,6 +21,7 @@
     Timecop.freeze
 
     project.team << [user, :master]
+    project.team.add_developer(user2)
 
     login_as(user)
 
@@ -101,6 +103,26 @@
       expect(card).to have_selector('.avatar')
     end
 
+    it 'adds multiple assignees' do
+      click_card(card)
+
+      page.within('.assignee') do
+        click_link 'Edit'
+
+        wait_for_ajax
+
+        page.within('.dropdown-menu-user') do
+          click_link user.name
+          click_link user2.name
+        end
+
+        expect(page).to have_content(user.name)
+        expect(page).to have_content(user2.name)
+      end
+
+      expect(card.all('.avatar').length).to eq(2)
+    end
+
     it 'removes the assignee' do
       card_two = first('.board').find('.card:nth-child(2)')
       click_card(card_two)
@@ -112,10 +134,11 @@
 
         page.within('.dropdown-menu-user') do
           click_link 'Unassigned'
-
-          wait_for_vue_resource
         end
 
+        find('.dropdown-menu-toggle').click
+        wait_for_vue_resource
+
         expect(page).to have_content('No assignee')
       end
 
@@ -128,7 +151,7 @@
       page.within(find('.assignee')) do
         expect(page).to have_content('No assignee')
 
-        click_link 'assign yourself'
+        click_button 'assign yourself'
 
         wait_for_vue_resource
 
@@ -138,7 +161,7 @@
       expect(card).to have_selector('.avatar')
     end
 
-    it 'resets assignee dropdown' do
+    it 'updates assignee dropdown' do
       click_card(card)
 
       page.within('.assignee') do
@@ -162,7 +185,7 @@
       page.within('.assignee') do
         click_link 'Edit'
 
-        expect(page).not_to have_selector('.is-active')
+        expect(page).to have_selector('.is-active')
       end
     end
   end
diff --git a/spec/features/cycle_analytics_spec.rb b/spec/features/cycle_analytics_spec.rb
index b93275c330bbc..7c9d522273bea 100644
--- a/spec/features/cycle_analytics_spec.rb
+++ b/spec/features/cycle_analytics_spec.rb
@@ -62,6 +62,25 @@
         expect_issue_to_be_present
       end
     end
+
+    context "when my preferred language is Spanish" do
+      before do
+        user.update_attribute(:preferred_language, 'es')
+
+        project.team << [user, :master]
+        login_as(user)
+        visit namespace_project_cycle_analytics_path(project.namespace, project)
+        wait_for_ajax
+      end
+
+      it 'shows the content in Spanish' do
+        expect(page).to have_content('Estado del Pipeline')
+      end
+
+      it 'resets the language to English' do
+        expect(I18n.locale).to eq(:en)
+      end
+    end
   end
 
   context "as a guest" do
diff --git a/spec/features/dashboard/issuables_counter_spec.rb b/spec/features/dashboard/issuables_counter_spec.rb
index 4fca7577e7431..6f7bf0eba6ebf 100644
--- a/spec/features/dashboard/issuables_counter_spec.rb
+++ b/spec/features/dashboard/issuables_counter_spec.rb
@@ -7,7 +7,7 @@
   let(:merge_request) { create(:merge_request, source_project: project) }
 
   before do
-    issue.update(assignee: user)
+    issue.assignees = [user]
     merge_request.update(assignee: user)
     login_as(user)
   end
@@ -17,7 +17,9 @@
 
     expect_counters('issues', '1')
 
-    issue.update(assignee: nil)
+    issue.assignees = []
+
+    user.update_cache_counts
 
     Timecop.travel(3.minutes.from_now) do
       visit issues_path
diff --git a/spec/features/dashboard/issues_spec.rb b/spec/features/dashboard/issues_spec.rb
index f4420814c3a1b..86c7954e60cf0 100644
--- a/spec/features/dashboard/issues_spec.rb
+++ b/spec/features/dashboard/issues_spec.rb
@@ -11,7 +11,7 @@
 
   let!(:authored_issue) { create :issue, author: current_user, project: project }
   let!(:authored_issue_on_public_project) { create :issue, author: current_user, project: public_project }
-  let!(:assigned_issue) { create :issue, assignee: current_user, project: project }
+  let!(:assigned_issue) { create :issue, assignees: [current_user], project: project }
   let!(:other_issue) { create :issue, project: project }
 
   before do
@@ -30,6 +30,11 @@
     find('#assignee_id', visible: false).set('')
     find('.js-author-search', match: :first).click
     find('.dropdown-menu-author li a', match: :first, text: current_user.to_reference).click
+    find('.js-author-search', match: :first).click
+
+    page.within '.dropdown-menu-user' do
+      expect(find('.dropdown-menu-author li a.is-active', match: :first, text: current_user.to_reference)).to be_visible
+    end
 
     expect(page).to have_content(authored_issue.title)
     expect(page).to have_content(authored_issue_on_public_project.title)
diff --git a/spec/features/dashboard_issues_spec.rb b/spec/features/dashboard_issues_spec.rb
index b6b879052315c..ad60fb2c74f51 100644
--- a/spec/features/dashboard_issues_spec.rb
+++ b/spec/features/dashboard_issues_spec.rb
@@ -10,8 +10,8 @@
       project.team << [user, :master]
       login_as(user)
 
-      create(:issue, project: project, author: user, assignee: user)
-      create(:issue, project: project, author: user, assignee: user, milestone: milestone)
+      create(:issue, project: project, author: user, assignees: [user])
+      create(:issue, project: project, author: user, assignees: [user], milestone: milestone)
 
       visit_issues
     end
diff --git a/spec/features/gitlab_flavored_markdown_spec.rb b/spec/features/gitlab_flavored_markdown_spec.rb
index f5b54463df8ff..005a029a39319 100644
--- a/spec/features/gitlab_flavored_markdown_spec.rb
+++ b/spec/features/gitlab_flavored_markdown_spec.rb
@@ -54,11 +54,11 @@
     before do
       @other_issue = create(:issue,
                             author: @user,
-                            assignee: @user,
+                            assignees: [@user],
                             project: project)
       @issue = create(:issue,
                       author: @user,
-                      assignee: @user,
+                      assignees: [@user],
                       project: project,
                       title: "fix #{@other_issue.to_reference}",
                       description: "ask #{fred.to_reference} for details")
diff --git a/spec/features/issues/award_emoji_spec.rb b/spec/features/issues/award_emoji_spec.rb
index 71df3c949db4b..853632614c447 100644
--- a/spec/features/issues/award_emoji_spec.rb
+++ b/spec/features/issues/award_emoji_spec.rb
@@ -7,7 +7,7 @@
   let!(:user)      { create(:user) }
   let(:issue) do
     create(:issue,
-           assignee: @user,
+           assignees: [user],
            project: project)
   end
 
diff --git a/spec/features/issues/csv_spec.rb b/spec/features/issues/csv_spec.rb
index 4667859c4c8f5..7dc83dd595831 100644
--- a/spec/features/issues/csv_spec.rb
+++ b/spec/features/issues/csv_spec.rb
@@ -76,7 +76,7 @@ def csv
     create_list(:labeled_issue,
                 10,
                 project: project,
-                assignee: user,
+                assignees: [user],
                 author: user,
                 milestone: milestone,
                 labels: [feature_label, idea_label])
diff --git a/spec/features/issues/filtered_search/filter_issues_spec.rb b/spec/features/issues/filtered_search/filter_issues_spec.rb
index c824aa6a41450..a8f4e2d7e10ad 100644
--- a/spec/features/issues/filtered_search/filter_issues_spec.rb
+++ b/spec/features/issues/filtered_search/filter_issues_spec.rb
@@ -51,15 +51,15 @@ def select_search_at_index(pos)
     create(:issue, project: project, title: "issue with 'single quotes'")
     create(:issue, project: project, title: "issue with \"double quotes\"")
     create(:issue, project: project, title: "issue with !@\#{$%^&*()-+")
-    create(:issue, project: project, title: "issue by assignee", milestone: milestone, author: user, assignee: user)
-    create(:issue, project: project, title: "issue by assignee with searchTerm", milestone: milestone, author: user, assignee: user)
+    create(:issue, project: project, title: "issue by assignee", milestone: milestone, author: user, assignees: [user])
+    create(:issue, project: project, title: "issue by assignee with searchTerm", milestone: milestone, author: user, assignees: [user])
 
     issue = create(:issue,
       title: "Bug 2",
       project: project,
       milestone: milestone,
       author: user,
-      assignee: user)
+      assignees: [user])
     issue.labels << bug_label
 
     issue_with_caps_label = create(:issue,
@@ -67,7 +67,7 @@ def select_search_at_index(pos)
       project: project,
       milestone: milestone,
       author: user,
-      assignee: user)
+      assignees: [user])
     issue_with_caps_label.labels << caps_sensitive_label
 
     issue_with_everything = create(:issue,
@@ -75,7 +75,7 @@ def select_search_at_index(pos)
       project: project,
       milestone: milestone,
       author: user,
-      assignee: user)
+      assignees: [user])
     issue_with_everything.labels << bug_label
     issue_with_everything.labels << caps_sensitive_label
 
diff --git a/spec/features/issues/filtered_search/filter_issues_weight_spec.rb b/spec/features/issues/filtered_search/filter_issues_weight_spec.rb
index b0c0f4679ad4a..f53bddfb50350 100644
--- a/spec/features/issues/filtered_search/filter_issues_weight_spec.rb
+++ b/spec/features/issues/filtered_search/filter_issues_weight_spec.rb
@@ -31,7 +31,7 @@ def expect_issues_list_count(open_count, closed_count = 0)
       title: 'Bug report 1',
       milestone: milestone,
       author: user,
-      assignee: user)
+      assignees: [user])
     issue.labels << label
 
     visit namespace_project_issues_path(project.namespace, project)
diff --git a/spec/features/issues/form_spec.rb b/spec/features/issues/form_spec.rb
index eb55a9f9fd494..4b4530c17110b 100644
--- a/spec/features/issues/form_spec.rb
+++ b/spec/features/issues/form_spec.rb
@@ -10,7 +10,7 @@
   let!(:milestone) { create(:milestone, project: project) }
   let!(:label)     { create(:label, project: project) }
   let!(:label2)    { create(:label, project: project) }
-  let!(:issue)     { create(:issue, project: project, assignee: user, milestone: milestone) }
+  let!(:issue)     { create(:issue, project: project, assignees: [user], milestone: milestone) }
 
   before do
     project.team << [user, :master]
@@ -23,25 +23,65 @@
       visit new_namespace_project_issue_path(project.namespace, project)
     end
 
+    describe 'multiple assignees' do
+      before do
+        click_button 'Unassigned'
+      end
+
+      it 'unselects other assignees when unassigned is selected' do
+        page.within '.dropdown-menu-user' do
+          click_link user2.name
+        end
+
+        page.within '.dropdown-menu-user' do
+          click_link 'Unassigned'
+        end
+
+        page.within '.js-assignee-search' do
+          expect(page).to have_content 'Unassigned'
+        end
+
+        expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match('0')
+      end
+
+      it 'toggles assign to me when current user is selected and unselected' do
+        page.within '.dropdown-menu-user' do
+          click_link user.name
+        end
+
+        expect(find('a', text: 'Assign to me', visible: false)).not_to be_visible
+
+        page.within '.dropdown-menu-user' do
+          click_link user.name
+        end
+
+        expect(find('a', text: 'Assign to me')).to be_visible
+      end
+    end
+
     it 'allows user to create new issue' do
       fill_in 'issue_title', with: 'title'
       fill_in 'issue_description', with: 'title'
 
       expect(find('a', text: 'Assign to me')).to be_visible
-      click_button 'Assignee'
+      click_button 'Unassigned'
       page.within '.dropdown-menu-user' do
         click_link user2.name
       end
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user2.id.to_s)
+      expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match(user2.id.to_s)
       page.within '.js-assignee-search' do
         expect(page).to have_content user2.name
       end
       expect(find('a', text: 'Assign to me')).to be_visible
 
       click_link 'Assign to me'
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user.id.to_s)
+      assignee_ids = page.all('input[name="issue[assignee_ids][]"]', visible: false)
+
+      expect(assignee_ids[0].value).to match(user2.id.to_s)
+      expect(assignee_ids[1].value).to match(user.id.to_s)
+
       page.within '.js-assignee-search' do
-        expect(page).to have_content user.name
+        expect(page).to have_content "#{user2.name} + 1 more"
       end
       expect(find('a', text: 'Assign to me', visible: false)).not_to be_visible
 
@@ -77,7 +117,7 @@
 
       page.within '.issuable-sidebar' do
         page.within '.assignee' do
-          expect(page).to have_content user.name
+          expect(page).to have_content "2 Assignees"
         end
 
         page.within '.milestone' do
@@ -120,15 +160,12 @@
     end
 
     it 'correctly updates the selected user when changing assignee' do
-      click_button 'Assignee'
+      click_button 'Unassigned'
       page.within '.dropdown-menu-user' do
         click_link user.name
       end
 
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user.id.to_s)
-
-      click_button user.name
-
+      expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match(user.id.to_s)
       expect(find('.dropdown-menu-user a.is-active').first(:xpath, '..')['data-user-id']).to eq(user.id.to_s)
 
       # check the ::before pseudo element to ensure checkmark icon is present
@@ -139,11 +176,13 @@
         click_link user2.name
       end
 
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user2.id.to_s)
+      expect(page.all('input[name="issue[assignee_ids][]"]', visible: false)[0].value).to match(user.id.to_s)
+      expect(page.all('input[name="issue[assignee_ids][]"]', visible: false)[1].value).to match(user2.id.to_s)
 
-      click_button user2.name
+      expect(page.all('.dropdown-menu-user a.is-active').length).to eq(2)
 
-      expect(find('.dropdown-menu-user a.is-active').first(:xpath, '..')['data-user-id']).to eq(user2.id.to_s)
+      expect(page.all('.dropdown-menu-user a.is-active')[0].first(:xpath, '..')['data-user-id']).to eq(user.id.to_s)
+      expect(page.all('.dropdown-menu-user a.is-active')[1].first(:xpath, '..')['data-user-id']).to eq(user2.id.to_s)
     end
   end
 
@@ -153,7 +192,7 @@
     end
 
     it 'allows user to update issue' do
-      expect(find('input[name="issue[assignee_id]"]', visible: false).value).to match(user.id.to_s)
+      expect(find('input[name="issue[assignee_ids][]"]', visible: false).value).to match(user.id.to_s)
       expect(find('input[name="issue[milestone_id]"]', visible: false).value).to match(milestone.id.to_s)
       expect(find('a', text: 'Assign to me', visible: false)).not_to be_visible
 
diff --git a/spec/features/issues/issue_sidebar_spec.rb b/spec/features/issues/issue_sidebar_spec.rb
index a1987e904a58c..e6ed8f1434483 100644
--- a/spec/features/issues/issue_sidebar_spec.rb
+++ b/spec/features/issues/issue_sidebar_spec.rb
@@ -42,6 +42,21 @@
         expect(page).to have_content(user2.name)
       end
     end
+
+    it 'assigns yourself' do
+      find('.block.assignee .dropdown-menu-toggle').click
+
+      click_button 'assign yourself'
+
+      wait_for_ajax
+
+      find('.block.assignee .edit-link').click
+
+      page.within '.dropdown-menu-user' do
+        expect(page.find('.dropdown-header')).to be_visible
+        expect(page.find('.dropdown-menu-user-link.is-active')).to have_content(user.name)
+      end
+    end
   end
 
   context 'as a allowed user' do
diff --git a/spec/features/issues/update_issues_spec.rb b/spec/features/issues/update_issues_spec.rb
index 7fa83c1fcf764..b250fa2ed3c80 100644
--- a/spec/features/issues/update_issues_spec.rb
+++ b/spec/features/issues/update_issues_spec.rb
@@ -99,7 +99,7 @@ def create_closed
   end
 
   def create_assigned
-    create(:issue, project: project, assignee: user)
+    create(:issue, project: project, assignees: [user])
   end
 
   def create_with_milestone
diff --git a/spec/features/issues_spec.rb b/spec/features/issues_spec.rb
index 8154d5d264c5e..2ee2abe8ed0fe 100644
--- a/spec/features/issues_spec.rb
+++ b/spec/features/issues_spec.rb
@@ -18,7 +18,7 @@
     let!(:issue) do
       create(:issue,
              author: @user,
-             assignee: @user,
+             assignees: [@user],
              project: project)
     end
 
@@ -43,7 +43,7 @@
     let!(:issue) do
       create(:issue,
              author: @user,
-             assignee: @user,
+             assignees: [@user],
              project: project)
     end
 
@@ -61,7 +61,7 @@
         expect(page).to have_content 'No assignee - assign yourself'
       end
 
-      expect(issue.reload.assignee).to be_nil
+      expect(issue.reload.assignees).to be_empty
     end
   end
 
@@ -138,7 +138,7 @@
 
   describe 'Issue info' do
     it 'excludes award_emoji from comment count' do
-      issue = create(:issue, author: @user, assignee: @user, project: project, title: 'foobar')
+      issue = create(:issue, author: @user, assignees: [@user], project: project, title: 'foobar')
       create(:award_emoji, awardable: issue)
 
       visit namespace_project_issues_path(project.namespace, project, assignee_id: @user.id)
@@ -164,14 +164,14 @@
       %w(foobar barbaz gitlab).each do |title|
         create(:issue,
                author: @user,
-               assignee: @user,
+               assignees: [@user],
                project: project,
                title: title)
       end
 
       @issue = Issue.find_by(title: 'foobar')
       @issue.milestone = create(:milestone, project: project)
-      @issue.assignee = nil
+      @issue.assignees = []
       @issue.save
     end
 
@@ -362,9 +362,9 @@
       let(:user2) { create(:user) }
 
       before do
-        foo.assignee = user2
+        foo.assignees << user2
         foo.save
-        bar.assignee = user2
+        bar.assignees << user2
         bar.save
       end
 
@@ -407,7 +407,7 @@
   end
 
   describe 'update labels from issue#show', js: true do
-    let(:issue) { create(:issue, project: project, author: @user, assignee: @user) }
+    let(:issue) { create(:issue, project: project, author: @user, assignees: [@user]) }
     let!(:label) { create(:label, project: project) }
 
     before do
@@ -426,7 +426,7 @@
   end
 
   describe 'update assignee from issue#show' do
-    let(:issue) { create(:issue, project: project, author: @user, assignee: @user) }
+    let(:issue) { create(:issue, project: project, author: @user, assignees: [@user]) }
 
     context 'by authorized user' do
       it 'allows user to select unassigned', js: true do
@@ -437,10 +437,14 @@
 
           click_link 'Edit'
           click_link 'Unassigned'
+          first('.title').click
           expect(page).to have_content 'No assignee'
         end
 
-        expect(issue.reload.assignee).to be_nil
+        # wait_for_ajax does not work with vue-resource at the moment
+        sleep 1
+
+        expect(issue.reload.assignees).to be_empty
       end
 
       it 'allows user to select an assignee', js: true do
@@ -472,14 +476,18 @@
           click_link 'Edit'
           click_link @user.name
 
-          page.within '.value' do
+          find('.dropdown-menu-toggle').click
+
+          page.within '.value .author' do
             expect(page).to have_content @user.name
           end
 
           click_link 'Edit'
           click_link @user.name
 
-          page.within '.value' do
+          find('.dropdown-menu-toggle').click
+
+          page.within '.value .assign-yourself' do
             expect(page).to have_content "No assignee"
           end
         end
@@ -498,7 +506,7 @@
         login_with guest
 
         visit namespace_project_issue_path(project.namespace, project, issue)
-        expect(page).to have_content issue.assignee.name
+        expect(page).to have_content issue.assignees.first.name
       end
     end
   end
@@ -590,7 +598,7 @@
       let(:user2) { create(:user) }
 
       before do
-        issue.assignee = user2
+        issue.assignees << user2
         issue.save
       end
     end
@@ -687,7 +695,7 @@
 
   describe 'due date' do
     context 'update due on issue#show', js: true do
-      let(:issue) { create(:issue, project: project, author: @user, assignee: @user) }
+      let(:issue) { create(:issue, project: project, author: @user, assignees: [@user]) }
 
       before do
         visit namespace_project_issue_path(project.namespace, project, issue)
@@ -734,7 +742,7 @@
     include WaitForVueResource
 
     it 'updates the title', js: true do
-      issue = create(:issue, author: @user, assignee: @user, project: project, title: 'new title')
+      issue = create(:issue, author: @user, assignees: [@user], project: project, title: 'new title')
 
       visit namespace_project_issue_path(project.namespace, project, issue)
 
diff --git a/spec/features/merge_requests/assign_issues_spec.rb b/spec/features/merge_requests/assign_issues_spec.rb
index 43cc6f2a2a7d7..ec49003772b82 100644
--- a/spec/features/merge_requests/assign_issues_spec.rb
+++ b/spec/features/merge_requests/assign_issues_spec.rb
@@ -33,7 +33,7 @@ def visit_merge_request(current_user = nil)
     end
 
     it "doesn't display if related issues are already assigned" do
-      [issue1, issue2].each { |issue| issue.update!(assignee: user) }
+      [issue1, issue2].each { |issue| issue.update!(assignees: [user]) }
 
       visit_merge_request
 
diff --git a/spec/features/merge_requests/user_posts_notes_spec.rb b/spec/features/merge_requests/user_posts_notes_spec.rb
index c7cc4d6bc724e..7fc0e2ce6eca4 100644
--- a/spec/features/merge_requests/user_posts_notes_spec.rb
+++ b/spec/features/merge_requests/user_posts_notes_spec.rb
@@ -98,6 +98,7 @@
           find('.btn-save').click
         end
 
+        wait_for_ajax
         find('.note').hover
         find('.js-note-edit').click
 
diff --git a/spec/features/merge_requests/user_uses_slash_commands_spec.rb b/spec/features/merge_requests/user_uses_slash_commands_spec.rb
index 3986439061ec8..08cd39fb8589c 100644
--- a/spec/features/merge_requests/user_uses_slash_commands_spec.rb
+++ b/spec/features/merge_requests/user_uses_slash_commands_spec.rb
@@ -160,6 +160,7 @@
         it 'changes target branch from a note' do
           write_note("message start \n/target_branch merge-test\n message end.")
 
+          wait_for_ajax
           expect(page).not_to have_content('/target_branch')
           expect(page).to have_content('message start')
           expect(page).to have_content('message end.')
diff --git a/spec/features/milestones/show_spec.rb b/spec/features/milestones/show_spec.rb
index 41f569678b6ed..1b24ec0dc4929 100644
--- a/spec/features/milestones/show_spec.rb
+++ b/spec/features/milestones/show_spec.rb
@@ -5,7 +5,7 @@
   let(:project) { create(:empty_project) }
   let(:milestone) { create(:milestone, project: project, start_date: Date.today, due_date: 7.days.from_now) }
   let(:labels) { create_list(:label, 2, project: project) }
-  let(:issue_params) { { project: project, assignee: user, author: user, milestone: milestone, labels: labels } }
+  let(:issue_params) { { project: project, assignees: [user], author: user, milestone: milestone, labels: labels } }
 
   before do
     project.add_user(user, :developer)
@@ -25,7 +25,7 @@ def visit_milestone
   end
 
   context 'burndown' do
-    let(:issue_params) { { project: project, assignee: user, author: user, milestone: milestone } }
+    let(:issue_params) { { project: project, assignees: [user], author: user, milestone: milestone } }
 
     context 'when any closed issues do not have closed_at value' do
       it 'shows warning' do
diff --git a/spec/features/projects/artifacts/file_spec.rb b/spec/features/projects/artifacts/file_spec.rb
new file mode 100644
index 0000000000000..74308a7e8dd39
--- /dev/null
+++ b/spec/features/projects/artifacts/file_spec.rb
@@ -0,0 +1,59 @@
+require 'spec_helper'
+
+feature 'Artifact file', :js, feature: true do
+  let(:project) { create(:project, :public) }
+  let(:pipeline) { create(:ci_empty_pipeline, project: project, sha: project.commit.sha, ref: 'master') }
+  let(:build) { create(:ci_build, :artifacts, pipeline: pipeline) }
+
+  def visit_file(path)
+    visit file_namespace_project_build_artifacts_path(project.namespace, project, build, path)
+  end
+
+  context 'Text file' do
+    before do
+      visit_file('other_artifacts_0.1.2/doc_sample.txt')
+
+      wait_for_ajax
+    end
+
+    it 'displays an error' do
+      aggregate_failures do
+        # shows an error message
+        expect(page).to have_content('The source could not be displayed because it is stored as a job artifact. You can download it instead.')
+
+        # does not show a viewer switcher
+        expect(page).not_to have_selector('.js-blob-viewer-switcher')
+
+        # does not show a copy button
+        expect(page).not_to have_selector('.js-copy-blob-source-btn')
+
+        # shows a download button
+        expect(page).to have_link('Download')
+      end
+    end
+  end
+
+  context 'JPG file' do
+    before do
+      visit_file('rails_sample.jpg')
+
+      wait_for_ajax
+    end
+
+    it 'displays the blob' do
+      aggregate_failures do
+        # shows rendered image
+        expect(page).to have_selector('.image_file img')
+
+        # does not show a viewer switcher
+        expect(page).not_to have_selector('.js-blob-viewer-switcher')
+
+        # does not show a copy button
+        expect(page).not_to have_selector('.js-copy-blob-source-btn')
+
+        # shows a download button
+        expect(page).to have_link('Download')
+      end
+    end
+  end
+end
diff --git a/spec/features/projects/audit_events_spec.rb b/spec/features/projects/audit_events_spec.rb
index d3c28177c8646..891a5c76ba2dd 100644
--- a/spec/features/projects/audit_events_spec.rb
+++ b/spec/features/projects/audit_events_spec.rb
@@ -26,7 +26,7 @@
       visit namespace_project_deploy_keys_path(project.namespace, project)
 
       accept_confirm do
-        click_link 'Remove'
+        click_on 'Remove'
       end
 
       visit namespace_project_audit_events_path(project.namespace, project)
diff --git a/spec/features/projects/branches/new_branch_ref_dropdown_spec.rb b/spec/features/projects/branches/new_branch_ref_dropdown_spec.rb
new file mode 100644
index 0000000000000..cfc782c98ad42
--- /dev/null
+++ b/spec/features/projects/branches/new_branch_ref_dropdown_spec.rb
@@ -0,0 +1,48 @@
+require 'spec_helper'
+
+describe 'New Branch Ref Dropdown', :js, :feature do
+  let(:user) { create(:user) }
+  let(:project) { create(:project, :public) }
+  let(:toggle) { find('.create-from .dropdown-toggle') }
+
+  before do
+    project.add_master(user)
+
+    login_as(user)
+    visit new_namespace_project_branch_path(project.namespace, project)
+  end
+
+  it 'filters a list of branches and tags' do
+    toggle.click
+
+    filter_by('v1.0.0')
+
+    expect(items_count).to be(1)
+
+    filter_by('video')
+
+    expect(items_count).to be(1)
+
+    find('.create-from .dropdown-content li').click
+
+    expect(toggle).to have_content 'video'
+  end
+
+  it 'accepts a manually entered commit SHA' do
+    toggle.click
+
+    filter_by('somecommitsha')
+
+    find('.create-from input[type=search]').send_keys(:enter)
+
+    expect(toggle).to have_content 'somecommitsha'
+  end
+
+  def items_count
+    all('.create-from .dropdown-content li').length
+  end
+
+  def filter_by(filter_text)
+    fill_in 'Filter by Git revision', with: filter_text
+  end
+end
diff --git a/spec/features/projects/deploy_keys_spec.rb b/spec/features/projects/deploy_keys_spec.rb
index 0b997f130ea6a..06abfbbc86b78 100644
--- a/spec/features/projects/deploy_keys_spec.rb
+++ b/spec/features/projects/deploy_keys_spec.rb
@@ -1,6 +1,6 @@
 require 'spec_helper'
 
-describe 'Project deploy keys', feature: true do
+describe 'Project deploy keys', :js, :feature do
   let(:user) { create(:user) }
   let(:project) { create(:project_empty_repo) }
 
@@ -17,9 +17,13 @@
     it 'removes association between project and deploy key' do
       visit namespace_project_settings_repository_path(project.namespace, project)
 
-      page.within '.deploy-keys' do
-        expect { click_on 'Remove' }
-          .to change { project.deploy_keys.count }.by(-1)
+      page.within(find('.deploy-keys')) do
+        expect(page).to have_selector('.deploy-keys li', count: 1)
+
+        click_on 'Remove'
+
+        expect(page).not_to have_selector('.fa-spinner', count: 0)
+        expect(page).to have_selector('.deploy-keys li', count: 0)
       end
     end
   end
diff --git a/spec/features/projects/issuable_templates_spec.rb b/spec/features/projects/issuable_templates_spec.rb
index d28a853bbc2fc..fa5e30075e305 100644
--- a/spec/features/projects/issuable_templates_spec.rb
+++ b/spec/features/projects/issuable_templates_spec.rb
@@ -12,7 +12,7 @@
   context 'user creates an issue using templates' do
     let(:template_content) { 'this is a test "bug" template' }
     let(:longtemplate_content) { %Q(this\n\n\n\n\nis\n\n\n\n\na\n\n\n\n\nbug\n\n\n\n\ntemplate) }
-    let(:issue) { create(:issue, author: user, assignee: user, project: project) }
+    let(:issue) { create(:issue, author: user, assignees: [user], project: project) }
     let(:description_addition) { ' appending to description' }
 
     background do
@@ -72,7 +72,7 @@
   context 'user creates an issue using templates, with a prior description' do
     let(:prior_description) { 'test issue description' }
     let(:template_content) { 'this is a test "bug" template' }
-    let(:issue) { create(:issue, author: user, assignee: user, project: project) }
+    let(:issue) { create(:issue, author: user, assignees: [user], project: project) }
 
     background do
       project.repository.create_file(
diff --git a/spec/features/projects/pipelines/pipeline_spec.rb b/spec/features/projects/pipelines/pipeline_spec.rb
index 5a53e48f5f81d..cfac54ef259ad 100644
--- a/spec/features/projects/pipelines/pipeline_spec.rb
+++ b/spec/features/projects/pipelines/pipeline_spec.rb
@@ -254,4 +254,57 @@
       it { expect(build_manual.reload).to be_pending }
     end
   end
+
+  describe 'GET /:project/pipelines/:id/failures' do
+    let(:project) { create(:project) }
+    let(:pipeline) { create(:ci_pipeline, project: project, ref: 'master', sha: project.commit.id) }
+    let(:pipeline_failures_page) { failures_namespace_project_pipeline_path(project.namespace, project, pipeline) }
+    let!(:failed_build) { create(:ci_build, :failed, pipeline: pipeline) }
+
+    context 'with failed build' do
+      before do
+        failed_build.trace.set('4 examples, 1 failure')
+
+        visit pipeline_failures_page
+      end
+
+      it 'shows jobs tab pane as active' do
+        expect(page).to have_content('Failed Jobs')
+        expect(page).to have_css('#js-tab-failures.active')
+      end
+
+      it 'lists failed builds' do
+        expect(page).to have_content(failed_build.name)
+        expect(page).to have_content(failed_build.stage)
+      end
+
+      it 'shows build failure logs' do
+        expect(page).to have_content('4 examples, 1 failure')
+      end
+    end
+
+    context 'when missing build logs' do
+      before do
+        visit pipeline_failures_page
+      end
+
+      it 'includes failed jobs' do
+        expect(page).to have_content('No job trace')
+      end
+    end
+
+    context 'without failures' do
+      before do
+        failed_build.update!(status: :success)
+
+        visit pipeline_failures_page
+      end
+
+      it 'displays the pipeline graph' do
+        expect(current_path).to eq(pipeline_path(pipeline))
+        expect(page).not_to have_content('Failed Jobs')
+        expect(page).to have_selector('.pipeline-visualization')
+      end
+    end
+  end
 end
diff --git a/spec/features/raven_js_spec.rb b/spec/features/raven_js_spec.rb
new file mode 100644
index 0000000000000..e8fa49c18cbd5
--- /dev/null
+++ b/spec/features/raven_js_spec.rb
@@ -0,0 +1,23 @@
+require 'spec_helper'
+
+feature 'RavenJS', :feature, :js do
+  let(:raven_path) { '/raven.bundle.js' }
+
+  it 'should not load raven if sentry is disabled' do
+    visit new_user_session_path
+
+    expect(has_requested_raven).to eq(false)
+  end
+
+  it 'should load raven if sentry is enabled' do
+    stub_application_setting(clientside_sentry_dsn: 'https://key@domain.com/id', clientside_sentry_enabled: true)
+
+    visit new_user_session_path
+
+    expect(has_requested_raven).to eq(true)
+  end
+
+  def has_requested_raven
+    page.driver.network_traffic.one? {|request| request.url.end_with?(raven_path)}
+  end
+end
diff --git a/spec/features/search_spec.rb b/spec/features/search_spec.rb
index da6388dcdf20d..498a4a5cba04c 100644
--- a/spec/features/search_spec.rb
+++ b/spec/features/search_spec.rb
@@ -5,7 +5,7 @@
 
   let(:user) { create(:user) }
   let(:project) { create(:empty_project, namespace: user.namespace) }
-  let!(:issue) { create(:issue, project: project, assignee: user) }
+  let!(:issue) { create(:issue, project: project, assignees: [user]) }
   let!(:issue2) { create(:issue, project: project, author: user) }
 
   before do
diff --git a/spec/features/unsubscribe_links_spec.rb b/spec/features/unsubscribe_links_spec.rb
index e2d9cfdd0b037..a23c4ca2b92f6 100644
--- a/spec/features/unsubscribe_links_spec.rb
+++ b/spec/features/unsubscribe_links_spec.rb
@@ -6,7 +6,7 @@
   let(:recipient) { create(:user) }
   let(:author) { create(:user) }
   let(:project) { create(:empty_project, :public) }
-  let(:params) { { title: 'A bug!', description: 'Fix it!', assignee: recipient } }
+  let(:params) { { title: 'A bug!', description: 'Fix it!', assignees: [recipient] } }
   let(:issue) { Issues::CreateService.new(project, author, params).execute }
 
   let(:mail) { ActionMailer::Base.deliveries.last }
diff --git a/spec/finders/issues_finder_spec.rb b/spec/finders/issues_finder_spec.rb
index 2b251b76ac8b2..1a7aec1dc782c 100644
--- a/spec/finders/issues_finder_spec.rb
+++ b/spec/finders/issues_finder_spec.rb
@@ -7,12 +7,12 @@
   set(:project2) { create(:empty_project) }
   set(:milestone) { create(:milestone, project: project1) }
   set(:label) { create(:label, project: project2) }
-  set(:issue1) { create(:issue, author: user, assignee: user, project: project1, milestone: milestone, title: 'gitlab') }
-  set(:issue2) { create(:issue, author: user, assignee: user, project: project2, description: 'gitlab') }
-  set(:issue3) { create(:issue, author: user2, assignee: user2, project: project2, title: 'tanuki', description: 'tanuki') }
+  set(:issue1) { create(:issue, author: user, assignees: [user], project: project1, milestone: milestone, title: 'gitlab') }
+  set(:issue2) { create(:issue, author: user, assignees: [user], project: project2, description: 'gitlab') }
+  set(:issue3) { create(:issue, author: user2, assignees: [user2], project: project2, title: 'tanuki', description: 'tanuki') }
 
   describe '#execute' do
-    set(:closed_issue) { create(:issue, author: user2, assignee: user2, project: project2, state: 'closed') }
+    set(:closed_issue) { create(:issue, author: user2, assignees: [user2], project: project2, state: 'closed') }
     set(:label_link) { create(:label_link, label: label, target: issue2) }
     let(:search_user) { user }
     let(:params) { {} }
@@ -107,7 +107,7 @@
 
         before do
           milestones.each do |milestone|
-            create(:issue, project: milestone.project, milestone: milestone, author: user, assignee: user)
+            create(:issue, project: milestone.project, milestone: milestone, author: user, assignees: [user])
           end
         end
 
@@ -142,7 +142,7 @@
 
         before do
           milestones.each do |milestone|
-            create(:issue, project: milestone.project, milestone: milestone, author: user, assignee: user)
+            create(:issue, project: milestone.project, milestone: milestone, author: user, assignees: [user])
           end
         end
 
diff --git a/spec/fixtures/api/schemas/issue.json b/spec/fixtures/api/schemas/issue.json
index e01af38a75e04..0020f267aef1e 100644
--- a/spec/fixtures/api/schemas/issue.json
+++ b/spec/fixtures/api/schemas/issue.json
@@ -40,21 +40,41 @@
         "additionalProperties": false
       }
     },
-    "assignee": {
-      "type": ["object", "null"],
-      "required": [
-        "id",
-        "name",
-        "username",
-        "avatar_url"
-      ],
-      "properties": {
-        "id": { "type": "integer" },
-        "name": { "type": "string" },
-        "username": { "type": "string" },
-        "avatar_url": { "type": "uri" }
-      },
-      "additionalProperties": false
+    "assignees": {
+      "type": "array",
+      "items": {
+        "type": ["object", "null"],
+        "required": [
+          "id",
+          "name",
+          "username",
+          "avatar_url"
+        ],
+        "properties": {
+          "id": { "type": "integer" },
+          "name": { "type": "string" },
+          "username": { "type": "string" },
+          "avatar_url": { "type": "uri" }
+        }
+      }
+    },
+    "assignees": {
+      "type": "array",
+      "items": {
+        "type": ["object", "null"],
+        "required": [
+          "id",
+          "name",
+          "username",
+          "avatar_url"
+        ],
+        "properties": {
+          "id": { "type": "integer" },
+          "name": { "type": "string" },
+          "username": { "type": "string" },
+          "avatar_url": { "type": "uri" }
+        }
+      }
     },
     "subscribed": { "type": ["boolean", "null"] }
   },
diff --git a/spec/fixtures/api/schemas/public_api/v4/issues.json b/spec/fixtures/api/schemas/public_api/v4/issues.json
index 0ca0a550a7c23..b2839c059d3e9 100644
--- a/spec/fixtures/api/schemas/public_api/v4/issues.json
+++ b/spec/fixtures/api/schemas/public_api/v4/issues.json
@@ -33,6 +33,21 @@
         },
         "additionalProperties": false
       },
+      "assignees": {
+        "type": "array",
+        "items": {
+          "type": ["object", "null"],
+          "properties": {
+            "name": { "type": "string" },
+            "username": { "type": "string" },
+            "id": { "type": "integer" },
+            "state": { "type": "string" },
+            "avatar_url": { "type": "uri" },
+            "web_url": { "type": "uri" }
+          },
+          "additionalProperties": false
+        }
+      },
       "assignee": {
         "type": ["object", "null"],
         "properties": {
@@ -68,7 +83,7 @@
     "required": [
       "id", "iid", "project_id", "title", "description",
       "state", "created_at", "updated_at", "labels",
-      "milestone", "assignee", "author", "user_notes_count",
+      "milestone", "assignees", "author", "user_notes_count",
       "upvotes", "downvotes", "due_date", "confidential",
       "web_url", "weight"
     ],
diff --git a/spec/helpers/issuables_helper_spec.rb b/spec/helpers/issuables_helper_spec.rb
index 93bb711f29ab3..c1ecb46aece84 100644
--- a/spec/helpers/issuables_helper_spec.rb
+++ b/spec/helpers/issuables_helper_spec.rb
@@ -4,6 +4,23 @@
   let(:label)  { build_stubbed(:label) }
   let(:label2) { build_stubbed(:label) }
 
+  describe '#users_dropdown_label' do
+    let(:user)  { build_stubbed(:user) }
+    let(:user2)  { build_stubbed(:user) }
+
+    it 'returns unassigned' do
+      expect(users_dropdown_label([])).to eq('Unassigned')
+    end
+
+    it 'returns selected user\'s name' do
+      expect(users_dropdown_label([user])).to eq(user.name)
+    end
+
+    it 'returns selected user\'s name and counter' do
+      expect(users_dropdown_label([user, user2])).to eq("#{user.name} + 1 more")
+    end
+  end
+
   describe '#issuable_labels_tooltip' do
     it 'returns label text' do
       expect(issuable_labels_tooltip([label])).to eq(label.title)
diff --git a/spec/javascripts/autosave_spec.js b/spec/javascripts/autosave_spec.js
new file mode 100644
index 0000000000000..9f9acc392c286
--- /dev/null
+++ b/spec/javascripts/autosave_spec.js
@@ -0,0 +1,134 @@
+import Autosave from '~/autosave';
+import AccessorUtilities from '~/lib/utils/accessor';
+
+describe('Autosave', () => {
+  let autosave;
+
+  describe('class constructor', () => {
+    const key = 'key';
+    const field = jasmine.createSpyObj('field', ['data', 'on']);
+
+    beforeEach(() => {
+      spyOn(AccessorUtilities, 'isLocalStorageAccessSafe').and.returnValue(true);
+      spyOn(Autosave.prototype, 'restore');
+
+      autosave = new Autosave(field, key);
+    });
+
+    it('should set .isLocalStorageAvailable', () => {
+      expect(AccessorUtilities.isLocalStorageAccessSafe).toHaveBeenCalled();
+      expect(autosave.isLocalStorageAvailable).toBe(true);
+    });
+  });
+
+  describe('restore', () => {
+    const key = 'key';
+    const field = jasmine.createSpyObj('field', ['trigger']);
+
+    beforeEach(() => {
+      autosave = {
+        field,
+        key,
+      };
+
+      spyOn(window.localStorage, 'getItem');
+    });
+
+    describe('if .isLocalStorageAvailable is `false`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = false;
+
+        Autosave.prototype.restore.call(autosave);
+      });
+
+      it('should not call .getItem', () => {
+        expect(window.localStorage.getItem).not.toHaveBeenCalled();
+      });
+    });
+
+    describe('if .isLocalStorageAvailable is `true`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = true;
+
+        Autosave.prototype.restore.call(autosave);
+      });
+
+      it('should call .getItem', () => {
+        expect(window.localStorage.getItem).toHaveBeenCalledWith(key);
+      });
+    });
+  });
+
+  describe('save', () => {
+    const field = jasmine.createSpyObj('field', ['val']);
+
+    beforeEach(() => {
+      autosave = jasmine.createSpyObj('autosave', ['reset']);
+      autosave.field = field;
+
+      field.val.and.returnValue('value');
+
+      spyOn(window.localStorage, 'setItem');
+    });
+
+    describe('if .isLocalStorageAvailable is `false`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = false;
+
+        Autosave.prototype.save.call(autosave);
+      });
+
+      it('should not call .setItem', () => {
+        expect(window.localStorage.setItem).not.toHaveBeenCalled();
+      });
+    });
+
+    describe('if .isLocalStorageAvailable is `true`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = true;
+
+        Autosave.prototype.save.call(autosave);
+      });
+
+      it('should call .setItem', () => {
+        expect(window.localStorage.setItem).toHaveBeenCalled();
+      });
+    });
+  });
+
+  describe('reset', () => {
+    const key = 'key';
+
+    beforeEach(() => {
+      autosave = {
+        key,
+      };
+
+      spyOn(window.localStorage, 'removeItem');
+    });
+
+    describe('if .isLocalStorageAvailable is `false`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = false;
+
+        Autosave.prototype.reset.call(autosave);
+      });
+
+      it('should not call .removeItem', () => {
+        expect(window.localStorage.removeItem).not.toHaveBeenCalled();
+      });
+    });
+
+    describe('if .isLocalStorageAvailable is `true`', () => {
+      beforeEach(() => {
+        autosave.isLocalStorageAvailable = true;
+
+        Autosave.prototype.reset.call(autosave);
+      });
+
+      it('should call .removeItem', () => {
+        expect(window.localStorage.removeItem).toHaveBeenCalledWith(key);
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/behaviors/gl_emoji/unicode_support_map_spec.js b/spec/javascripts/behaviors/gl_emoji/unicode_support_map_spec.js
new file mode 100644
index 0000000000000..1ed96a6747813
--- /dev/null
+++ b/spec/javascripts/behaviors/gl_emoji/unicode_support_map_spec.js
@@ -0,0 +1,47 @@
+import { getUnicodeSupportMap } from '~/behaviors/gl_emoji/unicode_support_map';
+import AccessorUtilities from '~/lib/utils/accessor';
+
+describe('Unicode Support Map', () => {
+  describe('getUnicodeSupportMap', () => {
+    const stringSupportMap = 'stringSupportMap';
+
+    beforeEach(() => {
+      spyOn(AccessorUtilities, 'isLocalStorageAccessSafe');
+      spyOn(window.localStorage, 'getItem');
+      spyOn(window.localStorage, 'setItem');
+      spyOn(JSON, 'parse');
+      spyOn(JSON, 'stringify').and.returnValue(stringSupportMap);
+    });
+
+    describe('if isLocalStorageAvailable is `true`', function () {
+      beforeEach(() => {
+        AccessorUtilities.isLocalStorageAccessSafe.and.returnValue(true);
+
+        getUnicodeSupportMap();
+      });
+
+      it('should call .getItem and .setItem', () => {
+        const allArgs = window.localStorage.setItem.calls.allArgs();
+
+        expect(window.localStorage.getItem).toHaveBeenCalledWith('gl-emoji-user-agent');
+        expect(allArgs[0][0]).toBe('gl-emoji-user-agent');
+        expect(allArgs[0][1]).toBe(navigator.userAgent);
+        expect(allArgs[1][0]).toBe('gl-emoji-unicode-support-map');
+        expect(allArgs[1][1]).toBe(stringSupportMap);
+      });
+    });
+
+    describe('if isLocalStorageAvailable is `false`', function () {
+      beforeEach(() => {
+        AccessorUtilities.isLocalStorageAccessSafe.and.returnValue(false);
+
+        getUnicodeSupportMap();
+      });
+
+      it('should not call .getItem or .setItem', () => {
+        expect(window.localStorage.getItem.calls.count()).toBe(1);
+        expect(window.localStorage.setItem).not.toHaveBeenCalled();
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js b/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
new file mode 100644
index 0000000000000..85816ee1f1161
--- /dev/null
+++ b/spec/javascripts/blob/balsamiq/balsamiq_viewer_spec.js
@@ -0,0 +1,342 @@
+import sqljs from 'sql.js';
+import BalsamiqViewer from '~/blob/balsamiq/balsamiq_viewer';
+import ClassSpecHelper from '../../helpers/class_spec_helper';
+
+describe('BalsamiqViewer', () => {
+  let balsamiqViewer;
+  let endpoint;
+  let viewer;
+
+  describe('class constructor', () => {
+    beforeEach(() => {
+      endpoint = 'endpoint';
+      viewer = {
+        dataset: {
+          endpoint,
+        },
+      };
+
+      balsamiqViewer = new BalsamiqViewer(viewer);
+    });
+
+    it('should set .viewer', () => {
+      expect(balsamiqViewer.viewer).toBe(viewer);
+    });
+
+    it('should set .endpoint', () => {
+      expect(balsamiqViewer.endpoint).toBe(endpoint);
+    });
+  });
+
+  describe('loadFile', () => {
+    let xhr;
+
+    beforeEach(() => {
+      endpoint = 'endpoint';
+      xhr = jasmine.createSpyObj('xhr', ['open', 'send']);
+
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['renderFile']);
+      balsamiqViewer.endpoint = endpoint;
+
+      spyOn(window, 'XMLHttpRequest').and.returnValue(xhr);
+
+      BalsamiqViewer.prototype.loadFile.call(balsamiqViewer);
+    });
+
+    it('should call .open', () => {
+      expect(xhr.open).toHaveBeenCalledWith('GET', endpoint, true);
+    });
+
+    it('should set .responseType', () => {
+      expect(xhr.responseType).toBe('arraybuffer');
+    });
+
+    it('should call .send', () => {
+      expect(xhr.send).toHaveBeenCalled();
+    });
+  });
+
+  describe('renderFile', () => {
+    let container;
+    let loadEvent;
+    let previews;
+
+    beforeEach(() => {
+      loadEvent = { target: { response: {} } };
+      viewer = jasmine.createSpyObj('viewer', ['appendChild']);
+      previews = [document.createElement('ul'), document.createElement('ul')];
+
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['initDatabase', 'getPreviews', 'renderPreview']);
+      balsamiqViewer.viewer = viewer;
+
+      balsamiqViewer.getPreviews.and.returnValue(previews);
+      balsamiqViewer.renderPreview.and.callFake(preview => preview);
+      viewer.appendChild.and.callFake((containerElement) => {
+        container = containerElement;
+      });
+
+      BalsamiqViewer.prototype.renderFile.call(balsamiqViewer, loadEvent);
+    });
+
+    it('should call .initDatabase', () => {
+      expect(balsamiqViewer.initDatabase).toHaveBeenCalledWith(loadEvent.target.response);
+    });
+
+    it('should call .getPreviews', () => {
+      expect(balsamiqViewer.getPreviews).toHaveBeenCalled();
+    });
+
+    it('should call .renderPreview for each preview', () => {
+      const allArgs = balsamiqViewer.renderPreview.calls.allArgs();
+
+      expect(allArgs.length).toBe(2);
+
+      previews.forEach((preview, i) => {
+        expect(allArgs[i][0]).toBe(preview);
+      });
+    });
+
+    it('should set the container HTML', () => {
+      expect(container.innerHTML).toBe('<ul></ul><ul></ul>');
+    });
+
+    it('should add inline preview classes', () => {
+      expect(container.classList[0]).toBe('list-inline');
+      expect(container.classList[1]).toBe('previews');
+    });
+
+    it('should call viewer.appendChild', () => {
+      expect(viewer.appendChild).toHaveBeenCalledWith(container);
+    });
+  });
+
+  describe('initDatabase', () => {
+    let database;
+    let uint8Array;
+    let data;
+
+    beforeEach(() => {
+      uint8Array = {};
+      database = {};
+      data = 'data';
+
+      balsamiqViewer = {};
+
+      spyOn(window, 'Uint8Array').and.returnValue(uint8Array);
+      spyOn(sqljs, 'Database').and.returnValue(database);
+
+      BalsamiqViewer.prototype.initDatabase.call(balsamiqViewer, data);
+    });
+
+    it('should instantiate Uint8Array', () => {
+      expect(window.Uint8Array).toHaveBeenCalledWith(data);
+    });
+
+    it('should call sqljs.Database', () => {
+      expect(sqljs.Database).toHaveBeenCalledWith(uint8Array);
+    });
+
+    it('should set .database', () => {
+      expect(balsamiqViewer.database).toBe(database);
+    });
+  });
+
+  describe('getPreviews', () => {
+    let database;
+    let thumbnails;
+    let getPreviews;
+
+    beforeEach(() => {
+      database = jasmine.createSpyObj('database', ['exec']);
+      thumbnails = [{ values: [0, 1, 2] }];
+
+      balsamiqViewer = {
+        database,
+      };
+
+      spyOn(BalsamiqViewer, 'parsePreview').and.callFake(preview => preview.toString());
+      database.exec.and.returnValue(thumbnails);
+
+      getPreviews = BalsamiqViewer.prototype.getPreviews.call(balsamiqViewer);
+    });
+
+    it('should call database.exec', () => {
+      expect(database.exec).toHaveBeenCalledWith('SELECT * FROM thumbnails');
+    });
+
+    it('should call .parsePreview for each value', () => {
+      const allArgs = BalsamiqViewer.parsePreview.calls.allArgs();
+
+      expect(allArgs.length).toBe(3);
+
+      thumbnails[0].values.forEach((value, i) => {
+        expect(allArgs[i][0]).toBe(value);
+      });
+    });
+
+    it('should return an array of parsed values', () => {
+      expect(getPreviews).toEqual(['0', '1', '2']);
+    });
+  });
+
+  describe('getResource', () => {
+    let database;
+    let resourceID;
+    let resource;
+    let getResource;
+
+    beforeEach(() => {
+      database = jasmine.createSpyObj('database', ['exec']);
+      resourceID = 4;
+      resource = ['resource'];
+
+      balsamiqViewer = {
+        database,
+      };
+
+      database.exec.and.returnValue(resource);
+
+      getResource = BalsamiqViewer.prototype.getResource.call(balsamiqViewer, resourceID);
+    });
+
+    it('should call database.exec', () => {
+      expect(database.exec).toHaveBeenCalledWith(`SELECT * FROM resources WHERE id = '${resourceID}'`);
+    });
+
+    it('should return the selected resource', () => {
+      expect(getResource).toBe(resource[0]);
+    });
+  });
+
+  describe('renderPreview', () => {
+    let previewElement;
+    let innerHTML;
+    let preview;
+    let renderPreview;
+
+    beforeEach(() => {
+      innerHTML = '<a>innerHTML</a>';
+      previewElement = {
+        outerHTML: '<p>outerHTML</p>',
+        classList: jasmine.createSpyObj('classList', ['add']),
+      };
+      preview = {};
+
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['renderTemplate']);
+
+      spyOn(document, 'createElement').and.returnValue(previewElement);
+      balsamiqViewer.renderTemplate.and.returnValue(innerHTML);
+
+      renderPreview = BalsamiqViewer.prototype.renderPreview.call(balsamiqViewer, preview);
+    });
+
+    it('should call classList.add', () => {
+      expect(previewElement.classList.add).toHaveBeenCalledWith('preview');
+    });
+
+    it('should call .renderTemplate', () => {
+      expect(balsamiqViewer.renderTemplate).toHaveBeenCalledWith(preview);
+    });
+
+    it('should set .innerHTML', () => {
+      expect(previewElement.innerHTML).toBe(innerHTML);
+    });
+
+    it('should return element', () => {
+      expect(renderPreview).toBe(previewElement);
+    });
+  });
+
+  describe('renderTemplate', () => {
+    let preview;
+    let name;
+    let resource;
+    let template;
+    let renderTemplate;
+
+    beforeEach(() => {
+      preview = { resourceID: 1, image: 'image' };
+      name = 'name';
+      resource = 'resource';
+      template = `
+        <div class="panel panel-default">
+          <div class="panel-heading">name</div>
+          <div class="panel-body">
+            <img class="img-thumbnail" src=""/>
+          </div>
+        </div>
+      `;
+
+      balsamiqViewer = jasmine.createSpyObj('balsamiqViewer', ['getResource']);
+
+      spyOn(BalsamiqViewer, 'parseTitle').and.returnValue(name);
+      balsamiqViewer.getResource.and.returnValue(resource);
+
+      renderTemplate = BalsamiqViewer.prototype.renderTemplate.call(balsamiqViewer, preview);
+    });
+
+    it('should call .getResource', () => {
+      expect(balsamiqViewer.getResource).toHaveBeenCalledWith(preview.resourceID);
+    });
+
+    it('should call .parseTitle', () => {
+      expect(BalsamiqViewer.parseTitle).toHaveBeenCalledWith(resource);
+    });
+
+    it('should return the template string', function () {
+      expect(renderTemplate.replace(/\s/g, '')).toEqual(template.replace(/\s/g, ''));
+    });
+  });
+
+  describe('parsePreview', () => {
+    let preview;
+    let parsePreview;
+
+    beforeEach(() => {
+      preview = ['{}', '{ "id": 1 }'];
+
+      spyOn(JSON, 'parse').and.callThrough();
+
+      parsePreview = BalsamiqViewer.parsePreview(preview);
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'parsePreview');
+
+    it('should return the parsed JSON', () => {
+      expect(parsePreview).toEqual(JSON.parse('{ "id": 1 }'));
+    });
+  });
+
+  describe('parseTitle', () => {
+    let title;
+    let parseTitle;
+
+    beforeEach(() => {
+      title = { values: [['{}', '{}', '{"name":"name"}']] };
+
+      spyOn(JSON, 'parse').and.callThrough();
+
+      parseTitle = BalsamiqViewer.parseTitle(title);
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'parsePreview');
+
+    it('should return the name value', () => {
+      expect(parseTitle).toBe('name');
+    });
+  });
+
+  describe('onError', () => {
+    beforeEach(() => {
+      spyOn(window, 'Flash');
+
+      BalsamiqViewer.onError();
+    });
+
+    ClassSpecHelper.itShouldBeAStaticMethod(BalsamiqViewer, 'onError');
+
+    it('should instantiate Flash', () => {
+      expect(window.Flash).toHaveBeenCalledWith('Balsamiq file could not be loaded.');
+    });
+  });
+});
diff --git a/spec/javascripts/boards/board_card_spec.js b/spec/javascripts/boards/board_card_spec.js
index de072e7e470bf..376e706d1db82 100644
--- a/spec/javascripts/boards/board_card_spec.js
+++ b/spec/javascripts/boards/board_card_spec.js
@@ -1,12 +1,12 @@
 /* global List */
-/* global ListUser */
+/* global ListAssignee */
 /* global ListLabel */
 /* global listObj */
 /* global boardsMockInterceptor */
 /* global BoardService */
 
 import Vue from 'vue';
-import '~/boards/models/user';
+import '~/boards/models/assignee';
 
 require('~/boards/models/list');
 require('~/boards/models/label');
@@ -133,12 +133,12 @@ describe('Issue card', () => {
     });
 
     it('does not set detail issue if img is clicked', (done) => {
-      vm.issue.assignee = new ListUser({
+      vm.issue.assignees = [new ListAssignee({
         id: 1,
         name: 'testing 123',
         username: 'test',
         avatar: 'test_image',
-      });
+      })];
 
       Vue.nextTick(() => {
         triggerEvent('mouseup', vm.$el.querySelector('img'));
diff --git a/spec/javascripts/boards/board_list_spec.js b/spec/javascripts/boards/board_list_spec.js
index 3f598887603af..a89be91166700 100644
--- a/spec/javascripts/boards/board_list_spec.js
+++ b/spec/javascripts/boards/board_list_spec.js
@@ -35,6 +35,7 @@ describe('Board list component', () => {
       iid: 1,
       confidential: false,
       labels: [],
+      assignees: [],
     });
     list.issuesSize = 1;
     list.issues.push(issue);
diff --git a/spec/javascripts/boards/boards_store_spec.js b/spec/javascripts/boards/boards_store_spec.js
index b55ff2f473a23..5ea160b7790c4 100644
--- a/spec/javascripts/boards/boards_store_spec.js
+++ b/spec/javascripts/boards/boards_store_spec.js
@@ -8,14 +8,14 @@
 import Vue from 'vue';
 import Cookies from 'js-cookie';
 
-require('~/lib/utils/url_utility');
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/services/board_service');
-require('~/boards/stores/boards_store');
-require('./mock_data');
+import '~/lib/utils/url_utility';
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/services/board_service';
+import '~/boards/stores/boards_store';
+import './mock_data';
 
 describe('Store', () => {
   beforeEach(() => {
@@ -212,7 +212,8 @@ describe('Store', () => {
         title: 'Testing',
         iid: 2,
         confidential: false,
-        labels: []
+        labels: [],
+        assignees: [],
       });
       const list = gl.issueBoards.BoardsStore.addList(listObj);
 
diff --git a/spec/javascripts/boards/issue_card_spec.js b/spec/javascripts/boards/issue_card_spec.js
index 1a5e9e9fd0794..fddde799d01ac 100644
--- a/spec/javascripts/boards/issue_card_spec.js
+++ b/spec/javascripts/boards/issue_card_spec.js
@@ -1,20 +1,20 @@
-/* global ListUser */
+/* global ListAssignee */
 /* global ListLabel */
 /* global listObj */
 /* global ListIssue */
 
 import Vue from 'vue';
 
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/stores/boards_store');
-require('~/boards/components/issue_card_inner');
-require('./mock_data');
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/stores/boards_store';
+import '~/boards/components/issue_card_inner';
+import './mock_data';
 
 describe('Issue card component', () => {
-  const user = new ListUser({
+  const user = new ListAssignee({
     id: 1,
     name: 'testing 123',
     username: 'test',
@@ -40,6 +40,7 @@ describe('Issue card component', () => {
       iid: 1,
       confidential: false,
       labels: [list.label],
+      assignees: [],
     });
 
     component = new Vue({
@@ -92,12 +93,12 @@ describe('Issue card component', () => {
   it('renders confidential icon', (done) => {
     component.issue.confidential = true;
 
-    setTimeout(() => {
+    Vue.nextTick(() => {
       expect(
         component.$el.querySelector('.confidential-icon'),
       ).not.toBeNull();
       done();
-    }, 0);
+    });
   });
 
   it('renders issue ID with #', () => {
@@ -109,34 +110,32 @@ describe('Issue card component', () => {
   describe('assignee', () => {
     it('does not render assignee', () => {
       expect(
-        component.$el.querySelector('.card-assignee'),
+        component.$el.querySelector('.card-assignee .avatar'),
       ).toBeNull();
     });
 
     describe('exists', () => {
       beforeEach((done) => {
-        component.issue.assignee = user;
+        component.issue.assignees = [user];
 
-        setTimeout(() => {
-          done();
-        }, 0);
+        Vue.nextTick(() => done());
       });
 
       it('renders assignee', () => {
         expect(
-          component.$el.querySelector('.card-assignee'),
+          component.$el.querySelector('.card-assignee .avatar'),
         ).not.toBeNull();
       });
 
       it('sets title', () => {
         expect(
-          component.$el.querySelector('.card-assignee').getAttribute('title'),
+          component.$el.querySelector('.card-assignee a').getAttribute('title'),
         ).toContain(`Assigned to ${user.name}`);
       });
 
       it('sets users path', () => {
         expect(
-          component.$el.querySelector('.card-assignee').getAttribute('href'),
+          component.$el.querySelector('.card-assignee a').getAttribute('href'),
         ).toBe('/test');
       });
 
@@ -146,6 +145,96 @@ describe('Issue card component', () => {
         ).not.toBeNull();
       });
     });
+
+    describe('assignee default avatar', () => {
+      beforeEach((done) => {
+        component.issue.assignees = [new ListAssignee({
+          id: 1,
+          name: 'testing 123',
+          username: 'test',
+        }, 'default_avatar')];
+
+        Vue.nextTick(done);
+      });
+
+      it('displays defaults avatar if users avatar is null', () => {
+        expect(
+          component.$el.querySelector('.card-assignee img'),
+        ).not.toBeNull();
+        expect(
+          component.$el.querySelector('.card-assignee img').getAttribute('src'),
+        ).toBe('default_avatar');
+      });
+    });
+  });
+
+  describe('multiple assignees', () => {
+    beforeEach((done) => {
+      component.issue.assignees = [
+        user,
+        new ListAssignee({
+          id: 2,
+          name: 'user2',
+          username: 'user2',
+          avatar: 'test_image',
+        }),
+        new ListAssignee({
+          id: 3,
+          name: 'user3',
+          username: 'user3',
+          avatar: 'test_image',
+        }),
+        new ListAssignee({
+          id: 4,
+          name: 'user4',
+          username: 'user4',
+          avatar: 'test_image',
+        })];
+
+      Vue.nextTick(() => done());
+    });
+
+    it('renders all four assignees', () => {
+      expect(component.$el.querySelectorAll('.card-assignee .avatar').length).toEqual(4);
+    });
+
+    describe('more than four assignees', () => {
+      beforeEach((done) => {
+        component.issue.assignees.push(new ListAssignee({
+          id: 5,
+          name: 'user5',
+          username: 'user5',
+          avatar: 'test_image',
+        }));
+
+        Vue.nextTick(() => done());
+      });
+
+      it('renders more avatar counter', () => {
+        expect(component.$el.querySelector('.card-assignee .avatar-counter').innerText).toEqual('+2');
+      });
+
+      it('renders three assignees', () => {
+        expect(component.$el.querySelectorAll('.card-assignee .avatar').length).toEqual(3);
+      });
+
+      it('renders 99+ avatar counter', (done) => {
+        for (let i = 5; i < 104; i += 1) {
+          const u = new ListAssignee({
+            id: i,
+            name: 'name',
+            username: 'username',
+            avatar: 'test_image',
+          });
+          component.issue.assignees.push(u);
+        }
+
+        Vue.nextTick(() => {
+          expect(component.$el.querySelector('.card-assignee .avatar-counter').innerText).toEqual('99+');
+          done();
+        });
+      });
+    });
   });
 
   describe('labels', () => {
@@ -159,9 +248,7 @@ describe('Issue card component', () => {
       beforeEach((done) => {
         component.issue.addLabel(label1);
 
-        setTimeout(() => {
-          done();
-        }, 0);
+        Vue.nextTick(() => done());
       });
 
       it('does not render list label', () => {
diff --git a/spec/javascripts/boards/issue_spec.js b/spec/javascripts/boards/issue_spec.js
index c96dfe94a4a5d..cd1497bc5e615 100644
--- a/spec/javascripts/boards/issue_spec.js
+++ b/spec/javascripts/boards/issue_spec.js
@@ -2,14 +2,15 @@
 /* global BoardService */
 /* global ListIssue */
 
-require('~/lib/utils/url_utility');
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/services/board_service');
-require('~/boards/stores/boards_store');
-require('./mock_data');
+import Vue from 'vue';
+import '~/lib/utils/url_utility';
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/services/board_service';
+import '~/boards/stores/boards_store';
+import './mock_data';
 
 describe('Issue model', () => {
   let issue;
@@ -27,7 +28,13 @@ describe('Issue model', () => {
         title: 'test',
         color: 'red',
         description: 'testing'
-      }]
+      }],
+      assignees: [{
+        id: 1,
+        name: 'name',
+        username: 'username',
+        avatar_url: 'http://avatar_url',
+      }],
     });
   });
 
@@ -80,6 +87,33 @@ describe('Issue model', () => {
     expect(issue.labels.length).toBe(0);
   });
 
+  it('adds assignee', () => {
+    issue.addAssignee({
+      id: 2,
+      name: 'Bruce Wayne',
+      username: 'batman',
+      avatar_url: 'http://batman',
+    });
+
+    expect(issue.assignees.length).toBe(2);
+  });
+
+  it('finds assignee', () => {
+    const assignee = issue.findAssignee(issue.assignees[0]);
+    expect(assignee).toBeDefined();
+  });
+
+  it('removes assignee', () => {
+    const assignee = issue.findAssignee(issue.assignees[0]);
+    issue.removeAssignee(assignee);
+    expect(issue.assignees.length).toBe(0);
+  });
+
+  it('removes all assignees', () => {
+    issue.removeAllAssignees();
+    expect(issue.assignees.length).toBe(0);
+  });
+
   it('sets position to infinity if no position is stored', () => {
     expect(issue.position).toBe(Infinity);
   });
@@ -90,9 +124,31 @@ describe('Issue model', () => {
       iid: 1,
       confidential: false,
       relative_position: 1,
-      labels: []
+      labels: [],
+      assignees: [],
     });
 
     expect(relativePositionIssue.position).toBe(1);
   });
+
+  describe('update', () => {
+    it('passes assignee ids when there are assignees', (done) => {
+      spyOn(Vue.http, 'patch').and.callFake((url, data) => {
+        expect(data.issue.assignee_ids).toEqual([1]);
+        done();
+      });
+
+      issue.update('url');
+    });
+
+    it('passes assignee ids of [0] when there are no assignees', (done) => {
+      spyOn(Vue.http, 'patch').and.callFake((url, data) => {
+        expect(data.issue.assignee_ids).toEqual([0]);
+        done();
+      });
+
+      issue.removeAllAssignees();
+      issue.update('url');
+    });
+  });
 });
diff --git a/spec/javascripts/boards/list_spec.js b/spec/javascripts/boards/list_spec.js
index 24a2da9f6b629..8e3d9fd77a073 100644
--- a/spec/javascripts/boards/list_spec.js
+++ b/spec/javascripts/boards/list_spec.js
@@ -8,14 +8,14 @@
 
 import Vue from 'vue';
 
-require('~/lib/utils/url_utility');
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/services/board_service');
-require('~/boards/stores/boards_store');
-require('./mock_data');
+import '~/lib/utils/url_utility';
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/services/board_service';
+import '~/boards/stores/boards_store';
+import './mock_data';
 
 describe('List model', () => {
   let list;
@@ -94,7 +94,8 @@ describe('List model', () => {
       title: 'Testing',
       iid: _.random(10000),
       confidential: false,
-      labels: [list.label, listDup.label]
+      labels: [list.label, listDup.label],
+      assignees: [],
     });
 
     list.issues.push(issue);
@@ -119,7 +120,8 @@ describe('List model', () => {
           title: 'Testing',
           iid: _.random(10000) + i,
           confidential: false,
-          labels: [list.label]
+          labels: [list.label],
+          assignees: [],
         }));
       }
       list.issuesSize = 50;
@@ -137,7 +139,8 @@ describe('List model', () => {
         title: 'Testing',
         iid: _.random(10000),
         confidential: false,
-        labels: [list.label]
+        labels: [list.label],
+        assignees: [],
       }));
       list.issuesSize = 2;
 
diff --git a/spec/javascripts/boards/mock_data.js b/spec/javascripts/boards/mock_data.js
index 00f9a76ccc182..24ace51f12852 100644
--- a/spec/javascripts/boards/mock_data.js
+++ b/spec/javascripts/boards/mock_data.js
@@ -38,7 +38,8 @@ const BoardsMockData = {
         title: 'Testing',
         iid: 1,
         confidential: false,
-        labels: []
+        labels: [],
+        assignees: [],
       }],
       size: 1
     },
diff --git a/spec/javascripts/boards/modal_store_spec.js b/spec/javascripts/boards/modal_store_spec.js
index 80db816aff852..32e6d04df9fed 100644
--- a/spec/javascripts/boards/modal_store_spec.js
+++ b/spec/javascripts/boards/modal_store_spec.js
@@ -1,10 +1,10 @@
 /* global ListIssue */
 
-require('~/boards/models/issue');
-require('~/boards/models/label');
-require('~/boards/models/list');
-require('~/boards/models/user');
-require('~/boards/stores/modal_store');
+import '~/boards/models/issue';
+import '~/boards/models/label';
+import '~/boards/models/list';
+import '~/boards/models/assignee';
+import '~/boards/stores/modal_store';
 
 describe('Modal store', () => {
   let issue;
@@ -21,12 +21,14 @@ describe('Modal store', () => {
       iid: 1,
       confidential: false,
       labels: [],
+      assignees: [],
     });
     issue2 = new ListIssue({
       title: 'Testing',
       iid: 2,
       confidential: false,
       labels: [],
+      assignees: [],
     });
     Store.store.issues.push(issue);
     Store.store.issues.push(issue2);
diff --git a/spec/javascripts/cycle_analytics/limit_warning_component_spec.js b/spec/javascripts/cycle_analytics/limit_warning_component_spec.js
index 50000c5a5f560..2fb9eb0ca8561 100644
--- a/spec/javascripts/cycle_analytics/limit_warning_component_spec.js
+++ b/spec/javascripts/cycle_analytics/limit_warning_component_spec.js
@@ -1,6 +1,9 @@
 import Vue from 'vue';
+import Translate from '~/vue_shared/translate';
 import limitWarningComp from '~/cycle_analytics/components/limit_warning_component';
 
+Vue.use(Translate);
+
 describe('Limit warning component', () => {
   let component;
   let LimitWarningComponent;
diff --git a/spec/javascripts/deploy_keys/components/action_btn_spec.js b/spec/javascripts/deploy_keys/components/action_btn_spec.js
new file mode 100644
index 0000000000000..5b93fbc5575d7
--- /dev/null
+++ b/spec/javascripts/deploy_keys/components/action_btn_spec.js
@@ -0,0 +1,70 @@
+import Vue from 'vue';
+import eventHub from '~/deploy_keys/eventhub';
+import actionBtn from '~/deploy_keys/components/action_btn.vue';
+
+describe('Deploy keys action btn', () => {
+  const data = getJSONFixture('deploy_keys/keys.json');
+  const deployKey = data.enabled_keys[0];
+  let vm;
+
+  beforeEach((done) => {
+    const ActionBtnComponent = Vue.extend(actionBtn);
+
+    vm = new ActionBtnComponent({
+      propsData: {
+        deployKey,
+        type: 'enable',
+      },
+    }).$mount();
+
+    setTimeout(done);
+  });
+
+  it('renders the type as uppercase', () => {
+    expect(
+      vm.$el.textContent.trim(),
+    ).toBe('Enable');
+  });
+
+  it('sends eventHub event with btn type', (done) => {
+    spyOn(eventHub, '$emit');
+
+    vm.$el.click();
+
+    setTimeout(() => {
+      expect(
+        eventHub.$emit,
+      ).toHaveBeenCalledWith('enable.key', deployKey);
+
+      done();
+    });
+  });
+
+  it('shows loading spinner after click', (done) => {
+    vm.$el.click();
+
+    setTimeout(() => {
+      expect(
+        vm.$el.querySelector('.fa'),
+      ).toBeDefined();
+
+      done();
+    });
+  });
+
+  it('disables button after click', (done) => {
+    vm.$el.click();
+
+    setTimeout(() => {
+      expect(
+        vm.$el.classList.contains('disabled'),
+      ).toBeTruthy();
+
+      expect(
+        vm.$el.getAttribute('disabled'),
+      ).toBe('disabled');
+
+      done();
+    });
+  });
+});
diff --git a/spec/javascripts/deploy_keys/components/app_spec.js b/spec/javascripts/deploy_keys/components/app_spec.js
new file mode 100644
index 0000000000000..700897f50b0ee
--- /dev/null
+++ b/spec/javascripts/deploy_keys/components/app_spec.js
@@ -0,0 +1,142 @@
+import Vue from 'vue';
+import eventHub from '~/deploy_keys/eventhub';
+import deployKeysApp from '~/deploy_keys/components/app.vue';
+
+describe('Deploy keys app component', () => {
+  const data = getJSONFixture('deploy_keys/keys.json');
+  let vm;
+
+  const deployKeysResponse = (request, next) => {
+    next(request.respondWith(JSON.stringify(data), {
+      status: 200,
+    }));
+  };
+
+  beforeEach((done) => {
+    const Component = Vue.extend(deployKeysApp);
+
+    Vue.http.interceptors.push(deployKeysResponse);
+
+    vm = new Component({
+      propsData: {
+        endpoint: '/test',
+      },
+    }).$mount();
+
+    setTimeout(done);
+  });
+
+  afterEach(() => {
+    Vue.http.interceptors = _.without(Vue.http.interceptors, deployKeysResponse);
+  });
+
+  it('renders loading icon', (done) => {
+    vm.store.keys = {};
+    vm.isLoading = false;
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelectorAll('.deploy-keys-panel').length,
+      ).toBe(0);
+
+      expect(
+        vm.$el.querySelector('.fa-spinner'),
+      ).toBeDefined();
+
+      done();
+    });
+  });
+
+  it('renders keys panels', () => {
+    expect(
+      vm.$el.querySelectorAll('.deploy-keys-panel').length,
+    ).toBe(3);
+  });
+
+  it('does not render key panels when keys object is empty', (done) => {
+    vm.store.keys = {};
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelectorAll('.deploy-keys-panel').length,
+      ).toBe(0);
+
+      done();
+    });
+  });
+
+  it('does not render public panel when empty', (done) => {
+    vm.store.keys.public_keys = [];
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelectorAll('.deploy-keys-panel').length,
+      ).toBe(2);
+
+      done();
+    });
+  });
+
+  it('re-fetches deploy keys when enabling a key', (done) => {
+    const key = data.public_keys[0];
+
+    spyOn(vm.service, 'getKeys');
+    spyOn(vm.service, 'enableKey').and.callFake(() => new Promise((resolve) => {
+      resolve();
+
+      setTimeout(() => {
+        expect(vm.service.getKeys).toHaveBeenCalled();
+
+        done();
+      });
+    }));
+
+    eventHub.$emit('enable.key', key);
+
+    expect(vm.service.enableKey).toHaveBeenCalledWith(key.id);
+  });
+
+  it('re-fetches deploy keys when disabling a key', (done) => {
+    const key = data.public_keys[0];
+
+    spyOn(window, 'confirm').and.returnValue(true);
+    spyOn(vm.service, 'getKeys');
+    spyOn(vm.service, 'disableKey').and.callFake(() => new Promise((resolve) => {
+      resolve();
+
+      setTimeout(() => {
+        expect(vm.service.getKeys).toHaveBeenCalled();
+
+        done();
+      });
+    }));
+
+    eventHub.$emit('disable.key', key);
+
+    expect(vm.service.disableKey).toHaveBeenCalledWith(key.id);
+  });
+
+  it('calls disableKey when removing a key', (done) => {
+    const key = data.public_keys[0];
+
+    spyOn(window, 'confirm').and.returnValue(true);
+    spyOn(vm.service, 'getKeys');
+    spyOn(vm.service, 'disableKey').and.callFake(() => new Promise((resolve) => {
+      resolve();
+
+      setTimeout(() => {
+        expect(vm.service.getKeys).toHaveBeenCalled();
+
+        done();
+      });
+    }));
+
+    eventHub.$emit('remove.key', key);
+
+    expect(vm.service.disableKey).toHaveBeenCalledWith(key.id);
+  });
+
+  it('hasKeys returns true when there are keys', () => {
+    expect(vm.hasKeys).toEqual(3);
+  });
+});
diff --git a/spec/javascripts/deploy_keys/components/key_spec.js b/spec/javascripts/deploy_keys/components/key_spec.js
new file mode 100644
index 0000000000000..793ab8c451d85
--- /dev/null
+++ b/spec/javascripts/deploy_keys/components/key_spec.js
@@ -0,0 +1,92 @@
+import Vue from 'vue';
+import DeployKeysStore from '~/deploy_keys/store';
+import key from '~/deploy_keys/components/key.vue';
+
+describe('Deploy keys key', () => {
+  let vm;
+  const KeyComponent = Vue.extend(key);
+  const data = getJSONFixture('deploy_keys/keys.json');
+  const createComponent = (deployKey) => {
+    const store = new DeployKeysStore();
+    store.keys = data;
+
+    vm = new KeyComponent({
+      propsData: {
+        deployKey,
+        store,
+      },
+    }).$mount();
+  };
+
+  describe('enabled key', () => {
+    const deployKey = data.enabled_keys[0];
+
+    beforeEach((done) => {
+      createComponent(deployKey);
+
+      setTimeout(done);
+    });
+
+    it('renders the keys title', () => {
+      expect(
+        vm.$el.querySelector('.title').textContent.trim(),
+      ).toContain('My title');
+    });
+
+    it('renders human friendly formatted created date', () => {
+      expect(
+        vm.$el.querySelector('.key-created-at').textContent.trim(),
+      ).toBe(`created ${gl.utils.getTimeago().format(deployKey.created_at)}`);
+    });
+
+    it('shows remove button', () => {
+      expect(
+        vm.$el.querySelector('.btn').textContent.trim(),
+      ).toBe('Remove');
+    });
+
+    it('shows write access text when key has write access', (done) => {
+      vm.deployKey.can_push = true;
+
+      Vue.nextTick(() => {
+        expect(
+          vm.$el.querySelector('.write-access-allowed'),
+        ).not.toBeNull();
+
+        expect(
+          vm.$el.querySelector('.write-access-allowed').textContent.trim(),
+        ).toBe('Write access allowed');
+
+        done();
+      });
+    });
+  });
+
+  describe('public keys', () => {
+    const deployKey = data.public_keys[0];
+
+    beforeEach((done) => {
+      createComponent(deployKey);
+
+      setTimeout(done);
+    });
+
+    it('shows enable button', () => {
+      expect(
+        vm.$el.querySelector('.btn').textContent.trim(),
+      ).toBe('Enable');
+    });
+
+    it('shows disable button when key is enabled', (done) => {
+      vm.store.keys.enabled_keys.push(deployKey);
+
+      Vue.nextTick(() => {
+        expect(
+          vm.$el.querySelector('.btn').textContent.trim(),
+        ).toBe('Disable');
+
+        done();
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/deploy_keys/components/keys_panel_spec.js b/spec/javascripts/deploy_keys/components/keys_panel_spec.js
new file mode 100644
index 0000000000000..a69b39c35c4f1
--- /dev/null
+++ b/spec/javascripts/deploy_keys/components/keys_panel_spec.js
@@ -0,0 +1,70 @@
+import Vue from 'vue';
+import DeployKeysStore from '~/deploy_keys/store';
+import deployKeysPanel from '~/deploy_keys/components/keys_panel.vue';
+
+describe('Deploy keys panel', () => {
+  const data = getJSONFixture('deploy_keys/keys.json');
+  let vm;
+
+  beforeEach((done) => {
+    const DeployKeysPanelComponent = Vue.extend(deployKeysPanel);
+    const store = new DeployKeysStore();
+    store.keys = data;
+
+    vm = new DeployKeysPanelComponent({
+      propsData: {
+        title: 'test',
+        keys: data.enabled_keys,
+        showHelpBox: true,
+        store,
+      },
+    }).$mount();
+
+    setTimeout(done);
+  });
+
+  it('renders the title with keys count', () => {
+    expect(
+      vm.$el.querySelector('h5').textContent.trim(),
+    ).toContain('test');
+
+    expect(
+      vm.$el.querySelector('h5').textContent.trim(),
+    ).toContain(`(${vm.keys.length})`);
+  });
+
+  it('renders list of keys', () => {
+    expect(
+      vm.$el.querySelectorAll('li').length,
+    ).toBe(vm.keys.length);
+  });
+
+  it('renders help box if keys are empty', (done) => {
+    vm.keys = [];
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelector('.settings-message'),
+      ).toBeDefined();
+
+      expect(
+        vm.$el.querySelector('.settings-message').textContent.trim(),
+      ).toBe('No deploy keys found. Create one with the form above.');
+
+      done();
+    });
+  });
+
+  it('does not render help box if keys are empty & showHelpBox is false', (done) => {
+    vm.keys = [];
+    vm.showHelpBox = false;
+
+    Vue.nextTick(() => {
+      expect(
+        vm.$el.querySelector('.settings-message'),
+      ).toBeNull();
+
+      done();
+    });
+  });
+});
diff --git a/spec/javascripts/droplab/constants_spec.js b/spec/javascripts/droplab/constants_spec.js
index fd153a49fcdbb..b9d28db74cc5b 100644
--- a/spec/javascripts/droplab/constants_spec.js
+++ b/spec/javascripts/droplab/constants_spec.js
@@ -27,6 +27,12 @@ describe('constants', function () {
     });
   });
 
+  describe('TEMPLATE_REGEX', function () {
+    it('should be a handlebars templating syntax regex', function() {
+      expect(constants.TEMPLATE_REGEX).toEqual(/\{\{(.+?)\}\}/g);
+    });
+  });
+
   describe('IGNORE_CLASS', function () {
     it('should be `droplab-item-ignore`', function() {
       expect(constants.IGNORE_CLASS).toBe('droplab-item-ignore');
diff --git a/spec/javascripts/droplab/drop_down_spec.js b/spec/javascripts/droplab/drop_down_spec.js
index 7516b30191782..e7786e8cc2ce1 100644
--- a/spec/javascripts/droplab/drop_down_spec.js
+++ b/spec/javascripts/droplab/drop_down_spec.js
@@ -451,7 +451,7 @@ describe('DropDown', function () {
       this.html = 'html';
       this.template = { firstChild: { outerHTML: 'outerHTML', style: {} } };
 
-      spyOn(utils, 't').and.returnValue(this.html);
+      spyOn(utils, 'template').and.returnValue(this.html);
       spyOn(document, 'createElement').and.returnValue(this.template);
       spyOn(this.dropdown, 'setImagesSrc');
 
@@ -459,7 +459,7 @@ describe('DropDown', function () {
     });
 
     it('should call utils.t with .templateString and data', function () {
-      expect(utils.t).toHaveBeenCalledWith(this.templateString, this.data);
+      expect(utils.template).toHaveBeenCalledWith(this.templateString, this.data);
     });
 
     it('should call document.createElement', function () {
diff --git a/spec/javascripts/filtered_search/components/recent_searches_dropdown_content_spec.js b/spec/javascripts/filtered_search/components/recent_searches_dropdown_content_spec.js
index 2722882375f44..d0f09a561d5ab 100644
--- a/spec/javascripts/filtered_search/components/recent_searches_dropdown_content_spec.js
+++ b/spec/javascripts/filtered_search/components/recent_searches_dropdown_content_spec.js
@@ -76,6 +76,26 @@ describe('RecentSearchesDropdownContent', () => {
     });
   });
 
+  describe('if isLocalStorageAvailable is `false`', () => {
+    let el;
+
+    beforeEach(() => {
+      const props = Object.assign({ isLocalStorageAvailable: false }, propsDataWithItems);
+
+      vm = createComponent(props);
+      el = vm.$el;
+    });
+
+    it('should render an info note', () => {
+      const note = el.querySelector('.dropdown-info-note');
+      const items = el.querySelectorAll('.filtered-search-history-dropdown-item');
+
+      expect(note).toBeDefined();
+      expect(note.innerText.trim()).toBe('This feature requires local storage to be enabled');
+      expect(items.length).toEqual(propsDataWithoutItems.items.length);
+    });
+  });
+
   describe('computed', () => {
     describe('processedItems', () => {
       it('with items', () => {
diff --git a/spec/javascripts/filtered_search/filtered_search_manager_spec.js b/spec/javascripts/filtered_search/filtered_search_manager_spec.js
index e747aa497c20b..063d547d00c93 100644
--- a/spec/javascripts/filtered_search/filtered_search_manager_spec.js
+++ b/spec/javascripts/filtered_search/filtered_search_manager_spec.js
@@ -1,3 +1,7 @@
+import * as recentSearchesStoreSrc from '~/filtered_search/stores/recent_searches_store';
+import RecentSearchesService from '~/filtered_search/services/recent_searches_service';
+import RecentSearchesServiceError from '~/filtered_search/services/recent_searches_service_error';
+
 require('~/lib/utils/url_utility');
 require('~/lib/utils/common_utils');
 require('~/filtered_search/filtered_search_token_keys');
@@ -60,6 +64,36 @@ describe('Filtered Search Manager', () => {
     manager.cleanup();
   });
 
+  describe('class constructor', () => {
+    const isLocalStorageAvailable = 'isLocalStorageAvailable';
+    let filteredSearchManager;
+
+    beforeEach(() => {
+      spyOn(RecentSearchesService, 'isAvailable').and.returnValue(isLocalStorageAvailable);
+      spyOn(recentSearchesStoreSrc, 'default');
+
+      filteredSearchManager = new gl.FilteredSearchManager();
+
+      return filteredSearchManager;
+    });
+
+    it('should instantiate RecentSearchesStore with isLocalStorageAvailable', () => {
+      expect(RecentSearchesService.isAvailable).toHaveBeenCalled();
+      expect(recentSearchesStoreSrc.default).toHaveBeenCalledWith({
+        isLocalStorageAvailable,
+      });
+    });
+
+    it('should not instantiate Flash if an RecentSearchesServiceError is caught', () => {
+      spyOn(RecentSearchesService.prototype, 'fetch').and.callFake(() => Promise.reject(new RecentSearchesServiceError()));
+      spyOn(window, 'Flash');
+
+      filteredSearchManager = new gl.FilteredSearchManager();
+
+      expect(window.Flash).not.toHaveBeenCalled();
+    });
+  });
+
   describe('search', () => {
     const defaultParams = '?scope=all&utf8=%E2%9C%93&state=opened';
 
diff --git a/spec/javascripts/filtered_search/filtered_search_visual_tokens_spec.js b/spec/javascripts/filtered_search/filtered_search_visual_tokens_spec.js
index d75b90612815e..8b750561eb7b5 100644
--- a/spec/javascripts/filtered_search/filtered_search_visual_tokens_spec.js
+++ b/spec/javascripts/filtered_search/filtered_search_visual_tokens_spec.js
@@ -1,3 +1,5 @@
+import AjaxCache from '~/lib/utils/ajax_cache';
+
 require('~/filtered_search/filtered_search_visual_tokens');
 const FilteredSearchSpecHelper = require('../helpers/filtered_search_spec_helper');
 
@@ -611,4 +613,103 @@ describe('Filtered Search Visual Tokens', () => {
       expect(token.querySelector('.value').innerText).toEqual('~bug');
     });
   });
+
+  describe('renderVisualTokenValue', () => {
+    let searchTokens;
+
+    beforeEach(() => {
+      tokensContainer.innerHTML = FilteredSearchSpecHelper.createTokensContainerHTML(`
+        ${FilteredSearchSpecHelper.createFilterVisualTokenHTML('label', 'none')}
+        ${FilteredSearchSpecHelper.createSearchVisualTokenHTML('search')}
+        ${FilteredSearchSpecHelper.createFilterVisualTokenHTML('milestone', 'upcoming')}
+      `);
+
+      searchTokens = document.querySelectorAll('.filtered-search-token');
+    });
+
+    it('renders a token value element', () => {
+      spyOn(gl.FilteredSearchVisualTokens, 'updateLabelTokenColor');
+      const updateLabelTokenColorSpy = gl.FilteredSearchVisualTokens.updateLabelTokenColor;
+
+      expect(searchTokens.length).toBe(2);
+      Array.prototype.forEach.call(searchTokens, (token) => {
+        updateLabelTokenColorSpy.calls.reset();
+
+        const tokenName = token.querySelector('.name').innerText;
+        const tokenValue = 'new value';
+        gl.FilteredSearchVisualTokens.renderVisualTokenValue(token, tokenName, tokenValue);
+
+        const tokenValueElement = token.querySelector('.value');
+        expect(tokenValueElement.innerText).toBe(tokenValue);
+
+        if (tokenName.toLowerCase() === 'label') {
+          const tokenValueContainer = token.querySelector('.value-container');
+          expect(updateLabelTokenColorSpy.calls.count()).toBe(1);
+          const expectedArgs = [tokenValueContainer, tokenValue];
+          expect(updateLabelTokenColorSpy.calls.argsFor(0)).toEqual(expectedArgs);
+        } else {
+          expect(updateLabelTokenColorSpy.calls.count()).toBe(0);
+        }
+      });
+    });
+  });
+
+  describe('updateLabelTokenColor', () => {
+    const jsonFixtureName = 'labels/project_labels.json';
+    const dummyEndpoint = '/dummy/endpoint';
+
+    preloadFixtures(jsonFixtureName);
+    const labelData = getJSONFixture(jsonFixtureName);
+    const findLabel = tokenValue => labelData.find(
+      label => tokenValue === `~${gl.DropdownUtils.getEscapedText(label.title)}`,
+    );
+
+    const bugLabelToken = FilteredSearchSpecHelper.createFilterVisualToken('label', '~bug');
+    const missingLabelToken = FilteredSearchSpecHelper.createFilterVisualToken('label', '~doesnotexist');
+    const spaceLabelToken = FilteredSearchSpecHelper.createFilterVisualToken('label', '~"some space"');
+
+    const parseColor = (color) => {
+      const dummyElement = document.createElement('div');
+      dummyElement.style.color = color;
+      return dummyElement.style.color;
+    };
+
+    beforeEach(() => {
+      tokensContainer.innerHTML = FilteredSearchSpecHelper.createTokensContainerHTML(`
+        ${bugLabelToken.outerHTML}
+        ${missingLabelToken.outerHTML}
+        ${spaceLabelToken.outerHTML}
+      `);
+
+      const filteredSearchInput = document.querySelector('.filtered-search');
+      filteredSearchInput.dataset.baseEndpoint = dummyEndpoint;
+
+      AjaxCache.internalStorage = { };
+      AjaxCache.internalStorage[`${dummyEndpoint}/labels.json`] = labelData;
+    });
+
+    const testCase = (token, done) => {
+      const tokenValueContainer = token.querySelector('.value-container');
+      const tokenValue = token.querySelector('.value').innerText;
+      const label = findLabel(tokenValue);
+
+      gl.FilteredSearchVisualTokens.updateLabelTokenColor(tokenValueContainer, tokenValue)
+      .then(() => {
+        if (label) {
+          expect(tokenValueContainer.getAttribute('style')).not.toBe(null);
+          expect(tokenValueContainer.style.backgroundColor).toBe(parseColor(label.color));
+          expect(tokenValueContainer.style.color).toBe(parseColor(label.text_color));
+        } else {
+          expect(token).toBe(missingLabelToken);
+          expect(tokenValueContainer.getAttribute('style')).toBe(null);
+        }
+      })
+      .then(done)
+      .catch(fail);
+    };
+
+    it('updates the color of a label token', done => testCase(bugLabelToken, done));
+    it('updates the color of a label token with spaces', done => testCase(spaceLabelToken, done));
+    it('does not change color of a missing label', done => testCase(missingLabelToken, done));
+  });
 });
diff --git a/spec/javascripts/filtered_search/recent_searches_root_spec.js b/spec/javascripts/filtered_search/recent_searches_root_spec.js
new file mode 100644
index 0000000000000..d8ba6de5f45d9
--- /dev/null
+++ b/spec/javascripts/filtered_search/recent_searches_root_spec.js
@@ -0,0 +1,31 @@
+import RecentSearchesRoot from '~/filtered_search/recent_searches_root';
+import * as vueSrc from 'vue';
+
+describe('RecentSearchesRoot', () => {
+  describe('render', () => {
+    let recentSearchesRoot;
+    let data;
+    let template;
+
+    beforeEach(() => {
+      recentSearchesRoot = {
+        store: {
+          state: 'state',
+        },
+      };
+
+      spyOn(vueSrc, 'default').and.callFake((options) => {
+        data = options.data;
+        template = options.template;
+      });
+
+      RecentSearchesRoot.prototype.render.call(recentSearchesRoot);
+    });
+
+    it('should instantiate Vue', () => {
+      expect(vueSrc.default).toHaveBeenCalled();
+      expect(data()).toBe(recentSearchesRoot.store.state);
+      expect(template).toContain(':is-local-storage-available="isLocalStorageAvailable"');
+    });
+  });
+});
diff --git a/spec/javascripts/filtered_search/services/recent_searches_service_error_spec.js b/spec/javascripts/filtered_search/services/recent_searches_service_error_spec.js
new file mode 100644
index 0000000000000..ea7c146fa4f01
--- /dev/null
+++ b/spec/javascripts/filtered_search/services/recent_searches_service_error_spec.js
@@ -0,0 +1,18 @@
+import RecentSearchesServiceError from '~/filtered_search/services/recent_searches_service_error';
+
+describe('RecentSearchesServiceError', () => {
+  let recentSearchesServiceError;
+
+  beforeEach(() => {
+    recentSearchesServiceError = new RecentSearchesServiceError();
+  });
+
+  it('instantiates an instance of RecentSearchesServiceError and not an Error', () => {
+    expect(recentSearchesServiceError).toEqual(jasmine.any(RecentSearchesServiceError));
+    expect(recentSearchesServiceError.name).toBe('RecentSearchesServiceError');
+  });
+
+  it('should set a default message', () => {
+    expect(recentSearchesServiceError.message).toBe('Recent Searches Service is unavailable');
+  });
+});
diff --git a/spec/javascripts/filtered_search/services/recent_searches_service_spec.js b/spec/javascripts/filtered_search/services/recent_searches_service_spec.js
index c255bf7c93996..31fa478804af9 100644
--- a/spec/javascripts/filtered_search/services/recent_searches_service_spec.js
+++ b/spec/javascripts/filtered_search/services/recent_searches_service_spec.js
@@ -1,6 +1,7 @@
 /* eslint-disable promise/catch-or-return */
 
 import RecentSearchesService from '~/filtered_search/services/recent_searches_service';
+import AccessorUtilities from '~/lib/utils/accessor';
 
 describe('RecentSearchesService', () => {
   let service;
@@ -11,6 +12,10 @@ describe('RecentSearchesService', () => {
   });
 
   describe('fetch', () => {
+    beforeEach(() => {
+      spyOn(RecentSearchesService, 'isAvailable').and.returnValue(true);
+    });
+
     it('should default to empty array', (done) => {
       const fetchItemsPromise = service.fetch();
 
@@ -29,11 +34,21 @@ describe('RecentSearchesService', () => {
       const fetchItemsPromise = service.fetch();
 
       fetchItemsPromise
-        .catch(() => {
+        .catch((error) => {
+          expect(error).toEqual(jasmine.any(SyntaxError));
           done();
         });
     });
 
+    it('should reject when service is unavailable', (done) => {
+      RecentSearchesService.isAvailable.and.returnValue(false);
+
+      service.fetch().catch((error) => {
+        expect(error).toEqual(jasmine.any(Error));
+        done();
+      });
+    });
+
     it('should return items from localStorage', (done) => {
       window.localStorage.setItem(service.localStorageKey, '["foo", "bar"]');
       const fetchItemsPromise = service.fetch();
@@ -44,15 +59,89 @@ describe('RecentSearchesService', () => {
           done();
         });
     });
+
+    describe('if .isAvailable returns `false`', () => {
+      beforeEach(() => {
+        RecentSearchesService.isAvailable.and.returnValue(false);
+
+        spyOn(window.localStorage, 'getItem');
+
+        RecentSearchesService.prototype.fetch();
+      });
+
+      it('should not call .getItem', () => {
+        expect(window.localStorage.getItem).not.toHaveBeenCalled();
+      });
+    });
   });
 
   describe('setRecentSearches', () => {
+    beforeEach(() => {
+      spyOn(RecentSearchesService, 'isAvailable').and.returnValue(true);
+    });
+
     it('should save things in localStorage', () => {
       const items = ['foo', 'bar'];
       service.save(items);
-      const newLocalStorageValue =
-        window.localStorage.getItem(service.localStorageKey);
+      const newLocalStorageValue = window.localStorage.getItem(service.localStorageKey);
       expect(JSON.parse(newLocalStorageValue)).toEqual(items);
     });
   });
+
+  describe('save', () => {
+    beforeEach(() => {
+      spyOn(window.localStorage, 'setItem');
+      spyOn(RecentSearchesService, 'isAvailable');
+    });
+
+    describe('if .isAvailable returns `true`', () => {
+      const searchesString = 'searchesString';
+      const localStorageKey = 'localStorageKey';
+      const recentSearchesService = {
+        localStorageKey,
+      };
+
+      beforeEach(() => {
+        RecentSearchesService.isAvailable.and.returnValue(true);
+
+        spyOn(JSON, 'stringify').and.returnValue(searchesString);
+
+        RecentSearchesService.prototype.save.call(recentSearchesService);
+      });
+
+      it('should call .setItem', () => {
+        expect(window.localStorage.setItem).toHaveBeenCalledWith(localStorageKey, searchesString);
+      });
+    });
+
+    describe('if .isAvailable returns `false`', () => {
+      beforeEach(() => {
+        RecentSearchesService.isAvailable.and.returnValue(false);
+
+        RecentSearchesService.prototype.save();
+      });
+
+      it('should not call .setItem', () => {
+        expect(window.localStorage.setItem).not.toHaveBeenCalled();
+      });
+    });
+  });
+
+  describe('isAvailable', () => {
+    let isAvailable;
+
+    beforeEach(() => {
+      spyOn(AccessorUtilities, 'isLocalStorageAccessSafe').and.callThrough();
+
+      isAvailable = RecentSearchesService.isAvailable();
+    });
+
+    it('should call .isLocalStorageAccessSafe', () => {
+      expect(AccessorUtilities.isLocalStorageAccessSafe).toHaveBeenCalled();
+    });
+
+    it('should return a boolean', () => {
+      expect(typeof isAvailable).toBe('boolean');
+    });
+  });
 });
diff --git a/spec/javascripts/fixtures/deploy_keys.rb b/spec/javascripts/fixtures/deploy_keys.rb
new file mode 100644
index 0000000000000..16e598a4b290c
--- /dev/null
+++ b/spec/javascripts/fixtures/deploy_keys.rb
@@ -0,0 +1,36 @@
+require 'spec_helper'
+
+describe Projects::DeployKeysController, '(JavaScript fixtures)', type: :controller do
+  include JavaScriptFixturesHelpers
+
+  let(:admin) { create(:admin) }
+  let(:namespace) { create(:namespace, name: 'frontend-fixtures' )}
+  let(:project) { create(:project_empty_repo, namespace: namespace, path: 'todos-project') }
+  let(:project2) { create(:empty_project, :internal)}
+
+  before(:all) do
+    clean_frontend_fixtures('deploy_keys/')
+  end
+
+  before(:each) do
+    sign_in(admin)
+  end
+
+  render_views
+
+  it 'deploy_keys/keys.json' do |example|
+    create(:deploy_key, public: true)
+    project_key = create(:deploy_key, key: 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQCdMHEHyhRjbhEZVddFn6lTWdgEy5Q6Bz4nwGB76xWZI5YT/1WJOMEW+sL5zYd31kk7sd3FJ5L9ft8zWMWrr/iWXQikC2cqZK24H1xy+ZUmrRuJD4qGAaIVoyyzBL+avL+lF8J5lg6YSw8gwJY/lX64/vnJHUlWw2n5BF8IFOWhiw== dummy@gitlab.com')
+    internal_key = create(:deploy_key, key: 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQDNd/UJWhPrpb+b/G5oL109y57yKuCxE+WUGJGYaj7WQKsYRJmLYh1mgjrl+KVyfsWpq4ylOxIfFSnN9xBBFN8mlb0Fma5DC7YsSsibJr3MZ19ZNBprwNcdogET7aW9I0In7Wu5f2KqI6e5W/spJHCy4JVxzVMUvk6Myab0LnJ2iQ== dummy@gitlab.com')
+    create(:deploy_keys_project, project: project, deploy_key: project_key)
+    create(:deploy_keys_project, project: project2, deploy_key: internal_key)
+
+    get :index,
+      namespace_id: project.namespace.to_param,
+      project_id: project,
+      format: :json
+
+    expect(response).to be_success
+    store_frontend_fixture(response, example.description)
+  end
+end
diff --git a/spec/javascripts/fixtures/labels.rb b/spec/javascripts/fixtures/labels.rb
new file mode 100644
index 0000000000000..2e4811b64a4b4
--- /dev/null
+++ b/spec/javascripts/fixtures/labels.rb
@@ -0,0 +1,56 @@
+require 'spec_helper'
+
+describe 'Labels (JavaScript fixtures)' do
+  include JavaScriptFixturesHelpers
+
+  let(:admin) { create(:admin) }
+  let(:group) { create(:group, name: 'frontend-fixtures-group' )}
+  let(:project) { create(:project_empty_repo, namespace: group, path: 'labels-project') }
+
+  let!(:project_label_bug) { create(:label, project: project, title: 'bug', color: '#FF0000') }
+  let!(:project_label_enhancement) { create(:label, project: project, title: 'enhancement', color: '#00FF00') }
+  let!(:project_label_feature) { create(:label, project: project, title: 'feature', color: '#0000FF') }
+
+  let!(:group_label_roses) { create(:group_label, group: group, title: 'roses', color: '#FF0000') }
+  let!(:groub_label_space) { create(:group_label, group: group, title: 'some space', color: '#FFFFFF') }
+  let!(:groub_label_violets) { create(:group_label, group: group, title: 'violets', color: '#0000FF') }
+
+  before(:all) do
+    clean_frontend_fixtures('labels/')
+  end
+
+  describe Groups::LabelsController, '(JavaScript fixtures)', type: :controller do
+    render_views
+
+    before(:each) do
+      sign_in(admin)
+    end
+
+    it 'labels/group_labels.json' do |example|
+      get :index,
+        group_id: group,
+        format: 'json'
+
+      expect(response).to be_success
+      store_frontend_fixture(response, example.description)
+    end
+  end
+
+  describe Projects::LabelsController, '(JavaScript fixtures)', type: :controller do
+    render_views
+
+    before(:each) do
+      sign_in(admin)
+    end
+
+    it 'labels/project_labels.json' do |example|
+      get :index,
+        namespace_id: group,
+        project_id: project,
+        format: 'json'
+
+      expect(response).to be_success
+      store_frontend_fixture(response, example.description)
+    end
+  end
+end
diff --git a/spec/javascripts/helpers/user_mock_data_helper.js b/spec/javascripts/helpers/user_mock_data_helper.js
new file mode 100644
index 0000000000000..a9783ea065cf7
--- /dev/null
+++ b/spec/javascripts/helpers/user_mock_data_helper.js
@@ -0,0 +1,16 @@
+export default {
+  createNumberRandomUsers(numberUsers) {
+    const users = [];
+    for (let i = 0; i < numberUsers; i = i += 1) {
+      users.push(
+        {
+          avatar: 'http://gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+          id: (i + 1),
+          name: `GitLab User ${i}`,
+          username: `gitlab${i}`,
+        },
+      );
+    }
+    return users;
+  },
+};
diff --git a/spec/javascripts/issuable_time_tracker_spec.js b/spec/javascripts/issuable_time_tracker_spec.js
index 0a830f25e29ed..8ff93c4f918d2 100644
--- a/spec/javascripts/issuable_time_tracker_spec.js
+++ b/spec/javascripts/issuable_time_tracker_spec.js
@@ -2,7 +2,7 @@
 
 import Vue from 'vue';
 
-require('~/issuable/time_tracking/components/time_tracker');
+import timeTracker from '~/sidebar/components/time_tracking/time_tracker';
 
 function initTimeTrackingComponent(opts) {
   setFixtures(`
@@ -16,187 +16,185 @@ function initTimeTrackingComponent(opts) {
     time_spent: opts.timeSpent,
     human_time_estimate: opts.timeEstimateHumanReadable,
     human_time_spent: opts.timeSpentHumanReadable,
-    docsUrl: '/help/workflow/time_tracking.md',
+    rootPath: '/',
   };
 
-  const TimeTrackingComponent = Vue.component('issuable-time-tracker');
+  const TimeTrackingComponent = Vue.extend(timeTracker);
   this.timeTracker = new TimeTrackingComponent({
     el: '#mock-container',
     propsData: this.initialData,
   });
 }
 
-((gl) => {
-  describe('Issuable Time Tracker', function() {
-    describe('Initialization', function() {
-      beforeEach(function() {
-        initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 5000, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '1h 23m' });
-      });
+describe('Issuable Time Tracker', function() {
+  describe('Initialization', function() {
+    beforeEach(function() {
+      initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 5000, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '1h 23m' });
+    });
 
-      it('should return something defined', function() {
-        expect(this.timeTracker).toBeDefined();
-      });
+    it('should return something defined', function() {
+      expect(this.timeTracker).toBeDefined();
+    });
 
-      it ('should correctly set timeEstimate', function(done) {
-        Vue.nextTick(() => {
-          expect(this.timeTracker.timeEstimate).toBe(this.initialData.time_estimate);
-          done();
-        });
+    it ('should correctly set timeEstimate', function(done) {
+      Vue.nextTick(() => {
+        expect(this.timeTracker.timeEstimate).toBe(this.initialData.time_estimate);
+        done();
       });
-      it ('should correctly set time_spent', function(done) {
-        Vue.nextTick(() => {
-          expect(this.timeTracker.timeSpent).toBe(this.initialData.time_spent);
-          done();
-        });
+    });
+    it ('should correctly set time_spent', function(done) {
+      Vue.nextTick(() => {
+        expect(this.timeTracker.timeSpent).toBe(this.initialData.time_spent);
+        done();
       });
     });
+  });
 
-    describe('Content Display', function() {
-      describe('Panes', function() {
-        describe('Comparison pane', function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 5000, timeEstimateHumanReadable: '', timeSpentHumanReadable: '' });
+  describe('Content Display', function() {
+    describe('Panes', function() {
+      describe('Comparison pane', function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 5000, timeEstimateHumanReadable: '', timeSpentHumanReadable: '' });
+        });
+
+        it('should show the "Comparison" pane when timeEstimate and time_spent are truthy', function(done) {
+          Vue.nextTick(() => {
+            const $comparisonPane = this.timeTracker.$el.querySelector('.time-tracking-comparison-pane');
+            expect(this.timeTracker.showComparisonState).toBe(true);
+            done();
           });
+        });
 
-          it('should show the "Comparison" pane when timeEstimate and time_spent are truthy', function(done) {
+        describe('Remaining meter', function() {
+          it('should display the remaining meter with the correct width', function(done) {
             Vue.nextTick(() => {
-              const $comparisonPane = this.timeTracker.$el.querySelector('.time-tracking-comparison-pane');
-              expect(this.timeTracker.showComparisonState).toBe(true);
+              const meterWidth = this.timeTracker.$el.querySelector('.time-tracking-comparison-pane .meter-fill').style.width;
+              const correctWidth = '5%';
+
+              expect(meterWidth).toBe(correctWidth);
               done();
-            });
+            })
           });
 
-          describe('Remaining meter', function() {
-            it('should display the remaining meter with the correct width', function(done) {
-              Vue.nextTick(() => {
-                const meterWidth = this.timeTracker.$el.querySelector('.time-tracking-comparison-pane .meter-fill').style.width;
-                const correctWidth = '5%';
-
-                expect(meterWidth).toBe(correctWidth);
-                done();
-              })
-            });
-
-            it('should display the remaining meter with the correct background color when within estimate', function(done) {
-              Vue.nextTick(() => {
-                const styledMeter = $(this.timeTracker.$el).find('.time-tracking-comparison-pane .within_estimate .meter-fill');
-                expect(styledMeter.length).toBe(1);
-                done()
-              });
+          it('should display the remaining meter with the correct background color when within estimate', function(done) {
+            Vue.nextTick(() => {
+              const styledMeter = $(this.timeTracker.$el).find('.time-tracking-comparison-pane .within_estimate .meter-fill');
+              expect(styledMeter.length).toBe(1);
+              done()
             });
+          });
 
-            it('should display the remaining meter with the correct background color when over estimate', function(done) {
-              this.timeTracker.time_estimate = 100000;
-              this.timeTracker.time_spent = 20000000;
-              Vue.nextTick(() => {
-                const styledMeter = $(this.timeTracker.$el).find('.time-tracking-comparison-pane .over_estimate .meter-fill');
-                expect(styledMeter.length).toBe(1);
-                done();
-              });
+          it('should display the remaining meter with the correct background color when over estimate', function(done) {
+            this.timeTracker.time_estimate = 100000;
+            this.timeTracker.time_spent = 20000000;
+            Vue.nextTick(() => {
+              const styledMeter = $(this.timeTracker.$el).find('.time-tracking-comparison-pane .over_estimate .meter-fill');
+              expect(styledMeter.length).toBe(1);
+              done();
             });
           });
         });
+      });
 
-        describe("Estimate only pane", function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 0, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '' });
-          });
+      describe("Estimate only pane", function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 100000, timeSpent: 0, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '' });
+        });
 
-          it('should display the human readable version of time estimated', function(done) {
-            Vue.nextTick(() => {
-              const estimateText = this.timeTracker.$el.querySelector('.time-tracking-estimate-only-pane').innerText;
-              const correctText = 'Estimated: 2h 46m';
+        it('should display the human readable version of time estimated', function(done) {
+          Vue.nextTick(() => {
+            const estimateText = this.timeTracker.$el.querySelector('.time-tracking-estimate-only-pane').innerText;
+            const correctText = 'Estimated: 2h 46m';
 
-              expect(estimateText).toBe(correctText);
-              done();
-            });
+            expect(estimateText).toBe(correctText);
+            done();
           });
         });
+      });
 
-        describe('Spent only pane', function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 5000, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '1h 23m' });
-          });
+      describe('Spent only pane', function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 5000, timeEstimateHumanReadable: '2h 46m', timeSpentHumanReadable: '1h 23m' });
+        });
 
-          it('should display the human readable version of time spent', function(done) {
-            Vue.nextTick(() => {
-              const spentText = this.timeTracker.$el.querySelector('.time-tracking-spend-only-pane').innerText;
-              const correctText = 'Spent: 1h 23m';
+        it('should display the human readable version of time spent', function(done) {
+          Vue.nextTick(() => {
+            const spentText = this.timeTracker.$el.querySelector('.time-tracking-spend-only-pane').innerText;
+            const correctText = 'Spent: 1h 23m';
 
-              expect(spentText).toBe(correctText);
-              done();
-            });
+            expect(spentText).toBe(correctText);
+            done();
           });
         });
+      });
 
-        describe('No time tracking pane', function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 0, timeEstimateHumanReadable: 0, timeSpentHumanReadable: 0 });
-          });
+      describe('No time tracking pane', function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 0, timeEstimateHumanReadable: '', timeSpentHumanReadable: '' });
+        });
 
-          it('should only show the "No time tracking" pane when both timeEstimate and time_spent are falsey', function(done) {
-            Vue.nextTick(() => {
-              const $noTrackingPane = this.timeTracker.$el.querySelector('.time-tracking-no-tracking-pane');
-              const noTrackingText =$noTrackingPane.innerText;
-              const correctText = 'No estimate or time spent';
+        it('should only show the "No time tracking" pane when both timeEstimate and time_spent are falsey', function(done) {
+          Vue.nextTick(() => {
+            const $noTrackingPane = this.timeTracker.$el.querySelector('.time-tracking-no-tracking-pane');
+            const noTrackingText =$noTrackingPane.innerText;
+            const correctText = 'No estimate or time spent';
 
-              expect(this.timeTracker.showNoTimeTrackingState).toBe(true);
-              expect($noTrackingPane).toBeVisible();
-              expect(noTrackingText).toBe(correctText);
-              done();
-            });
+            expect(this.timeTracker.showNoTimeTrackingState).toBe(true);
+            expect($noTrackingPane).toBeVisible();
+            expect(noTrackingText).toBe(correctText);
+            done();
           });
         });
+      });
 
-        describe("Help pane", function() {
-          beforeEach(function() {
-            initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 0 });
-          });
+      describe("Help pane", function() {
+        beforeEach(function() {
+          initTimeTrackingComponent.call(this, { timeEstimate: 0, timeSpent: 0 });
+        });
 
-          it('should not show the "Help" pane by default', function(done) {
-            Vue.nextTick(() => {
-              const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
+        it('should not show the "Help" pane by default', function(done) {
+          Vue.nextTick(() => {
+            const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
 
-              expect(this.timeTracker.showHelpState).toBe(false);
-              expect($helpPane).toBeNull();
-              done();
-            });
+            expect(this.timeTracker.showHelpState).toBe(false);
+            expect($helpPane).toBeNull();
+            done();
           });
+        });
 
-          it('should show the "Help" pane when help button is clicked', function(done) {
-            Vue.nextTick(() => {
-              $(this.timeTracker.$el).find('.help-button').click();
+        it('should show the "Help" pane when help button is clicked', function(done) {
+          Vue.nextTick(() => {
+            $(this.timeTracker.$el).find('.help-button').click();
 
-              setTimeout(() => {
-                const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
-                expect(this.timeTracker.showHelpState).toBe(true);
-                expect($helpPane).toBeVisible();
-                done();
-              }, 10);
-            });
+            setTimeout(() => {
+              const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
+              expect(this.timeTracker.showHelpState).toBe(true);
+              expect($helpPane).toBeVisible();
+              done();
+            }, 10);
           });
+        });
 
-          it('should not show the "Help" pane when help button is clicked and then closed', function(done) {
-            Vue.nextTick(() => {
-              $(this.timeTracker.$el).find('.help-button').click();
+        it('should not show the "Help" pane when help button is clicked and then closed', function(done) {
+          Vue.nextTick(() => {
+            $(this.timeTracker.$el).find('.help-button').click();
 
-              setTimeout(() => {
+            setTimeout(() => {
 
-                $(this.timeTracker.$el).find('.close-help-button').click();
+              $(this.timeTracker.$el).find('.close-help-button').click();
 
-                setTimeout(() => {
-                  const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
+              setTimeout(() => {
+                const $helpPane = this.timeTracker.$el.querySelector('.time-tracking-help-state');
 
-                  expect(this.timeTracker.showHelpState).toBe(false);
-                  expect($helpPane).toBeNull();
+                expect(this.timeTracker.showHelpState).toBe(false);
+                expect($helpPane).toBeNull();
 
-                  done();
-                }, 1000);
+                done();
               }, 1000);
-            });
+            }, 1000);
           });
         });
       });
     });
   });
-})(window.gl || (window.gl = {}));
+});
diff --git a/spec/javascripts/lib/utils/accessor_spec.js b/spec/javascripts/lib/utils/accessor_spec.js
new file mode 100644
index 0000000000000..b768d6f2a68b4
--- /dev/null
+++ b/spec/javascripts/lib/utils/accessor_spec.js
@@ -0,0 +1,78 @@
+import AccessorUtilities from '~/lib/utils/accessor';
+
+describe('AccessorUtilities', () => {
+  const testError = new Error('test error');
+
+  describe('isPropertyAccessSafe', () => {
+    let base;
+
+    it('should return `true` if access is safe', () => {
+      base = { testProp: 'testProp' };
+
+      expect(AccessorUtilities.isPropertyAccessSafe(base, 'testProp')).toBe(true);
+    });
+
+    it('should return `false` if access throws an error', () => {
+      base = { get testProp() { throw testError; } };
+
+      expect(AccessorUtilities.isPropertyAccessSafe(base, 'testProp')).toBe(false);
+    });
+
+    it('should return `false` if property is undefined', () => {
+      base = {};
+
+      expect(AccessorUtilities.isPropertyAccessSafe(base, 'testProp')).toBe(false);
+    });
+  });
+
+  describe('isFunctionCallSafe', () => {
+    const base = {};
+
+    it('should return `true` if calling is safe', () => {
+      base.func = () => {};
+
+      expect(AccessorUtilities.isFunctionCallSafe(base, 'func')).toBe(true);
+    });
+
+    it('should return `false` if calling throws an error', () => {
+      base.func = () => { throw new Error('test error'); };
+
+      expect(AccessorUtilities.isFunctionCallSafe(base, 'func')).toBe(false);
+    });
+
+    it('should return `false` if function is undefined', () => {
+      base.func = undefined;
+
+      expect(AccessorUtilities.isFunctionCallSafe(base, 'func')).toBe(false);
+    });
+  });
+
+  describe('isLocalStorageAccessSafe', () => {
+    beforeEach(() => {
+      spyOn(window.localStorage, 'setItem');
+      spyOn(window.localStorage, 'removeItem');
+    });
+
+    it('should return `true` if access is safe', () => {
+      expect(AccessorUtilities.isLocalStorageAccessSafe()).toBe(true);
+    });
+
+    it('should return `false` if access to .setItem isnt safe', () => {
+      window.localStorage.setItem.and.callFake(() => { throw testError; });
+
+      expect(AccessorUtilities.isLocalStorageAccessSafe()).toBe(false);
+    });
+
+    it('should set a test item if access is safe', () => {
+      AccessorUtilities.isLocalStorageAccessSafe();
+
+      expect(window.localStorage.setItem).toHaveBeenCalledWith('isLocalStorageAccessSafe', 'true');
+    });
+
+    it('should remove the test item if access is safe', () => {
+      AccessorUtilities.isLocalStorageAccessSafe();
+
+      expect(window.localStorage.removeItem).toHaveBeenCalledWith('isLocalStorageAccessSafe');
+    });
+  });
+});
diff --git a/spec/javascripts/lib/utils/ajax_cache_spec.js b/spec/javascripts/lib/utils/ajax_cache_spec.js
new file mode 100644
index 0000000000000..7b466a11b9240
--- /dev/null
+++ b/spec/javascripts/lib/utils/ajax_cache_spec.js
@@ -0,0 +1,129 @@
+import AjaxCache from '~/lib/utils/ajax_cache';
+
+describe('AjaxCache', () => {
+  const dummyEndpoint = '/AjaxCache/dummyEndpoint';
+  const dummyResponse = {
+    important: 'dummy data',
+  };
+  let ajaxSpy = (url) => {
+    expect(url).toBe(dummyEndpoint);
+    const deferred = $.Deferred();
+    deferred.resolve(dummyResponse);
+    return deferred.promise();
+  };
+
+  beforeEach(() => {
+    AjaxCache.internalStorage = { };
+    spyOn(jQuery, 'ajax').and.callFake(url => ajaxSpy(url));
+  });
+
+  describe('#get', () => {
+    it('returns undefined if cache is empty', () => {
+      const data = AjaxCache.get(dummyEndpoint);
+
+      expect(data).toBe(undefined);
+    });
+
+    it('returns undefined if cache contains no matching data', () => {
+      AjaxCache.internalStorage['not matching'] = dummyResponse;
+
+      const data = AjaxCache.get(dummyEndpoint);
+
+      expect(data).toBe(undefined);
+    });
+
+    it('returns matching data', () => {
+      AjaxCache.internalStorage[dummyEndpoint] = dummyResponse;
+
+      const data = AjaxCache.get(dummyEndpoint);
+
+      expect(data).toBe(dummyResponse);
+    });
+  });
+
+  describe('#hasData', () => {
+    it('returns false if cache is empty', () => {
+      expect(AjaxCache.hasData(dummyEndpoint)).toBe(false);
+    });
+
+    it('returns false if cache contains no matching data', () => {
+      AjaxCache.internalStorage['not matching'] = dummyResponse;
+
+      expect(AjaxCache.hasData(dummyEndpoint)).toBe(false);
+    });
+
+    it('returns true if data is available', () => {
+      AjaxCache.internalStorage[dummyEndpoint] = dummyResponse;
+
+      expect(AjaxCache.hasData(dummyEndpoint)).toBe(true);
+    });
+  });
+
+  describe('#purge', () => {
+    it('does nothing if cache is empty', () => {
+      AjaxCache.purge(dummyEndpoint);
+
+      expect(AjaxCache.internalStorage).toEqual({ });
+    });
+
+    it('does nothing if cache contains no matching data', () => {
+      AjaxCache.internalStorage['not matching'] = dummyResponse;
+
+      AjaxCache.purge(dummyEndpoint);
+
+      expect(AjaxCache.internalStorage['not matching']).toBe(dummyResponse);
+    });
+
+    it('removes matching data', () => {
+      AjaxCache.internalStorage[dummyEndpoint] = dummyResponse;
+
+      AjaxCache.purge(dummyEndpoint);
+
+      expect(AjaxCache.internalStorage).toEqual({ });
+    });
+  });
+
+  describe('#retrieve', () => {
+    it('stores and returns data from Ajax call if cache is empty', (done) => {
+      AjaxCache.retrieve(dummyEndpoint)
+      .then((data) => {
+        expect(data).toBe(dummyResponse);
+        expect(AjaxCache.internalStorage[dummyEndpoint]).toBe(dummyResponse);
+      })
+      .then(done)
+      .catch(fail);
+    });
+
+    it('returns undefined if Ajax call fails and cache is empty', (done) => {
+      const dummyStatusText = 'exploded';
+      const dummyErrorMessage = 'server exploded';
+      ajaxSpy = (url) => {
+        expect(url).toBe(dummyEndpoint);
+        const deferred = $.Deferred();
+        deferred.reject(null, dummyStatusText, dummyErrorMessage);
+        return deferred.promise();
+      };
+
+      AjaxCache.retrieve(dummyEndpoint)
+      .then(data => fail(`Received unexpected data: ${JSON.stringify(data)}`))
+      .catch((error) => {
+        expect(error.message).toBe(`${dummyEndpoint}: ${dummyErrorMessage}`);
+        expect(error.textStatus).toBe(dummyStatusText);
+        done();
+      })
+      .catch(fail);
+    });
+
+    it('makes no Ajax call if matching data exists', (done) => {
+      AjaxCache.internalStorage[dummyEndpoint] = dummyResponse;
+      ajaxSpy = () => fail(new Error('expected no Ajax call!'));
+
+      AjaxCache.retrieve(dummyEndpoint)
+      .then((data) => {
+        expect(data).toBe(dummyResponse);
+      })
+      .then(done)
+      .catch(fail);
+    });
+  });
+});
diff --git a/spec/javascripts/lib/utils/common_utils_spec.js b/spec/javascripts/lib/utils/common_utils_spec.js
index a00efa10119ce..5eb147ed88806 100644
--- a/spec/javascripts/lib/utils/common_utils_spec.js
+++ b/spec/javascripts/lib/utils/common_utils_spec.js
@@ -362,5 +362,16 @@ require('~/lib/utils/common_utils');
         gl.utils.setCiStatusFavicon(BUILD_URL);
       });
     });
+
+    describe('gl.utils.ajaxPost', () => {
+      it('should perform `$.ajax` call and do `POST` request', () => {
+        const requestURL = '/some/random/api';
+        const data = { keyname: 'value' };
+        const ajaxSpy = spyOn($, 'ajax').and.callFake(() => {});
+
+        gl.utils.ajaxPost(requestURL, data);
+        expect(ajaxSpy.calls.allArgs()[0][0].type).toEqual('POST');
+      });
+    });
   });
 })();
diff --git a/spec/javascripts/notes_spec.js b/spec/javascripts/notes_spec.js
index cdc5c4510ffcd..cfd599f793e73 100644
--- a/spec/javascripts/notes_spec.js
+++ b/spec/javascripts/notes_spec.js
@@ -26,10 +26,10 @@ import '~/notes';
 
     describe('task lists', function() {
       beforeEach(function() {
-        $('form').on('submit', function(e) {
+        $('.js-comment-button').on('click', function(e) {
           e.preventDefault();
         });
-        this.notes = new Notes();
+        this.notes = new Notes('', []);
       });
 
       it('modifies the Markdown field', function() {
@@ -51,7 +51,7 @@ import '~/notes';
       var textarea = '.js-note-text';
 
       beforeEach(function() {
-        this.notes = new Notes();
+        this.notes = new Notes('', []);
 
         this.autoSizeSpy = spyOnEvent($(textarea), 'autosize:update');
         spyOn(this.notes, 'renderNote').and.stub();
@@ -60,9 +60,12 @@ import '~/notes';
           reset: function() {}
         });
 
-        $('form').on('submit', function(e) {
+        $('.js-comment-button').on('click', (e) => {
+          const $form = $(this);
           e.preventDefault();
-          $('.js-main-target-form').trigger('ajax:success');
+          this.notes.addNote($form);
+          this.notes.reenableTargetFormSubmitButton(e);
+          this.notes.resetMainTargetForm(e);
         });
       });
 
@@ -238,8 +241,8 @@ import '~/notes';
         $resultantNote = Notes.animateAppendNote(noteHTML, $notesList);
       });
 
-      it('should have `fade-in` class', () => {
-        expect($resultantNote.hasClass('fade-in')).toEqual(true);
+      it('should have `fade-in-full` class', () => {
+        expect($resultantNote.hasClass('fade-in-full')).toEqual(true);
       });
 
       it('should append note to the notes list', () => {
@@ -269,5 +272,180 @@ import '~/notes';
         expect($note.replaceWith).toHaveBeenCalledWith($updatedNote);
       });
     });
+
+    describe('postComment & updateComment', () => {
+      const sampleComment = 'foo';
+      const updatedComment = 'bar';
+      const note = {
+        id: 1234,
+        html: `<li class="note note-row-1234 timeline-entry" id="note_1234">
+                <div class="note-text">${sampleComment}</div>
+               </li>`,
+        note: sampleComment,
+        valid: true
+      };
+      let $form;
+      let $notesContainer;
+
+      beforeEach(() => {
+        this.notes = new Notes('', []);
+        window.gon.current_username = 'root';
+        window.gon.current_user_fullname = 'Administrator';
+        $form = $('form.js-main-target-form');
+        $notesContainer = $('ul.main-notes-list');
+        $form.find('textarea.js-note-text').val(sampleComment);
+      });
+
+      it('should show placeholder note while new comment is being posted', () => {
+        $('.js-comment-button').click();
+        expect($notesContainer.find('.note.being-posted').length > 0).toEqual(true);
+      });
+
+      it('should remove placeholder note when new comment is done posting', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.resolve(note);
+        expect($notesContainer.find('.note.being-posted').length).toEqual(0);
+      });
+
+      it('should show actual note element when new comment is done posting', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.resolve(note);
+        expect($notesContainer.find(`#note_${note.id}`).length > 0).toEqual(true);
+      });
+
+      it('should reset Form when new comment is done posting', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.resolve(note);
+        expect($form.find('textarea.js-note-text').val()).toEqual('');
+      });
+
+      it('should show flash error message when new comment failed to be posted', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.reject();
+        expect($notesContainer.parent().find('.flash-container .flash-text').is(':visible')).toEqual(true);
+      });
+
+      it('should show flash error message when comment failed to be updated', () => {
+        const deferred = $.Deferred();
+        spyOn($, 'ajax').and.returnValue(deferred.promise());
+        $('.js-comment-button').click();
+
+        deferred.resolve(note);
+        const $noteEl = $notesContainer.find(`#note_${note.id}`);
+        $noteEl.find('.js-note-edit').click();
+        $noteEl.find('textarea.js-note-text').val(updatedComment);
+        $noteEl.find('.js-comment-save-button').click();
+
+        deferred.reject();
+        const $updatedNoteEl = $notesContainer.find(`#note_${note.id}`);
+        expect($updatedNoteEl.hasClass('.being-posted')).toEqual(false); // Remove being-posted visuals
+        expect($updatedNoteEl.find('.note-text').text().trim()).toEqual(sampleComment); // See if comment reverted back to original
+        expect($('.flash-container').is(':visible')).toEqual(true); // Flash error message shown
+      });
+    });
+
+    describe('getFormData', () => {
+      it('should return form metadata object from form reference', () => {
+        this.notes = new Notes('', []);
+
+        const $form = $('form');
+        const sampleComment = 'foobar';
+        $form.find('textarea.js-note-text').val(sampleComment);
+        const { formData, formContent, formAction } = this.notes.getFormData($form);
+
+        expect(formData.indexOf(sampleComment) > -1).toBe(true);
+        expect(formContent).toEqual(sampleComment);
+        expect(formAction).toEqual($form.attr('action'));
+      });
+    });
+
+    describe('hasSlashCommands', () => {
+      beforeEach(() => {
+        this.notes = new Notes('', []);
+      });
+
+      it('should return true when comment has slash commands', () => {
+        const sampleComment = '/wip /milestone %1.0 /merge /unassign Merging this';
+        const hasSlashCommands = this.notes.hasSlashCommands(sampleComment);
+
+        expect(hasSlashCommands).toBeTruthy();
+      });
+
+      it('should return false when comment does NOT have any slash commands', () => {
+        const sampleComment = 'Looking good, Awesome!';
+        const hasSlashCommands = this.notes.hasSlashCommands(sampleComment);
+
+        expect(hasSlashCommands).toBeFalsy();
+      });
+    });
+
+    describe('stripSlashCommands', () => {
+      const REGEX_SLASH_COMMANDS = /\/\w+/g;
+
+      it('should strip slash commands from the comment', () => {
+        this.notes = new Notes();
+        const sampleComment = '/wip /milestone %1.0 /merge /unassign Merging this';
+        const stripedComment = this.notes.stripSlashCommands(sampleComment);
+
+        expect(REGEX_SLASH_COMMANDS.test(stripedComment)).toBeFalsy();
+      });
+    });
+
+    describe('createPlaceholderNote', () => {
+      const sampleComment = 'foobar';
+      const uniqueId = 'b1234-a4567';
+      const currentUsername = 'root';
+      const currentUserFullname = 'Administrator';
+
+      beforeEach(() => {
+        this.notes = new Notes('', []);
+      });
+
+      it('should return constructed placeholder element for regular note based on form contents', () => {
+        const $tempNote = this.notes.createPlaceholderNote({
+          formContent: sampleComment,
+          uniqueId,
+          isDiscussionNote: false,
+          currentUsername,
+          currentUserFullname
+        });
+        const $tempNoteHeader = $tempNote.find('.note-header');
+
+        expect($tempNote.prop('nodeName')).toEqual('LI');
+        expect($tempNote.attr('id')).toEqual(uniqueId);
+        $tempNote.find('.timeline-icon > a, .note-header-info > a').each(function() {
+          expect($(this).attr('href')).toEqual(`/${currentUsername}`);
+        });
+        expect($tempNote.find('.timeline-content').hasClass('discussion')).toBeFalsy();
+        expect($tempNoteHeader.find('.hidden-xs').text().trim()).toEqual(currentUserFullname);
+        expect($tempNoteHeader.find('.note-headline-light').text().trim()).toEqual(`@${currentUsername}`);
+        expect($tempNote.find('.note-body .note-text').text().trim()).toEqual(sampleComment);
+      });
+
+      it('should return constructed placeholder element for discussion note based on form contents', () => {
+        const $tempNote = this.notes.createPlaceholderNote({
+          formContent: sampleComment,
+          uniqueId,
+          isDiscussionNote: true,
+          currentUsername,
+          currentUserFullname
+        });
+
+        expect($tempNote.prop('nodeName')).toEqual('LI');
+        expect($tempNote.find('.timeline-content').hasClass('discussion')).toBeTruthy();
+      });
+    });
   });
 }).call(window);
diff --git a/spec/javascripts/raven/index_spec.js b/spec/javascripts/raven/index_spec.js
new file mode 100644
index 0000000000000..b5662cd0331c7
--- /dev/null
+++ b/spec/javascripts/raven/index_spec.js
@@ -0,0 +1,42 @@
+import RavenConfig from '~/raven/raven_config';
+import index from '~/raven/index';
+
+describe('RavenConfig options', () => {
+  let sentryDsn;
+  let currentUserId;
+  let gitlabUrl;
+  let isProduction;
+  let indexReturnValue;
+
+  beforeEach(() => {
+    sentryDsn = 'sentryDsn';
+    currentUserId = 'currentUserId';
+    gitlabUrl = 'gitlabUrl';
+    isProduction = 'isProduction';
+
+    window.gon = {
+      sentry_dsn: sentryDsn,
+      current_user_id: currentUserId,
+      gitlab_url: gitlabUrl,
+    };
+
+    process.env.NODE_ENV = isProduction;
+
+    spyOn(RavenConfig, 'init');
+
+    indexReturnValue = index();
+  });
+
+  it('should init with .sentryDsn, .currentUserId, .whitelistUrls and .isProduction', () => {
+    expect(RavenConfig.init).toHaveBeenCalledWith({
+      sentryDsn,
+      currentUserId,
+      whitelistUrls: [gitlabUrl],
+      isProduction,
+    });
+  });
+
+  it('should return RavenConfig', () => {
+    expect(indexReturnValue).toBe(RavenConfig);
+  });
+});
diff --git a/spec/javascripts/raven/raven_config_spec.js b/spec/javascripts/raven/raven_config_spec.js
new file mode 100644
index 0000000000000..a2d720760fce6
--- /dev/null
+++ b/spec/javascripts/raven/raven_config_spec.js
@@ -0,0 +1,276 @@
+import Raven from 'raven-js';
+import RavenConfig from '~/raven/raven_config';
+
+describe('RavenConfig', () => {
+  describe('IGNORE_ERRORS', () => {
+    it('should be an array of strings', () => {
+      const areStrings = RavenConfig.IGNORE_ERRORS.every(error => typeof error === 'string');
+
+      expect(areStrings).toBe(true);
+    });
+  });
+
+  describe('IGNORE_URLS', () => {
+    it('should be an array of regexps', () => {
+      const areRegExps = RavenConfig.IGNORE_URLS.every(url => url instanceof RegExp);
+
+      expect(areRegExps).toBe(true);
+    });
+  });
+
+  describe('SAMPLE_RATE', () => {
+    it('should be a finite number', () => {
+      expect(typeof RavenConfig.SAMPLE_RATE).toEqual('number');
+    });
+  });
+
+  describe('init', () => {
+    let options;
+
+    beforeEach(() => {
+      options = {
+        sentryDsn: '//sentryDsn',
+        ravenAssetUrl: '//ravenAssetUrl',
+        currentUserId: 1,
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      };
+
+      spyOn(RavenConfig, 'configure');
+      spyOn(RavenConfig, 'bindRavenErrors');
+      spyOn(RavenConfig, 'setUser');
+
+      RavenConfig.init(options);
+    });
+
+    it('should set the options property', () => {
+      expect(RavenConfig.options).toEqual(options);
+    });
+
+    it('should call the configure method', () => {
+      expect(RavenConfig.configure).toHaveBeenCalled();
+    });
+
+    it('should call the error bindings method', () => {
+      expect(RavenConfig.bindRavenErrors).toHaveBeenCalled();
+    });
+
+    it('should call setUser', () => {
+      expect(RavenConfig.setUser).toHaveBeenCalled();
+    });
+
+    it('should not call setUser if there is no current user ID', () => {
+      RavenConfig.setUser.calls.reset();
+
+      RavenConfig.init({
+        sentryDsn: '//sentryDsn',
+        ravenAssetUrl: '//ravenAssetUrl',
+        currentUserId: undefined,
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      });
+
+      expect(RavenConfig.setUser).not.toHaveBeenCalled();
+    });
+  });
+
+  describe('configure', () => {
+    let options;
+    let raven;
+    let ravenConfig;
+
+    beforeEach(() => {
+      options = {
+        sentryDsn: '//sentryDsn',
+        whitelistUrls: ['//gitlabUrl'],
+        isProduction: true,
+      };
+
+      ravenConfig = jasmine.createSpyObj('ravenConfig', ['shouldSendSample']);
+      raven = jasmine.createSpyObj('raven', ['install']);
+
+      spyOn(Raven, 'config').and.returnValue(raven);
+
+      ravenConfig.options = options;
+      ravenConfig.IGNORE_ERRORS = 'ignore_errors';
+      ravenConfig.IGNORE_URLS = 'ignore_urls';
+
+      RavenConfig.configure.call(ravenConfig);
+    });
+
+    it('should call Raven.config', () => {
+      expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
+        whitelistUrls: options.whitelistUrls,
+        environment: 'production',
+        ignoreErrors: ravenConfig.IGNORE_ERRORS,
+        ignoreUrls: ravenConfig.IGNORE_URLS,
+        shouldSendCallback: jasmine.any(Function),
+      });
+    });
+
+    it('should call Raven.install', () => {
+      expect(raven.install).toHaveBeenCalled();
+    });
+
+    it('should set .environment to development if isProduction is false', () => {
+      ravenConfig.options.isProduction = false;
+
+      RavenConfig.configure.call(ravenConfig);
+
+      expect(Raven.config).toHaveBeenCalledWith(options.sentryDsn, {
+        whitelistUrls: options.whitelistUrls,
+        environment: 'development',
+        ignoreErrors: ravenConfig.IGNORE_ERRORS,
+        ignoreUrls: ravenConfig.IGNORE_URLS,
+        shouldSendCallback: jasmine.any(Function),
+      });
+    });
+  });
+
+  describe('setUser', () => {
+    let ravenConfig;
+
+    beforeEach(() => {
+      ravenConfig = { options: { currentUserId: 1 } };
+      spyOn(Raven, 'setUserContext');
+
+      RavenConfig.setUser.call(ravenConfig);
+    });
+
+    it('should call .setUserContext', function () {
+      expect(Raven.setUserContext).toHaveBeenCalledWith({
+        id: ravenConfig.options.currentUserId,
+      });
+    });
+  });
+
+  describe('bindRavenErrors', () => {
+    let $document;
+    let $;
+
+    beforeEach(() => {
+      $document = jasmine.createSpyObj('$document', ['on']);
+      $ = jasmine.createSpy('$').and.returnValue($document);
+
+      window.$ = $;
+
+      RavenConfig.bindRavenErrors();
+    });
+
+    it('should call .on', function () {
+      expect($document.on).toHaveBeenCalledWith('ajaxError.raven', RavenConfig.handleRavenErrors);
+    });
+  });
+
+  describe('handleRavenErrors', () => {
+    let event;
+    let req;
+    let config;
+    let err;
+
+    beforeEach(() => {
+      event = {};
+      req = { status: 'status', responseText: 'responseText', statusText: 'statusText' };
+      config = { type: 'type', url: 'url', data: 'data' };
+      err = {};
+
+      spyOn(Raven, 'captureMessage');
+
+      RavenConfig.handleRavenErrors(event, req, config, err);
+    });
+
+    it('should call Raven.captureMessage', () => {
+      expect(Raven.captureMessage).toHaveBeenCalledWith(err, {
+        extra: {
+          type: config.type,
+          url: config.url,
+          data: config.data,
+          status: req.status,
+          response: req.responseText,
+          error: err,
+          event,
+        },
+      });
+    });
+
+    describe('if no err is provided', () => {
+      beforeEach(() => {
+        Raven.captureMessage.calls.reset();
+
+        RavenConfig.handleRavenErrors(event, req, config);
+      });
+
+      it('should use req.statusText as the error value', () => {
+        expect(Raven.captureMessage).toHaveBeenCalledWith(req.statusText, {
+          extra: {
+            type: config.type,
+            url: config.url,
+            data: config.data,
+            status: req.status,
+            response: req.responseText,
+            error: req.statusText,
+            event,
+          },
+        });
+      });
+    });
+
+    describe('if no req.responseText is provided', () => {
+      beforeEach(() => {
+        req.responseText = undefined;
+
+        Raven.captureMessage.calls.reset();
+
+        RavenConfig.handleRavenErrors(event, req, config, err);
+      });
+
+      it('should use `Unknown response text` as the response', () => {
+        expect(Raven.captureMessage).toHaveBeenCalledWith(err, {
+          extra: {
+            type: config.type,
+            url: config.url,
+            data: config.data,
+            status: req.status,
+            response: 'Unknown response text',
+            error: err,
+            event,
+          },
+        });
+      });
+    });
+  });
+
+  describe('shouldSendSample', () => {
+    let randomNumber;
+
+    beforeEach(() => {
+      RavenConfig.SAMPLE_RATE = 50;
+
+      spyOn(Math, 'random').and.callFake(() => randomNumber);
+    });
+
+    it('should call Math.random', () => {
+      RavenConfig.shouldSendSample();
+
+      expect(Math.random).toHaveBeenCalled();
+    });
+
+    it('should return true if the sample rate is greater than the random number * 100', () => {
+      randomNumber = 0.1;
+
+      expect(RavenConfig.shouldSendSample()).toBe(true);
+    });
+
+    it('should return false if the sample rate is less than the random number * 100', () => {
+      randomNumber = 0.9;
+
+      expect(RavenConfig.shouldSendSample()).toBe(false);
+    });
+
+    it('should return true if the sample rate is equal to the random number * 100', () => {
+      randomNumber = 0.5;
+
+      expect(RavenConfig.shouldSendSample()).toBe(true);
+    });
+  });
+});
diff --git a/spec/javascripts/sidebar/assignee_title_spec.js b/spec/javascripts/sidebar/assignee_title_spec.js
new file mode 100644
index 0000000000000..5b5b1bf414010
--- /dev/null
+++ b/spec/javascripts/sidebar/assignee_title_spec.js
@@ -0,0 +1,80 @@
+import Vue from 'vue';
+import AssigneeTitle from '~/sidebar/components/assignees/assignee_title';
+
+describe('AssigneeTitle component', () => {
+  let component;
+  let AssigneeTitleComponent;
+
+  beforeEach(() => {
+    AssigneeTitleComponent = Vue.extend(AssigneeTitle);
+  });
+
+  describe('assignee title', () => {
+    it('renders assignee', () => {
+      component = new AssigneeTitleComponent({
+        propsData: {
+          numberOfAssignees: 1,
+          editable: false,
+        },
+      }).$mount();
+
+      expect(component.$el.innerText.trim()).toEqual('Assignee');
+    });
+
+    it('renders 2 assignees', () => {
+      component = new AssigneeTitleComponent({
+        propsData: {
+          numberOfAssignees: 2,
+          editable: false,
+        },
+      }).$mount();
+
+      expect(component.$el.innerText.trim()).toEqual('2 Assignees');
+    });
+  });
+
+  it('does not render spinner by default', () => {
+    component = new AssigneeTitleComponent({
+      propsData: {
+        numberOfAssignees: 0,
+        editable: false,
+      },
+    }).$mount();
+
+    expect(component.$el.querySelector('.fa')).toBeNull();
+  });
+
+  it('renders spinner when loading', () => {
+    component = new AssigneeTitleComponent({
+      propsData: {
+        loading: true,
+        numberOfAssignees: 0,
+        editable: false,
+      },
+    }).$mount();
+
+    expect(component.$el.querySelector('.fa')).not.toBeNull();
+  });
+
+  it('does not render edit link when not editable', () => {
+    component = new AssigneeTitleComponent({
+      propsData: {
+        numberOfAssignees: 0,
+        editable: false,
+      },
+    }).$mount();
+
+    expect(component.$el.querySelector('.edit-link')).toBeNull();
+  });
+
+  it('renders edit link when editable', () => {
+    component = new AssigneeTitleComponent({
+      propsData: {
+        numberOfAssignees: 0,
+        editable: true,
+      },
+    }).$mount();
+
+    expect(component.$el.querySelector('.edit-link')).not.toBeNull();
+  });
+});
diff --git a/spec/javascripts/sidebar/assignees_spec.js b/spec/javascripts/sidebar/assignees_spec.js
new file mode 100644
index 0000000000000..c9453a2118972
--- /dev/null
+++ b/spec/javascripts/sidebar/assignees_spec.js
@@ -0,0 +1,272 @@
+import Vue from 'vue';
+import Assignee from '~/sidebar/components/assignees/assignees';
+import UsersMock from './mock_data';
+import UsersMockHelper from '../helpers/user_mock_data_helper';
+
+describe('Assignee component', () => {
+  let component;
+  let AssigneeComponent;
+
+  beforeEach(() => {
+    AssigneeComponent = Vue.extend(Assignee);
+  });
+
+  describe('No assignees/users', () => {
+    it('displays no assignee icon when collapsed', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [],
+          editable: false,
+        },
+      }).$mount();
+
+      const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
+      expect(collapsed.childElementCount).toEqual(1);
+      expect(collapsed.children[0].getAttribute('aria-label')).toEqual('No Assignee');
+      expect(collapsed.children[0].classList.contains('fa')).toEqual(true);
+      expect(collapsed.children[0].classList.contains('fa-user')).toEqual(true);
+    });
+
+    it('displays only "No assignee" when no users are assigned and the issue is read-only', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [],
+          editable: false,
+        },
+      }).$mount();
+      const componentTextNoUsers = component.$el.querySelector('.assign-yourself').innerText.trim();
+
+      expect(componentTextNoUsers).toBe('No assignee');
+      expect(componentTextNoUsers.indexOf('assign yourself')).toEqual(-1);
+    });
+
+    it('displays only "No assignee" when no users are assigned and the issue can be edited', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [],
+          editable: true,
+        },
+      }).$mount();
+      const componentTextNoUsers = component.$el.querySelector('.assign-yourself').innerText.trim();
+
+      expect(componentTextNoUsers.indexOf('No assignee')).toEqual(0);
+      expect(componentTextNoUsers.indexOf('assign yourself')).toBeGreaterThan(0);
+    });
+
+    it('emits the assign-self event when "assign yourself" is clicked', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [],
+          editable: true,
+        },
+      }).$mount();
+
+      spyOn(component, '$emit');
+      component.$el.querySelector('.assign-yourself .btn-link').click();
+      expect(component.$emit).toHaveBeenCalledWith('assign-self');
+    });
+  });
+
+  describe('One assignee/user', () => {
+    it('displays one assignee icon when collapsed', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users: [
+            UsersMock.user,
+          ],
+          editable: false,
+        },
+      }).$mount();
+
+      const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
+      const assignee = collapsed.children[0];
+      expect(collapsed.childElementCount).toEqual(1);
+      expect(assignee.querySelector('.avatar').getAttribute('src')).toEqual(UsersMock.user.avatar);
+      expect(assignee.querySelector('.avatar').getAttribute('alt')).toEqual(`${UsersMock.user.name}'s avatar`);
+      expect(assignee.querySelector('.author').innerText.trim()).toEqual(UsersMock.user.name);
+    });
+
+    it('Shows one user with avatar, username and author name', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000/',
+          users: [
+            UsersMock.user,
+          ],
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.$el.querySelector('.author_link')).not.toBeNull();
+      // The image
+      expect(component.$el.querySelector('.author_link img').getAttribute('src')).toEqual(UsersMock.user.avatar);
+      // Author name
+      expect(component.$el.querySelector('.author_link .author').innerText.trim()).toEqual(UsersMock.user.name);
+      // Username
+      expect(component.$el.querySelector('.author_link .username').innerText.trim()).toEqual(`@${UsersMock.user.username}`);
+    });
+
+    it('has the root url present in the assigneeUrl method', () => {
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000/',
+          users: [
+            UsersMock.user,
+          ],
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.assigneeUrl(UsersMock.user).indexOf('http://localhost:3000/')).not.toEqual(-1);
+    });
+  });
+
+  describe('Two or more assignees/users', () => {
+    it('displays two assignee icons when collapsed', () => {
+      const users = UsersMockHelper.createNumberRandomUsers(2);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: false,
+        },
+      }).$mount();
+
+      const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
+      expect(collapsed.childElementCount).toEqual(2);
+
+      const first = collapsed.children[0];
+      expect(first.querySelector('.avatar').getAttribute('src')).toEqual(users[0].avatar);
+      expect(first.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[0].name}'s avatar`);
+      expect(first.querySelector('.author').innerText.trim()).toEqual(users[0].name);
+
+      const second = collapsed.children[1];
+      expect(second.querySelector('.avatar').getAttribute('src')).toEqual(users[1].avatar);
+      expect(second.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[1].name}'s avatar`);
+      expect(second.querySelector('.author').innerText.trim()).toEqual(users[1].name);
+    });
+
+    it('displays one assignee icon and counter when collapsed', () => {
+      const users = UsersMockHelper.createNumberRandomUsers(3);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: false,
+        },
+      }).$mount();
+
+      const collapsed = component.$el.querySelector('.sidebar-collapsed-icon');
+      expect(collapsed.childElementCount).toEqual(2);
+
+      const first = collapsed.children[0];
+      expect(first.querySelector('.avatar').getAttribute('src')).toEqual(users[0].avatar);
+      expect(first.querySelector('.avatar').getAttribute('alt')).toEqual(`${users[0].name}'s avatar`);
+      expect(first.querySelector('.author').innerText.trim()).toEqual(users[0].name);
+
+      const second = collapsed.children[1];
+      expect(second.querySelector('.avatar-counter').innerText.trim()).toEqual('+2');
+    });
+
+    it('Shows two assignees', () => {
+      const users = UsersMockHelper.createNumberRandomUsers(2);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.$el.querySelectorAll('.user-item').length).toEqual(users.length);
+      expect(component.$el.querySelector('.user-list-more')).toBe(null);
+    });
+
+    it('Shows the "show-less" assignees label', (done) => {
+      const users = UsersMockHelper.createNumberRandomUsers(6);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.$el.querySelectorAll('.user-item').length).toEqual(component.defaultRenderCount);
+      expect(component.$el.querySelector('.user-list-more')).not.toBe(null);
+      const usersLabelExpectation = users.length - component.defaultRenderCount;
+      expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+        .not.toBe(`+${usersLabelExpectation} more`);
+      component.toggleShowLess();
+      Vue.nextTick(() => {
+        expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+          .toBe('- show less');
+        done();
+      });
+    });
+
+    it('Shows the "show-less" when "n+ more " label is clicked', (done) => {
+      const users = UsersMockHelper.createNumberRandomUsers(6);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: true,
+        },
+      }).$mount();
+
+      component.$el.querySelector('.user-list-more .btn-link').click();
+      Vue.nextTick(() => {
+        expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+          .toBe('- show less');
+        done();
+      });
+    });
+
+    it('gets the count of avatar via a computed property ', () => {
+      const users = UsersMockHelper.createNumberRandomUsers(6);
+      component = new AssigneeComponent({
+        propsData: {
+          rootPath: 'http://localhost:3000',
+          users,
+          editable: true,
+        },
+      }).$mount();
+
+      expect(component.sidebarAvatarCounter).toEqual(`+${users.length - 1}`);
+    });
+
+    describe('n+ more label', () => {
+      beforeEach(() => {
+        const users = UsersMockHelper.createNumberRandomUsers(6);
+        component = new AssigneeComponent({
+          propsData: {
+            rootPath: 'http://localhost:3000',
+            users,
+            editable: true,
+          },
+        }).$mount();
+      });
+
+      it('shows "+1 more" label', () => {
+        expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+          .toBe('+ 1 more');
+      });
+
+      it('shows "show less" label', (done) => {
+        component.toggleShowLess();
+
+        Vue.nextTick(() => {
+          expect(component.$el.querySelector('.user-list-more .btn-link').innerText.trim())
+            .toBe('- show less');
+          done();
+        });
+      });
+    });
+  });
+});
diff --git a/spec/javascripts/sidebar/mock_data.js b/spec/javascripts/sidebar/mock_data.js
new file mode 100644
index 0000000000000..9fc8667ecc9a7
--- /dev/null
+++ b/spec/javascripts/sidebar/mock_data.js
@@ -0,0 +1,109 @@
+/* eslint-disable quote-props*/
+
+const sidebarMockData = {
+  'GET': {
+    '/gitlab-org/gitlab-shell/issues/5.json': {
+      id: 45,
+      iid: 5,
+      author_id: 23,
+      description: 'Nulla ullam commodi delectus adipisci quis sit.',
+      lock_version: null,
+      milestone_id: 21,
+      position: 0,
+      state: 'closed',
+      title: 'Vel et nulla voluptatibus corporis dolor iste saepe laborum.',
+      updated_by_id: 1,
+      created_at: '2017-02-02T21: 49: 49.664Z',
+      updated_at: '2017-05-03T22: 26: 03.760Z',
+      deleted_at: null,
+      time_estimate: 0,
+      total_time_spent: 0,
+      human_time_estimate: null,
+      human_total_time_spent: null,
+      branch_name: null,
+      confidential: false,
+      assignees: [
+        {
+          name: 'User 0',
+          username: 'user0',
+          id: 22,
+          state: 'active',
+          avatar_url: 'http: //www.gravatar.com/avatar/52e4ce24a915fb7e51e1ad3b57f4b00a?s=80\u0026d=identicon',
+          web_url: 'http: //localhost:3001/user0',
+        },
+        {
+          name: 'Marguerite Bartell',
+          username: 'tajuana',
+          id: 18,
+          state: 'active',
+          avatar_url: 'http: //www.gravatar.com/avatar/4852a41fb41616bf8f140d3701673f53?s=80\u0026d=identicon',
+          web_url: 'http: //localhost:3001/tajuana',
+        },
+        {
+          name: 'Laureen Ritchie',
+          username: 'michaele.will',
+          id: 16,
+          state: 'active',
+          avatar_url: 'http: //www.gravatar.com/avatar/e301827eb03be955c9c172cb9a8e4e8a?s=80\u0026d=identicon',
+          web_url: 'http: //localhost:3001/michaele.will',
+        },
+      ],
+      due_date: null,
+      moved_to_id: null,
+      project_id: 4,
+      weight: null,
+      milestone: {
+        id: 21,
+        iid: 1,
+        project_id: 4,
+        title: 'v0.0',
+        description: 'Molestiae commodi laboriosam odio sunt eaque reprehenderit.',
+        state: 'active',
+        created_at: '2017-02-02T21: 49: 30.530Z',
+        updated_at: '2017-02-02T21: 49: 30.530Z',
+        due_date: null,
+        start_date: null,
+      },
+      labels: [],
+    },
+  },
+  'PUT': {
+    '/gitlab-org/gitlab-shell/issues/5.json': {
+      data: {},
+    },
+  },
+};
+
+export default {
+  mediator: {
+    endpoint: '/gitlab-org/gitlab-shell/issues/5.json',
+    editable: true,
+    currentUser: {
+      id: 1,
+      name: 'Administrator',
+      username: 'root',
+      avatar_url: 'http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+    },
+    rootPath: '/',
+  },
+  time: {
+    time_estimate: 3600,
+    total_time_spent: 0,
+    human_time_estimate: '1h',
+    human_total_time_spent: null,
+  },
+  user: {
+    avatar: 'http://gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+    id: 1,
+    name: 'Administrator',
+    username: 'root',
+  },
+
+  sidebarMockInterceptor(request, next) {
+    const body = sidebarMockData[request.method.toUpperCase()][request.url];
+
+    next(request.respondWith(JSON.stringify(body), {
+      status: 200,
+    }));
+  },
+};
diff --git a/spec/javascripts/sidebar/sidebar_assignees_spec.js b/spec/javascripts/sidebar/sidebar_assignees_spec.js
new file mode 100644
index 0000000000000..e0df0a3228f9f
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_assignees_spec.js
@@ -0,0 +1,45 @@
+import Vue from 'vue';
+import SidebarAssignees from '~/sidebar/components/assignees/sidebar_assignees';
+import SidebarMediator from '~/sidebar/sidebar_mediator';
+import SidebarService from '~/sidebar/services/sidebar_service';
+import SidebarStore from '~/sidebar/stores/sidebar_store';
+import Mock from './mock_data';
+
+describe('sidebar assignees', () => {
+  let component;
+  let SidebarAssigneeComponent;
+  preloadFixtures('issues/open-issue.html.raw');
+
+  beforeEach(() => {
+    Vue.http.interceptors.push(Mock.sidebarMockInterceptor);
+    SidebarAssigneeComponent = Vue.extend(SidebarAssignees);
+    spyOn(SidebarMediator.prototype, 'saveAssignees').and.callThrough();
+    spyOn(SidebarMediator.prototype, 'assignYourself').and.callThrough();
+    this.mediator = new SidebarMediator(Mock.mediator);
+    loadFixtures('issues/open-issue.html.raw');
+    this.sidebarAssigneesEl = document.querySelector('#js-vue-sidebar-assignees');
+  });
+
+  afterEach(() => {
+    SidebarService.singleton = null;
+    SidebarStore.singleton = null;
+    SidebarMediator.singleton = null;
+  });
+
+  it('calls the mediator when saves the assignees', () => {
+    component = new SidebarAssigneeComponent()
+      .$mount(this.sidebarAssigneesEl);
+    component.saveAssignees();
+
+    expect(SidebarMediator.prototype.saveAssignees).toHaveBeenCalled();
+  });
+
+  it('calls the mediator when "assignSelf" method is called', () => {
+    component = new SidebarAssigneeComponent()
+      .$mount(this.sidebarAssigneesEl);
+    component.assignSelf();
+
+    expect(SidebarMediator.prototype.assignYourself).toHaveBeenCalled();
+    expect(this.mediator.store.assignees.length).toEqual(1);
+  });
+});
diff --git a/spec/javascripts/sidebar/sidebar_bundle_spec.js b/spec/javascripts/sidebar/sidebar_bundle_spec.js
new file mode 100644
index 0000000000000..7760b34e07198
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_bundle_spec.js
@@ -0,0 +1,42 @@
+import Vue from 'vue';
+import SidebarBundleDomContentLoaded from '~/sidebar/sidebar_bundle';
+import SidebarTimeTracking from '~/sidebar/components/time_tracking/sidebar_time_tracking';
+import SidebarMediator from '~/sidebar/sidebar_mediator';
+import SidebarService from '~/sidebar/services/sidebar_service';
+import SidebarStore from '~/sidebar/stores/sidebar_store';
+import Mock from './mock_data';
+
+describe('sidebar bundle', () => {
+  gl.sidebarOptions = Mock.mediator;
+
+  beforeEach(() => {
+    spyOn(SidebarTimeTracking.methods, 'listenForSlashCommands').and.callFake(() => { });
+    preloadFixtures('issues/open-issue.html.raw');
+    Vue.http.interceptors.push(Mock.sidebarMockInterceptor);
+    loadFixtures('issues/open-issue.html.raw');
+    spyOn(Vue.prototype, '$mount');
+    SidebarBundleDomContentLoaded();
+    this.mediator = new SidebarMediator();
+  });
+
+  afterEach(() => {
+    SidebarService.singleton = null;
+    SidebarStore.singleton = null;
+    SidebarMediator.singleton = null;
+  });
+
+  it('the mediator should be already defined with some data', () => {
+    SidebarBundleDomContentLoaded();
+
+    expect(this.mediator.store).toBeDefined();
+    expect(this.mediator.service).toBeDefined();
+    expect(this.mediator.store.currentUser).toEqual(Mock.mediator.currentUser);
+    expect(this.mediator.store.rootPath).toEqual(Mock.mediator.rootPath);
+    expect(this.mediator.store.endPoint).toEqual(Mock.mediator.endPoint);
+    expect(this.mediator.store.editable).toEqual(Mock.mediator.editable);
+  });
+
+  it('the sidebar time tracking and assignees components to have been mounted', () => {
+    expect(Vue.prototype.$mount).toHaveBeenCalledTimes(2);
+  });
+});
diff --git a/spec/javascripts/sidebar/sidebar_mediator_spec.js b/spec/javascripts/sidebar/sidebar_mediator_spec.js
new file mode 100644
index 0000000000000..2b00fa173346b
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_mediator_spec.js
@@ -0,0 +1,40 @@
+import Vue from 'vue';
+import SidebarMediator from '~/sidebar/sidebar_mediator';
+import SidebarStore from '~/sidebar/stores/sidebar_store';
+import SidebarService from '~/sidebar/services/sidebar_service';
+import Mock from './mock_data';
+
+describe('Sidebar mediator', () => {
+  beforeEach(() => {
+    Vue.http.interceptors.push(Mock.sidebarMockInterceptor);
+    this.mediator = new SidebarMediator(Mock.mediator);
+  });
+
+  afterEach(() => {
+    SidebarService.singleton = null;
+    SidebarStore.singleton = null;
+    SidebarMediator.singleton = null;
+  });
+
+  it('assigns yourself ', () => {
+    this.mediator.assignYourself();
+
+    expect(this.mediator.store.currentUser).toEqual(Mock.mediator.currentUser);
+    expect(this.mediator.store.assignees[0]).toEqual(Mock.mediator.currentUser);
+  });
+
+  it('saves assignees', (done) => {
+    this.mediator.saveAssignees('issue[assignee_ids]')
+      .then((resp) => {
+        expect(resp.status).toEqual(200);
+        done();
+      })
+      .catch(() => {});
+  });
+
+  it('fetches the data', () => {
+    spyOn(this.mediator.service, 'get').and.callThrough();
+    this.mediator.fetch();
+    expect(this.mediator.service.get).toHaveBeenCalled();
+  });
+});
diff --git a/spec/javascripts/sidebar/sidebar_service_spec.js b/spec/javascripts/sidebar/sidebar_service_spec.js
new file mode 100644
index 0000000000000..d41162096a68f
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_service_spec.js
@@ -0,0 +1,32 @@
+import Vue from 'vue';
+import SidebarService from '~/sidebar/services/sidebar_service';
+import Mock from './mock_data';
+
+describe('Sidebar service', () => {
+  beforeEach(() => {
+    Vue.http.interceptors.push(Mock.sidebarMockInterceptor);
+    this.service = new SidebarService('/gitlab-org/gitlab-shell/issues/5.json');
+  });
+
+  afterEach(() => {
+    SidebarService.singleton = null;
+  });
+
+  it('gets the data', (done) => {
+    this.service.get()
+      .then((resp) => {
+        expect(resp).toBeDefined();
+        done();
+      })
+      .catch(() => {});
+  });
+
+  it('updates the data', (done) => {
+    this.service.update('issue[assignee_ids]', [1])
+      .then((resp) => {
+        expect(resp).toBeDefined();
+        done();
+      })
+      .catch(() => {});
+  });
+});
diff --git a/spec/javascripts/sidebar/sidebar_store_spec.js b/spec/javascripts/sidebar/sidebar_store_spec.js
new file mode 100644
index 0000000000000..29facf483b54e
--- /dev/null
+++ b/spec/javascripts/sidebar/sidebar_store_spec.js
@@ -0,0 +1,80 @@
+import SidebarStore from '~/sidebar/stores/sidebar_store';
+import Mock from './mock_data';
+import UsersMockHelper from '../helpers/user_mock_data_helper';
+
+describe('Sidebar store', () => {
+  const assignee = {
+    id: 2,
+    name: 'gitlab user 2',
+    username: 'gitlab2',
+    avatar_url: 'http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+  };
+
+  const anotherAssignee = {
+    id: 3,
+    name: 'gitlab user 3',
+    username: 'gitlab3',
+    avatar_url: 'http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+  };
+
+  beforeEach(() => {
+    this.store = new SidebarStore({
+      currentUser: {
+        id: 1,
+        name: 'Administrator',
+        username: 'root',
+        avatar_url: 'http://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon',
+      },
+      editable: true,
+      rootPath: '/',
+      endpoint: '/gitlab-org/gitlab-shell/issues/5.json',
+    });
+  });
+
+  afterEach(() => {
+    SidebarStore.singleton = null;
+  });
+
+  it('adds a new assignee', () => {
+    this.store.addAssignee(assignee);
+    expect(this.store.assignees.length).toEqual(1);
+  });
+
+  it('removes an assignee', () => {
+    this.store.removeAssignee(assignee);
+    expect(this.store.assignees.length).toEqual(0);
+  });
+
+  it('finds an existent assignee', () => {
+    let foundAssignee;
+
+    this.store.addAssignee(assignee);
+    foundAssignee = this.store.findAssignee(assignee);
+    expect(foundAssignee).toBeDefined();
+    expect(foundAssignee).toEqual(assignee);
+    foundAssignee = this.store.findAssignee(anotherAssignee);
+    expect(foundAssignee).toBeUndefined();
+  });
+
+  it('removes all assignees', () => {
+    this.store.removeAllAssignees();
+    expect(this.store.assignees.length).toEqual(0);
+  });
+
+  it('set assigned data', () => {
+    const users = {
+      assignees: UsersMockHelper.createNumberRandomUsers(3),
+    };
+
+    this.store.setAssigneeData(users);
+    expect(this.store.assignees.length).toEqual(3);
+  });
+
+  it('set time tracking data', () => {
+    this.store.setTimeTrackingData(Mock.time);
+    expect(this.store.timeEstimate).toEqual(Mock.time.time_estimate);
+    expect(this.store.totalTimeSpent).toEqual(Mock.time.total_time_spent);
+    expect(this.store.humanTimeEstimate).toEqual(Mock.time.human_time_estimate);
+    expect(this.store.humanTotalTimeSpent).toEqual(Mock.time.human_total_time_spent);
+  });
+});
diff --git a/spec/javascripts/signin_tabs_memoizer_spec.js b/spec/javascripts/signin_tabs_memoizer_spec.js
index d83d9a57b4246..5b4f5933b34de 100644
--- a/spec/javascripts/signin_tabs_memoizer_spec.js
+++ b/spec/javascripts/signin_tabs_memoizer_spec.js
@@ -1,3 +1,5 @@
+import AccessorUtilities from '~/lib/utils/accessor';
+
 require('~/signin_tabs_memoizer');
 
 ((global) => {
@@ -19,6 +21,8 @@ require('~/signin_tabs_memoizer');
 
     beforeEach(() => {
       loadFixtures(fixtureTemplate);
+
+      spyOn(AccessorUtilities, 'isLocalStorageAccessSafe').and.returnValue(true);
     });
 
     it('does nothing if no tab was previously selected', () => {
@@ -49,5 +53,91 @@ require('~/signin_tabs_memoizer');
 
       expect(memo.readData()).toEqual('#standard');
     });
+
+    describe('class constructor', () => {
+      beforeEach(() => {
+        memo = createMemoizer();
+      });
+
+      it('should set .isLocalStorageAvailable', () => {
+        expect(AccessorUtilities.isLocalStorageAccessSafe).toHaveBeenCalled();
+        expect(memo.isLocalStorageAvailable).toBe(true);
+      });
+    });
+
+    describe('saveData', () => {
+      beforeEach(() => {
+        memo = {
+          currentTabKey,
+        };
+
+        spyOn(localStorage, 'setItem');
+      });
+
+      describe('if .isLocalStorageAvailable is `false`', () => {
+        beforeEach(function () {
+          memo.isLocalStorageAvailable = false;
+
+          global.ActiveTabMemoizer.prototype.saveData.call(memo);
+        });
+
+        it('should not call .setItem', () => {
+          expect(localStorage.setItem).not.toHaveBeenCalled();
+        });
+      });
+
+      describe('if .isLocalStorageAvailable is `true`', () => {
+        const value = 'value';
+
+        beforeEach(function () {
+          memo.isLocalStorageAvailable = true;
+
+          global.ActiveTabMemoizer.prototype.saveData.call(memo, value);
+        });
+
+        it('should call .setItem', () => {
+          expect(localStorage.setItem).toHaveBeenCalledWith(currentTabKey, value);
+        });
+      });
+    });
+
+    describe('readData', () => {
+      const itemValue = 'itemValue';
+      let readData;
+
+      beforeEach(() => {
+        memo = {
+          currentTabKey,
+        };
+
+        spyOn(localStorage, 'getItem').and.returnValue(itemValue);
+      });
+
+      describe('if .isLocalStorageAvailable is `false`', () => {
+        beforeEach(function () {
+          memo.isLocalStorageAvailable = false;
+
+          readData = global.ActiveTabMemoizer.prototype.readData.call(memo);
+        });
+
+        it('should not call .getItem and should return `null`', () => {
+          expect(localStorage.getItem).not.toHaveBeenCalled();
+          expect(readData).toBe(null);
+        });
+      });
+
+      describe('if .isLocalStorageAvailable is `true`', () => {
+        beforeEach(function () {
+          memo.isLocalStorageAvailable = true;
+
+          readData = global.ActiveTabMemoizer.prototype.readData.call(memo);
+        });
+
+        it('should call .getItem and return the localStorage value', () => {
+          expect(window.localStorage.getItem).toHaveBeenCalledWith(currentTabKey);
+          expect(readData).toBe(itemValue);
+        });
+      });
+    });
   });
 })(window);
diff --git a/spec/javascripts/subbable_resource_spec.js b/spec/javascripts/subbable_resource_spec.js
deleted file mode 100644
index 454386697f54f..0000000000000
--- a/spec/javascripts/subbable_resource_spec.js
+++ /dev/null
@@ -1,63 +0,0 @@
-/* eslint-disable max-len, arrow-parens, comma-dangle */
-
-require('~/subbable_resource');
-
-/*
-* Test that each rest verb calls the publish and subscribe function and passes the correct value back
-*
-*
-* */
-((global) => {
-  describe('Subbable Resource', function () {
-    describe('PubSub', function () {
-      beforeEach(function () {
-        this.MockResource = new global.SubbableResource('https://example.com');
-      });
-      it('should successfully add a single subscriber', function () {
-        const callback = () => {};
-        this.MockResource.subscribe(callback);
-
-        expect(this.MockResource.subscribers.length).toBe(1);
-        expect(this.MockResource.subscribers[0]).toBe(callback);
-      });
-
-      it('should successfully add multiple subscribers', function () {
-        const callbackOne = () => {};
-        const callbackTwo = () => {};
-        const callbackThree = () => {};
-
-        this.MockResource.subscribe(callbackOne);
-        this.MockResource.subscribe(callbackTwo);
-        this.MockResource.subscribe(callbackThree);
-
-        expect(this.MockResource.subscribers.length).toBe(3);
-      });
-
-      it('should successfully publish an update to a single subscriber', function () {
-        const state = { myprop: 1 };
-
-        const callbacks = {
-          one: (data) => expect(data.myprop).toBe(2),
-          two: (data) => expect(data.myprop).toBe(2),
-          three: (data) => expect(data.myprop).toBe(2)
-        };
-
-        const spyOne = spyOn(callbacks, 'one');
-        const spyTwo = spyOn(callbacks, 'two');
-        const spyThree = spyOn(callbacks, 'three');
-
-        this.MockResource.subscribe(callbacks.one);
-        this.MockResource.subscribe(callbacks.two);
-        this.MockResource.subscribe(callbacks.three);
-
-        state.myprop += 1;
-
-        this.MockResource.publish(state);
-
-        expect(spyOne).toHaveBeenCalled();
-        expect(spyTwo).toHaveBeenCalled();
-        expect(spyThree).toHaveBeenCalled();
-      });
-    });
-  });
-})(window.gl || (window.gl = {}));
diff --git a/spec/javascripts/vue_shared/translate_spec.js b/spec/javascripts/vue_shared/translate_spec.js
new file mode 100644
index 0000000000000..cbb3cbdff4675
--- /dev/null
+++ b/spec/javascripts/vue_shared/translate_spec.js
@@ -0,0 +1,90 @@
+import Vue from 'vue';
+import Translate from '~/vue_shared/translate';
+
+Vue.use(Translate);
+
+describe('Vue translate filter', () => {
+  let el;
+
+  beforeEach(() => {
+    el = document.createElement('div');
+
+    document.body.appendChild(el);
+  });
+
+  it('translate single text', (done) => {
+    const comp = new Vue({
+      el,
+      template: `
+        <span>
+          {{ __('testing') }}
+        </span>
+      `,
+    }).$mount();
+
+    Vue.nextTick(() => {
+      expect(
+        comp.$el.textContent.trim(),
+      ).toBe('testing');
+
+      done();
+    });
+  });
+
+  it('translate plural text with single count', (done) => {
+    const comp = new Vue({
+      el,
+      template: `
+        <span>
+          {{ n__('%d day', '%d days', 1) }}
+        </span>
+      `,
+    }).$mount();
+
+    Vue.nextTick(() => {
+      expect(
+        comp.$el.textContent.trim(),
+      ).toBe('1 day');
+
+      done();
+    });
+  });
+
+  it('translate plural text with multiple count', (done) => {
+    const comp = new Vue({
+      el,
+      template: `
+        <span>
+          {{ n__('%d day', '%d days', 2) }}
+        </span>
+      `,
+    }).$mount();
+
+    Vue.nextTick(() => {
+      expect(
+        comp.$el.textContent.trim(),
+      ).toBe('2 days');
+
+      done();
+    });
+  });
+
+  it('translate plural without replacing any text', (done) => {
+    const comp = new Vue({
+      el,
+      template: `
+        <span>
+          {{ n__('day', 'days', 2) }}
+        </span>
+      `,
+    }).$mount();
+
+    Vue.nextTick(() => {
+      expect(
+        comp.$el.textContent.trim(),
+      ).toBe('days');
+
+      done();
+    });
+  });
+});
diff --git a/spec/lib/banzai/filter/redactor_filter_spec.rb b/spec/lib/banzai/filter/redactor_filter_spec.rb
index 8a6fe1ad6a39d..7c4a0f32c7b73 100644
--- a/spec/lib/banzai/filter/redactor_filter_spec.rb
+++ b/spec/lib/banzai/filter/redactor_filter_spec.rb
@@ -113,7 +113,7 @@ def reference_link(data)
       it 'allows references for assignee' do
         assignee = create(:user)
         project = create(:empty_project, :public)
-        issue = create(:issue, :confidential, project: project, assignee: assignee)
+        issue = create(:issue, :confidential, project: project, assignees: [assignee])
 
         link = reference_link(project: project.id, issue: issue.id, reference_type: 'issue')
         doc = filter(link, current_user: assignee)
diff --git a/spec/lib/gitlab/ci/build/artifacts/metadata/entry_spec.rb b/spec/lib/gitlab/ci/build/artifacts/metadata/entry_spec.rb
index abc93e1b44a92..3b9056114674f 100644
--- a/spec/lib/gitlab/ci/build/artifacts/metadata/entry_spec.rb
+++ b/spec/lib/gitlab/ci/build/artifacts/metadata/entry_spec.rb
@@ -135,6 +135,17 @@ def entry(path)
       subject { |example| path(example).nodes }
       it { is_expected.to eq 4 }
     end
+
+    describe '#blob' do
+      let(:file_entry) { |example| path(example) }
+      subject { file_entry.blob }
+
+      it 'returns a blob representing the entry data' do
+        expect(subject).to be_a(Blob)
+        expect(subject.path).to eq(file_entry.path)
+        expect(subject.size).to eq(file_entry.metadata[:size])
+      end
+    end
   end
 
   describe 'non-existent/', path: 'non-existent/' do
diff --git a/spec/lib/gitlab/elastic/project_search_results_spec.rb b/spec/lib/gitlab/elastic/project_search_results_spec.rb
index c97ba15c06fb8..f7d7ffec2f78e 100644
--- a/spec/lib/gitlab/elastic/project_search_results_spec.rb
+++ b/spec/lib/gitlab/elastic/project_search_results_spec.rb
@@ -100,7 +100,7 @@
     let(:admin) { create(:admin) }
     let!(:issue) { create(:issue, project: project, title: 'Issue 1') }
     let!(:security_issue_1) { create(:issue, :confidential, project: project, title: 'Security issue 1', author: author) }
-    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignee: assignee) }
+    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignees: [assignee]) }
 
     before do
       Gitlab::Elastic::Helper.refresh_index
diff --git a/spec/lib/gitlab/elastic/search_results_spec.rb b/spec/lib/gitlab/elastic/search_results_spec.rb
index c98c91fe2f91b..a2819da397366 100644
--- a/spec/lib/gitlab/elastic/search_results_spec.rb
+++ b/spec/lib/gitlab/elastic/search_results_spec.rb
@@ -89,9 +89,9 @@
     before do
       @issue = create(:issue, project: project_1, title: 'Issue 1', iid: 1)
       @security_issue_1 = create(:issue, :confidential, project: project_1, title: 'Security issue 1', author: author, iid: 2)
-      @security_issue_2 = create(:issue, :confidential, title: 'Security issue 2', project: project_1, assignee: assignee, iid: 3)
+      @security_issue_2 = create(:issue, :confidential, title: 'Security issue 2', project: project_1, assignees: [assignee], iid: 3)
       @security_issue_3 = create(:issue, :confidential, project: project_2, title: 'Security issue 3', author: author, iid: 1)
-      @security_issue_4 = create(:issue, :confidential, project: project_3, title: 'Security issue 4', assignee: assignee, iid: 1)
+      @security_issue_4 = create(:issue, :confidential, project: project_3, title: 'Security issue 4', assignees: [assignee], iid: 1)
       @security_issue_5 = create(:issue, :confidential, project: project_4, title: 'Security issue 5', iid: 1)
 
       Gitlab::Elastic::Helper.refresh_index
diff --git a/spec/lib/gitlab/git/repository_spec.rb b/spec/lib/gitlab/git/repository_spec.rb
index ddedb7c34438b..fea186fd4f457 100644
--- a/spec/lib/gitlab/git/repository_spec.rb
+++ b/spec/lib/gitlab/git/repository_spec.rb
@@ -1062,7 +1062,7 @@ def commit_files(commit)
       end
 
       it "allows ordering by date" do
-        expect_any_instance_of(Rugged::Walker).to receive(:sorting).with(Rugged::SORT_DATE)
+        expect_any_instance_of(Rugged::Walker).to receive(:sorting).with(Rugged::SORT_DATE | Rugged::SORT_TOPO)
 
         repository.find_commits(order: :date)
       end
diff --git a/spec/lib/gitlab/github_import/issue_formatter_spec.rb b/spec/lib/gitlab/github_import/issue_formatter_spec.rb
index f34d09f2c1d9a..a4089592cf215 100644
--- a/spec/lib/gitlab/github_import/issue_formatter_spec.rb
+++ b/spec/lib/gitlab/github_import/issue_formatter_spec.rb
@@ -43,7 +43,7 @@
           description: "*Created by: octocat*\n\nI'm having a problem with this.",
           state: 'opened',
           author_id: project.creator_id,
-          assignee_id: nil,
+          assignee_ids: [],
           created_at: created_at,
           updated_at: updated_at
         }
@@ -64,7 +64,7 @@
           description: "*Created by: octocat*\n\nI'm having a problem with this.",
           state: 'closed',
           author_id: project.creator_id,
-          assignee_id: nil,
+          assignee_ids: [],
           created_at: created_at,
           updated_at: updated_at
         }
@@ -77,19 +77,19 @@
       let(:raw_data) { double(base_data.merge(assignee: octocat)) }
 
       it 'returns nil as assignee_id when is not a GitLab user' do
-        expect(issue.attributes.fetch(:assignee_id)).to be_nil
+        expect(issue.attributes.fetch(:assignee_ids)).to be_empty
       end
 
       it 'returns GitLab user id associated with GitHub id as assignee_id' do
         gl_user = create(:omniauth_user, extern_uid: octocat.id, provider: 'github')
 
-        expect(issue.attributes.fetch(:assignee_id)).to eq gl_user.id
+        expect(issue.attributes.fetch(:assignee_ids)).to eq [gl_user.id]
       end
 
       it 'returns GitLab user id associated with GitHub email as assignee_id' do
         gl_user = create(:user, email: octocat.email)
 
-        expect(issue.attributes.fetch(:assignee_id)).to eq gl_user.id
+        expect(issue.attributes.fetch(:assignee_ids)).to eq [gl_user.id]
       end
     end
 
diff --git a/spec/lib/gitlab/google_code_import/importer_spec.rb b/spec/lib/gitlab/google_code_import/importer_spec.rb
index ccaa88a5c798f..622a0f513f43a 100644
--- a/spec/lib/gitlab/google_code_import/importer_spec.rb
+++ b/spec/lib/gitlab/google_code_import/importer_spec.rb
@@ -49,7 +49,7 @@
       expect(issue).not_to be_nil
       expect(issue.iid).to eq(169)
       expect(issue.author).to eq(project.creator)
-      expect(issue.assignee).to eq(mapped_user)
+      expect(issue.assignees).to eq([mapped_user])
       expect(issue.state).to eq("closed")
       expect(issue.label_names).to include("Priority: Medium")
       expect(issue.label_names).to include("Status: Fixed")
diff --git a/spec/lib/gitlab/health_checks/fs_shards_check_spec.rb b/spec/lib/gitlab/health_checks/fs_shards_check_spec.rb
index 4cd8cf313a54a..45ccd3d6459d6 100644
--- a/spec/lib/gitlab/health_checks/fs_shards_check_spec.rb
+++ b/spec/lib/gitlab/health_checks/fs_shards_check_spec.rb
@@ -82,9 +82,9 @@
         it { is_expected.to include(metric_class.new(:filesystem_readable, 0, shard: :default)) }
         it { is_expected.to include(metric_class.new(:filesystem_writable, 0, shard: :default)) }
 
-        it { is_expected.to include(have_attributes(name: :filesystem_access_latency, value: be > 0, labels: { shard: :default })) }
-        it { is_expected.to include(have_attributes(name: :filesystem_read_latency, value: be > 0, labels: { shard: :default })) }
-        it { is_expected.to include(have_attributes(name: :filesystem_write_latency, value: be > 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_access_latency, value: be >= 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_read_latency, value: be >= 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_write_latency, value: be >= 0, labels: { shard: :default })) }
       end
 
       context 'storage points to directory that has both read and write rights' do
@@ -96,9 +96,9 @@
         it { is_expected.to include(metric_class.new(:filesystem_readable, 1, shard: :default)) }
         it { is_expected.to include(metric_class.new(:filesystem_writable, 1, shard: :default)) }
 
-        it { is_expected.to include(have_attributes(name: :filesystem_access_latency, value: be > 0, labels: { shard: :default })) }
-        it { is_expected.to include(have_attributes(name: :filesystem_read_latency, value: be > 0, labels: { shard: :default })) }
-        it { is_expected.to include(have_attributes(name: :filesystem_write_latency, value: be > 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_access_latency, value: be >= 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_read_latency, value: be >= 0, labels: { shard: :default })) }
+        it { is_expected.to include(have_attributes(name: :filesystem_write_latency, value: be >= 0, labels: { shard: :default })) }
       end
     end
   end
diff --git a/spec/lib/gitlab/health_checks/simple_check_shared.rb b/spec/lib/gitlab/health_checks/simple_check_shared.rb
index 1fa6d0faef983..3f871d6603442 100644
--- a/spec/lib/gitlab/health_checks/simple_check_shared.rb
+++ b/spec/lib/gitlab/health_checks/simple_check_shared.rb
@@ -8,7 +8,7 @@
 
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_success", value: 1)) }
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_timeout", value: 0)) }
-      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be > 0)) }
+      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be >= 0)) }
     end
 
     context 'Check is misbehaving' do
@@ -18,7 +18,7 @@
 
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_success", value: 0)) }
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_timeout", value: 0)) }
-      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be > 0)) }
+      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be >= 0)) }
     end
 
     context 'Check is timeouting' do
@@ -28,7 +28,7 @@
 
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_success", value: 0)) }
       it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_timeout", value: 1)) }
-      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be > 0)) }
+      it { is_expected.to include(have_attributes(name: "#{metrics_prefix}_latency", value: be >= 0)) }
     end
   end
 
diff --git a/spec/lib/gitlab/i18n_spec.rb b/spec/lib/gitlab/i18n_spec.rb
new file mode 100644
index 0000000000000..52f2614d5cab6
--- /dev/null
+++ b/spec/lib/gitlab/i18n_spec.rb
@@ -0,0 +1,27 @@
+require 'spec_helper'
+
+module Gitlab
+  describe I18n, lib: true do
+    let(:user) { create(:user, preferred_language: 'es') }
+
+    describe '.set_locale' do
+      it 'sets the locale based on current user preferred language' do
+        Gitlab::I18n.set_locale(user)
+
+        expect(FastGettext.locale).to eq('es')
+        expect(::I18n.locale).to eq(:es)
+      end
+    end
+
+    describe '.reset_locale' do
+      it 'resets the locale to the default language' do
+        Gitlab::I18n.set_locale(user)
+
+        Gitlab::I18n.reset_locale
+
+        expect(FastGettext.locale).to eq('en')
+        expect(::I18n.locale).to eq(:en)
+      end
+    end
+  end
+end
diff --git a/spec/lib/gitlab/import_export/all_models.yml b/spec/lib/gitlab/import_export/all_models.yml
index 99739b6b4af53..41d7c7fff27fc 100644
--- a/spec/lib/gitlab/import_export/all_models.yml
+++ b/spec/lib/gitlab/import_export/all_models.yml
@@ -3,12 +3,13 @@ issues:
 - subscriptions
 - award_emoji
 - author
-- assignee
+- assignees
 - updated_by
 - milestone
 - notes
 - label_links
 - labels
+- last_edited_by
 - todos
 - user_agent_detail
 - moved_to
@@ -16,6 +17,7 @@ issues:
 - merge_requests_closing_issues
 - metrics
 - timelogs
+- issue_assignees
 events:
 - author
 - project
@@ -26,6 +28,7 @@ notes:
 - noteable
 - author
 - updated_by
+- last_edited_by
 - resolved_by
 - todos
 - events
@@ -72,6 +75,7 @@ merge_requests:
 - notes
 - label_links
 - labels
+- last_edited_by
 - todos
 - target_project
 - source_project
diff --git a/spec/lib/gitlab/import_export/project_tree_saver_spec.rb b/spec/lib/gitlab/import_export/project_tree_saver_spec.rb
index 1035428b2e7d3..5aeb29b7fecb7 100644
--- a/spec/lib/gitlab/import_export/project_tree_saver_spec.rb
+++ b/spec/lib/gitlab/import_export/project_tree_saver_spec.rb
@@ -203,7 +203,7 @@
   end
 
   def setup_project
-    issue = create(:issue, assignee: user)
+    issue = create(:issue, assignees: [user])
     snippet = create(:project_snippet)
     release = create(:release)
     group = create(:group)
diff --git a/spec/lib/gitlab/import_export/safe_model_attributes.yml b/spec/lib/gitlab/import_export/safe_model_attributes.yml
index bb83f1f53f84a..f25722bc1c14b 100644
--- a/spec/lib/gitlab/import_export/safe_model_attributes.yml
+++ b/spec/lib/gitlab/import_export/safe_model_attributes.yml
@@ -24,6 +24,8 @@ Issue:
 - time_estimate
 - relative_position
 - service_desk_reply_to
+- last_edited_at
+- last_edited_by_id
 Event:
 - id
 - target_type
@@ -158,6 +160,8 @@ MergeRequest:
 - rebase_commit_sha
 - time_estimate
 - squash
+- last_edited_at
+- last_edited_by_id
 MergeRequestDiff:
 - id
 - state
diff --git a/spec/lib/gitlab/project_search_results_spec.rb b/spec/lib/gitlab/project_search_results_spec.rb
index 6618ce49a9c7d..0bfa1d39d8b59 100644
--- a/spec/lib/gitlab/project_search_results_spec.rb
+++ b/spec/lib/gitlab/project_search_results_spec.rb
@@ -90,7 +90,7 @@
     let(:project) { create(:empty_project, :internal) }
     let!(:issue) { create(:issue, project: project, title: 'Issue 1') }
     let!(:security_issue_1) { create(:issue, :confidential, project: project, title: 'Security issue 1', author: author) }
-    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignee: assignee) }
+    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignees: [assignee]) }
 
     it 'does not list project confidential issues for non project members' do
       results = described_class.new(non_member, project, query)
diff --git a/spec/lib/gitlab/prometheus_spec.rb b/spec/lib/gitlab/prometheus_spec.rb
index 280264188e2e2..fc453a2704b36 100644
--- a/spec/lib/gitlab/prometheus_spec.rb
+++ b/spec/lib/gitlab/prometheus_spec.rb
@@ -49,6 +49,36 @@
     end
   end
 
+  describe 'failure to reach a provided prometheus url' do
+    let(:prometheus_url) {"https://prometheus.invalid.example.com"}
+
+    context 'exceptions are raised' do
+      it 'raises a Gitlab::PrometheusError error when a SocketError is rescued' do
+        req_stub = stub_prometheus_request_with_exception(prometheus_url, SocketError)
+
+        expect { subject.send(:get, prometheus_url) }
+          .to raise_error(Gitlab::PrometheusError, "Can't connect to #{prometheus_url}")
+        expect(req_stub).to have_been_requested
+      end
+
+      it 'raises a Gitlab::PrometheusError error when a SSLError is rescued' do
+        req_stub = stub_prometheus_request_with_exception(prometheus_url, OpenSSL::SSL::SSLError)
+
+        expect { subject.send(:get, prometheus_url) }
+          .to raise_error(Gitlab::PrometheusError, "#{prometheus_url} contains invalid SSL data")
+        expect(req_stub).to have_been_requested
+      end
+
+      it 'raises a Gitlab::PrometheusError error when a HTTParty::Error is rescued' do
+        req_stub = stub_prometheus_request_with_exception(prometheus_url, HTTParty::Error)
+
+        expect { subject.send(:get, prometheus_url) }
+          .to raise_error(Gitlab::PrometheusError, "Network connection error")
+        expect(req_stub).to have_been_requested
+      end
+    end
+  end
+
   describe '#query' do
     let(:prometheus_query) { prometheus_cpu_query('env-slug') }
     let(:query_url) { prometheus_query_url(prometheus_query) }
diff --git a/spec/lib/gitlab/search_results_spec.rb b/spec/lib/gitlab/search_results_spec.rb
index 847fb97740015..31c3cd4d53c8e 100644
--- a/spec/lib/gitlab/search_results_spec.rb
+++ b/spec/lib/gitlab/search_results_spec.rb
@@ -72,9 +72,9 @@
     let(:admin) { create(:admin) }
     let!(:issue) { create(:issue, project: project_1, title: 'Issue 1') }
     let!(:security_issue_1) { create(:issue, :confidential, project: project_1, title: 'Security issue 1', author: author) }
-    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project_1, assignee: assignee) }
+    let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project_1, assignees: [assignee]) }
     let!(:security_issue_3) { create(:issue, :confidential, project: project_2, title: 'Security issue 3', author: author) }
-    let!(:security_issue_4) { create(:issue, :confidential, project: project_3, title: 'Security issue 4', assignee: assignee) }
+    let!(:security_issue_4) { create(:issue, :confidential, project: project_3, title: 'Security issue 4', assignees: [assignee]) }
     let!(:security_issue_5) { create(:issue, :confidential, project: project_4, title: 'Security issue 5') }
 
     it 'does not list confidential issues for non project members' do
diff --git a/spec/lib/gitlab/shell_spec.rb b/spec/lib/gitlab/shell_spec.rb
index 7f037b1916b6d..22c69eeef2f02 100644
--- a/spec/lib/gitlab/shell_spec.rb
+++ b/spec/lib/gitlab/shell_spec.rb
@@ -147,4 +147,45 @@
       end
     end
   end
+
+  describe 'projects commands' do
+    let(:projects_path) { 'tmp/tests/shell-projects-test/bin/gitlab-projects' }
+
+    before do
+      allow(Gitlab.config.gitlab_shell).to receive(:path).and_return('tmp/tests/shell-projects-test')
+      allow(Gitlab.config.gitlab_shell).to receive(:git_timeout).and_return(800)
+    end
+
+    describe '#fetch_remote' do
+      it 'returns true when the command succeeds' do
+        expect(Gitlab::Popen).to receive(:popen)
+          .with([projects_path, 'fetch-remote', 'current/storage', 'project/path.git', 'new/storage', '800']).and_return([nil, 0])
+
+        expect(gitlab_shell.fetch_remote('current/storage', 'project/path', 'new/storage')).to be true
+      end
+
+      it 'raises an exception when the command fails' do
+        expect(Gitlab::Popen).to receive(:popen)
+        .with([projects_path, 'fetch-remote', 'current/storage', 'project/path.git', 'new/storage', '800']).and_return(["error", 1])
+
+        expect { gitlab_shell.fetch_remote('current/storage', 'project/path', 'new/storage') }.to raise_error(Gitlab::Shell::Error, "error")
+      end
+    end
+
+    describe '#import_repository' do
+      it 'returns true when the command succeeds' do
+        expect(Gitlab::Popen).to receive(:popen)
+          .with([projects_path, 'import-project', 'current/storage', 'project/path.git', 'https://gitlab.com/gitlab-org/gitlab-ce.git', "800"]).and_return([nil, 0])
+
+        expect(gitlab_shell.import_repository('current/storage', 'project/path', 'https://gitlab.com/gitlab-org/gitlab-ce.git')).to be true
+      end
+
+      it 'raises an exception when the command fails' do
+        expect(Gitlab::Popen).to receive(:popen)
+        .with([projects_path, 'import-project', 'current/storage', 'project/path.git', 'https://gitlab.com/gitlab-org/gitlab-ce.git', "800"]).and_return(["error", 1])
+
+        expect { gitlab_shell.import_repository('current/storage', 'project/path', 'https://gitlab.com/gitlab-org/gitlab-ce.git') }.to raise_error(Gitlab::Shell::Error, "error")
+      end
+    end
+  end
 end
diff --git a/spec/mailers/notify_spec.rb b/spec/mailers/notify_spec.rb
index 5c89d001c8d22..80e8440b35d4d 100644
--- a/spec/mailers/notify_spec.rb
+++ b/spec/mailers/notify_spec.rb
@@ -36,11 +36,11 @@ def have_referable_subject(referable, reply: false)
       end
 
       context 'for issues' do
-        let(:issue) { create(:issue, author: current_user, assignee: assignee, project: project) }
-        let(:issue_with_description) { create(:issue, author: current_user, assignee: assignee, project: project, description: 'My awesome description') }
+        let(:issue) { create(:issue, author: current_user, assignees: [assignee], project: project) }
+        let(:issue_with_description) { create(:issue, author: current_user, assignees: [assignee], project: project, description: 'My awesome description') }
 
         describe 'that are new' do
-          subject { described_class.new_issue_email(issue.assignee_id, issue.id) }
+          subject { described_class.new_issue_email(issue.assignees.first.id, issue.id) }
 
           it_behaves_like 'an assignee email'
           it_behaves_like 'an email starting a new thread with reply-by-email enabled' do
@@ -69,7 +69,7 @@ def have_referable_subject(referable, reply: false)
         end
 
         describe 'that are new with a description' do
-          subject { described_class.new_issue_email(issue_with_description.assignee_id, issue_with_description.id) }
+          subject { described_class.new_issue_email(issue_with_description.assignees.first.id, issue_with_description.id) }
 
           it_behaves_like 'it should show Gmail Actions View Issue link'
 
@@ -79,7 +79,7 @@ def have_referable_subject(referable, reply: false)
         end
 
         describe 'that have been reassigned' do
-          subject { described_class.reassigned_issue_email(recipient.id, issue.id, previous_assignee.id, current_user.id) }
+          subject { described_class.reassigned_issue_email(recipient.id, issue.id, [previous_assignee.id], current_user.id) }
 
           it_behaves_like 'a multiple recipients email'
           it_behaves_like 'an answer to an existing thread with reply-by-email enabled' do
diff --git a/spec/models/ci/artifact_blob_spec.rb b/spec/models/ci/artifact_blob_spec.rb
new file mode 100644
index 0000000000000..968593d7e9bef
--- /dev/null
+++ b/spec/models/ci/artifact_blob_spec.rb
@@ -0,0 +1,44 @@
+require 'spec_helper'
+
+describe Ci::ArtifactBlob, models: true do
+  let(:build) { create(:ci_build, :artifacts) }
+  let(:entry) { build.artifacts_metadata_entry('other_artifacts_0.1.2/another-subdirectory/banana_sample.gif') }
+
+  subject { described_class.new(entry) }
+
+  describe '#id' do
+    it 'returns a hash of the path' do
+      expect(subject.id).to eq(Digest::SHA1.hexdigest(entry.path))
+    end
+  end
+
+  describe '#name' do
+    it 'returns the entry name' do
+      expect(subject.name).to eq(entry.name)
+    end
+  end
+
+  describe '#path' do
+    it 'returns the entry path' do
+      expect(subject.path).to eq(entry.path)
+    end
+  end
+
+  describe '#size' do
+    it 'returns the entry size' do
+      expect(subject.size).to eq(entry.metadata[:size])
+    end
+  end
+
+  describe '#mode' do
+    it 'returns the entry mode' do
+      expect(subject.mode).to eq(entry.metadata[:mode])
+    end
+  end
+
+  describe '#external_storage' do
+    it 'returns :build_artifact' do
+      expect(subject.external_storage).to eq(:build_artifact)
+    end
+  end
+end
diff --git a/spec/models/concerns/elastic/issue_spec.rb b/spec/models/concerns/elastic/issue_spec.rb
index f7c6861c1cd1c..9995ab255a81c 100644
--- a/spec/models/concerns/elastic/issue_spec.rb
+++ b/spec/models/concerns/elastic/issue_spec.rb
@@ -31,11 +31,14 @@
   end
 
   it "returns json with all needed elements" do
-    issue = create :issue, project: project
+    assignee = create(:user)
+    issue = create :issue, project: project, assignees: [assignee]
 
     expected_hash = issue.attributes.extract!('id', 'iid', 'title', 'description', 'created_at',
                                                 'updated_at', 'state', 'project_id', 'author_id',
-                                                'assignee_id', 'confidential')
+                                                'confidential')
+
+    expected_hash['assignee_id'] = [assignee.id]
 
     expect(issue.as_indexed_json).to eq(expected_hash)
   end
diff --git a/spec/models/concerns/issuable_spec.rb b/spec/models/concerns/issuable_spec.rb
index 37d7513b3ead7..46e0fd78d702a 100644
--- a/spec/models/concerns/issuable_spec.rb
+++ b/spec/models/concerns/issuable_spec.rb
@@ -10,7 +10,6 @@
 
     it { is_expected.to belong_to(:project) }
     it { is_expected.to belong_to(:author) }
-    it { is_expected.to belong_to(:assignee) }
     it { is_expected.to have_many(:notes).dependent(:destroy) }
     it { is_expected.to have_many(:todos).dependent(:destroy) }
 
@@ -66,60 +65,6 @@
     end
   end
 
-  describe 'assignee_name' do
-    it 'is delegated to assignee' do
-      issue.update!(assignee: create(:user))
-
-      expect(issue.assignee_name).to eq issue.assignee.name
-    end
-
-    it 'returns nil when assignee is nil' do
-      issue.assignee_id = nil
-      issue.save(validate: false)
-
-      expect(issue.assignee_name).to eq nil
-    end
-  end
-
-  describe "before_save" do
-    describe "#update_cache_counts" do
-      context "when previous assignee exists" do
-        before do
-          assignee = create(:user)
-          issue.project.team << [assignee, :developer]
-          issue.update(assignee: assignee)
-        end
-
-        it "updates cache counts for new assignee" do
-          user = create(:user)
-
-          expect(user).to receive(:update_cache_counts)
-
-          issue.update(assignee: user)
-        end
-
-        it "updates cache counts for previous assignee" do
-          old_assignee = issue.assignee
-          allow(User).to receive(:find_by_id).with(old_assignee.id).and_return(old_assignee)
-
-          expect(old_assignee).to receive(:update_cache_counts)
-
-          issue.update(assignee: nil)
-        end
-      end
-
-      context "when previous assignee does not exist" do
-        before{ issue.update(assignee: nil) }
-
-        it "updates cache count for the new assignee" do
-          expect_any_instance_of(User).to receive(:update_cache_counts)
-
-          issue.update(assignee: user)
-        end
-      end
-    end
-  end
-
   describe ".search" do
     let!(:searchable_issue) { create(:issue, title: "Searchable issue") }
 
@@ -324,7 +269,20 @@
     end
 
     context "issue is assigned" do
-      before { issue.update_attribute(:assignee, user) }
+      before { issue.assignees << user }
+
+      it "returns correct hook data" do
+        expect(data[:assignees].first).to eq(user.hook_attrs)
+      end
+    end
+
+    context "merge_request is assigned" do
+      let(:merge_request) { create(:merge_request) }
+      let(:data) { merge_request.to_hook_data(user) }
+
+      before do
+        merge_request.update_attribute(:assignee, user)
+      end
 
       it "returns correct hook data" do
         expect(data[:object_attributes]['assignee_id']).to eq(user.id)
@@ -346,24 +304,6 @@
     include_examples 'deprecated repository hook data'
   end
 
-  describe '#card_attributes' do
-    it 'includes the author name' do
-      allow(issue).to receive(:author).and_return(double(name: 'Robert'))
-      allow(issue).to receive(:assignee).and_return(nil)
-
-      expect(issue.card_attributes).
-        to eq({ 'Author' => 'Robert', 'Assignee' => nil })
-    end
-
-    it 'includes the assignee name' do
-      allow(issue).to receive(:author).and_return(double(name: 'Robert'))
-      allow(issue).to receive(:assignee).and_return(double(name: 'Douwe'))
-
-      expect(issue.card_attributes).
-        to eq({ 'Author' => 'Robert', 'Assignee' => 'Douwe' })
-    end
-  end
-
   describe '#labels_array' do
     let(:project) { create(:empty_project) }
     let(:bug) { create(:label, project: project, title: 'bug') }
@@ -502,27 +442,6 @@ def create_issue(milestone, labels)
     end
   end
 
-  describe '#assignee_or_author?' do
-    let(:user) { build(:user, id: 1) }
-    let(:issue) { build(:issue) }
-
-    it 'returns true for a user that is assigned to an issue' do
-      issue.assignee = user
-
-      expect(issue.assignee_or_author?(user)).to eq(true)
-    end
-
-    it 'returns true for a user that is the author of an issue' do
-      issue.author = user
-
-      expect(issue.assignee_or_author?(user)).to eq(true)
-    end
-
-    it 'returns false for a user that is not the assignee or author' do
-      expect(issue.assignee_or_author?(user)).to eq(false)
-    end
-  end
-
   describe '#spend_time' do
     let(:user) { create(:user) }
     let(:issue) { create(:issue) }
diff --git a/spec/models/concerns/milestoneish_spec.rb b/spec/models/concerns/milestoneish_spec.rb
index 68e4c0a522bc8..675b730c5575a 100644
--- a/spec/models/concerns/milestoneish_spec.rb
+++ b/spec/models/concerns/milestoneish_spec.rb
@@ -11,13 +11,13 @@
   let(:milestone) { create(:milestone, project: project) }
   let!(:issue) { create(:issue, project: project, milestone: milestone) }
   let!(:security_issue_1) { create(:issue, :confidential, project: project, author: author, milestone: milestone) }
-  let!(:security_issue_2) { create(:issue, :confidential, project: project, assignee: assignee, milestone: milestone) }
+  let!(:security_issue_2) { create(:issue, :confidential, project: project, assignees: [assignee], milestone: milestone) }
   let!(:closed_issue_1) { create(:issue, :closed, project: project, milestone: milestone) }
   let!(:closed_issue_2) { create(:issue, :closed, project: project, milestone: milestone) }
   let!(:closed_security_issue_1) { create(:issue, :confidential, :closed, project: project, author: author, milestone: milestone) }
-  let!(:closed_security_issue_2) { create(:issue, :confidential, :closed, project: project, assignee: assignee, milestone: milestone) }
+  let!(:closed_security_issue_2) { create(:issue, :confidential, :closed, project: project, assignees: [assignee], milestone: milestone) }
   let!(:closed_security_issue_3) { create(:issue, :confidential, :closed, project: project, author: author, milestone: milestone) }
-  let!(:closed_security_issue_4) { create(:issue, :confidential, :closed, project: project, assignee: assignee, milestone: milestone) }
+  let!(:closed_security_issue_4) { create(:issue, :confidential, :closed, project: project, assignees: [assignee], milestone: milestone) }
   let!(:merge_request) { create(:merge_request, source_project: project, target_project: project, milestone: milestone) }
 
   before do
diff --git a/spec/models/event_spec.rb b/spec/models/event_spec.rb
index a9c5b604268cd..b8cb967c4cc16 100644
--- a/spec/models/event_spec.rb
+++ b/spec/models/event_spec.rb
@@ -118,8 +118,8 @@
     let(:author) { create(:author) }
     let(:assignee) { create(:user) }
     let(:admin) { create(:admin) }
-    let(:issue) { create(:issue, project: project, author: author, assignee: assignee) }
-    let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee) }
+    let(:issue) { create(:issue, project: project, author: author, assignees: [assignee]) }
+    let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee]) }
     let(:note_on_commit) { create(:note_on_commit, project: project) }
     let(:note_on_issue) { create(:note_on_issue, noteable: issue, project: project) }
     let(:note_on_confidential_issue) { create(:note_on_issue, noteable: confidential_issue, project: project) }
diff --git a/spec/models/issue_collection_spec.rb b/spec/models/issue_collection_spec.rb
index d8aed25c041f5..93c2c538e1056 100644
--- a/spec/models/issue_collection_spec.rb
+++ b/spec/models/issue_collection_spec.rb
@@ -28,7 +28,7 @@
       end
 
       it 'returns the issues the user is assigned to' do
-        issue1.assignee = user
+        issue1.assignees << user
 
         expect(collection.updatable_by_user(user)).to eq([issue1])
       end
diff --git a/spec/models/issue_spec.rb b/spec/models/issue_spec.rb
index 8748b98a4e357..725f5c2311fd3 100644
--- a/spec/models/issue_spec.rb
+++ b/spec/models/issue_spec.rb
@@ -3,6 +3,7 @@
 describe Issue, models: true do
   describe "Associations" do
     it { is_expected.to belong_to(:milestone) }
+    it { is_expected.to have_many(:assignees) }
   end
 
   describe 'modules' do
@@ -37,6 +38,64 @@
     end
   end
 
+  describe "before_save" do
+    describe "#update_cache_counts when an issue is reassigned" do
+      let(:issue) { create(:issue) }
+      let(:assignee) { create(:user) }
+
+      context "when previous assignee exists" do
+        before do
+          issue.project.team << [assignee, :developer]
+          issue.assignees << assignee
+        end
+
+        it "updates cache counts for new assignee" do
+          user = create(:user)
+
+          expect(user).to receive(:update_cache_counts)
+
+          issue.assignees << user
+        end
+
+        it "updates cache counts for previous assignee" do
+          issue.assignees.first
+
+          expect_any_instance_of(User).to receive(:update_cache_counts)
+
+          issue.assignees.destroy_all
+        end
+      end
+
+      context "when previous assignee does not exist" do
+        it "updates cache count for the new assignee" do
+          issue.assignees = []
+
+          expect_any_instance_of(User).to receive(:update_cache_counts)
+
+          issue.assignees << assignee
+        end
+      end
+    end
+  end
+
+  describe '#card_attributes' do
+    it 'includes the author name' do
+      allow(subject).to receive(:author).and_return(double(name: 'Robert'))
+      allow(subject).to receive(:assignees).and_return([])
+
+      expect(subject.card_attributes).
+        to eq({ 'Author' => 'Robert', 'Assignee' => '' })
+    end
+
+    it 'includes the assignee name' do
+      allow(subject).to receive(:author).and_return(double(name: 'Robert'))
+      allow(subject).to receive(:assignees).and_return([double(name: 'Douwe')])
+
+      expect(subject.card_attributes).
+        to eq({ 'Author' => 'Robert', 'Assignee' => 'Douwe' })
+    end
+  end
+
   describe '#closed_at' do
     after do
       Timecop.return
@@ -124,13 +183,24 @@
     end
   end
 
-  describe '#is_being_reassigned?' do
-    it 'returns true if the issue assignee has changed' do
-      subject.assignee = create(:user)
-      expect(subject.is_being_reassigned?).to be_truthy
+  describe '#assignee_or_author?' do
+    let(:user) { create(:user) }
+    let(:issue) { create(:issue) }
+
+    it 'returns true for a user that is assigned to an issue' do
+      issue.assignees << user
+
+      expect(issue.assignee_or_author?(user)).to be_truthy
     end
-    it 'returns false if the issue assignee has not changed' do
-      expect(subject.is_being_reassigned?).to be_falsey
+
+    it 'returns true for a user that is the author of an issue' do
+      issue.update(author: user)
+
+      expect(issue.assignee_or_author?(user)).to be_truthy
+    end
+
+    it 'returns false for a user that is not the assignee or author' do
+      expect(issue.assignee_or_author?(user)).to be_falsey
     end
   end
 
@@ -383,14 +453,14 @@
       user1 = create(:user)
       user2 = create(:user)
       project = create(:empty_project)
-      issue = create(:issue, assignee: user1, project: project)
+      issue = create(:issue, assignees: [user1], project: project)
       project.add_developer(user1)
       project.add_developer(user2)
 
       expect(user1.assigned_open_issues_count).to eq(1)
       expect(user2.assigned_open_issues_count).to eq(0)
 
-      issue.assignee = user2
+      issue.assignees = [user2]
       issue.save
 
       expect(user1.assigned_open_issues_count).to eq(0)
@@ -676,6 +746,11 @@
       expect(attrs_hash).to include(:human_total_time_spent)
       expect(attrs_hash).to include('time_estimate')
     end
+
+    it 'includes assignee_ids and deprecated assignee_id' do
+      expect(attrs_hash).to include(:assignee_id)
+      expect(attrs_hash).to include(:assignee_ids)
+    end
   end
 
   describe '#check_for_spam' do
diff --git a/spec/models/merge_request_spec.rb b/spec/models/merge_request_spec.rb
index 49568109a4c42..dad4e923bc682 100644
--- a/spec/models/merge_request_spec.rb
+++ b/spec/models/merge_request_spec.rb
@@ -9,6 +9,7 @@
     it { is_expected.to belong_to(:target_project).class_name('Project') }
     it { is_expected.to belong_to(:source_project).class_name('Project') }
     it { is_expected.to belong_to(:merge_user).class_name("User") }
+    it { is_expected.to belong_to(:assignee) }
     it { is_expected.to have_many(:merge_request_diffs).dependent(:destroy) }
     it { is_expected.to have_many(:approver_groups).dependent(:destroy) }
   end
@@ -87,6 +88,86 @@
     end
   end
 
+  describe "before_save" do
+    describe "#update_cache_counts when a merge request is reassigned" do
+      let(:project) { create :project }
+      let(:merge_request) { create(:merge_request, source_project: project, target_project: project) }
+      let(:assignee) { create :user }
+
+      context "when previous assignee exists" do
+        before do
+          project.team << [assignee, :developer]
+          merge_request.update(assignee: assignee)
+        end
+
+        it "updates cache counts for new assignee" do
+          user = create(:user)
+
+          expect(user).to receive(:update_cache_counts)
+
+          merge_request.update(assignee: user)
+        end
+
+        it "updates cache counts for previous assignee" do
+          old_assignee = merge_request.assignee
+          allow(User).to receive(:find_by_id).with(old_assignee.id).and_return(old_assignee)
+
+          expect(old_assignee).to receive(:update_cache_counts)
+
+          merge_request.update(assignee: nil)
+        end
+      end
+
+      context "when previous assignee does not exist" do
+        it "updates cache count for the new assignee" do
+          merge_request.update(assignee: nil)
+
+          expect_any_instance_of(User).to receive(:update_cache_counts)
+
+          merge_request.update(assignee: assignee)
+        end
+      end
+    end
+  end
+
+  describe '#card_attributes' do
+    it 'includes the author name' do
+      allow(subject).to receive(:author).and_return(double(name: 'Robert'))
+      allow(subject).to receive(:assignee).and_return(nil)
+
+      expect(subject.card_attributes).
+        to eq({ 'Author' => 'Robert', 'Assignee' => nil })
+    end
+
+    it 'includes the assignee name' do
+      allow(subject).to receive(:author).and_return(double(name: 'Robert'))
+      allow(subject).to receive(:assignee).and_return(double(name: 'Douwe'))
+
+      expect(subject.card_attributes).
+        to eq({ 'Author' => 'Robert', 'Assignee' => 'Douwe' })
+    end
+  end
+
+  describe '#assignee_or_author?' do
+    let(:user) { create(:user) }
+
+    it 'returns true for a user that is assigned to a merge request' do
+      subject.assignee = user
+
+      expect(subject.assignee_or_author?(user)).to eq(true)
+    end
+
+    it 'returns true for a user that is the author of a merge request' do
+      subject.author = user
+
+      expect(subject.assignee_or_author?(user)).to eq(true)
+    end
+
+    it 'returns false for a user that is not the assignee or author' do
+      expect(subject.assignee_or_author?(user)).to eq(false)
+    end
+  end
+
   describe '#cache_merge_request_closes_issues!' do
     before do
       subject.project.team << [subject.author, :developer]
@@ -296,16 +377,6 @@ def set_compare(merge_request)
     end
   end
 
-  describe '#is_being_reassigned?' do
-    it 'returns true if the merge_request assignee has changed' do
-      subject.assignee = create(:user)
-      expect(subject.is_being_reassigned?).to be_truthy
-    end
-    it 'returns false if the merge request assignee has not changed' do
-      expect(subject.is_being_reassigned?).to be_falsey
-    end
-  end
-
   describe '#for_fork?' do
     it 'returns true if the merge request is for a fork' do
       subject.source_project = build_stubbed(:empty_project, namespace: create(:group))
diff --git a/spec/models/network/graph_spec.rb b/spec/models/network/graph_spec.rb
index 46b36e11c230f..0fe8a591a4557 100644
--- a/spec/models/network/graph_spec.rb
+++ b/spec/models/network/graph_spec.rb
@@ -10,17 +10,17 @@
     expect(graph.notes).to eq( { note_on_commit.commit_id => 1 } )
   end
 
-  describe "#commits" do
+  describe '#commits' do
     let(:graph) { described_class.new(project, 'refs/heads/master', project.repository.commit, nil) }
 
-    it "returns a list of commits" do
+    it 'returns a list of commits' do
       commits = graph.commits
 
       expect(commits).not_to be_empty
       expect(commits).to all( be_kind_of(Network::Commit) )
     end
 
-    it "sorts the commits by commit date (descending)" do
+    it 'it the commits by commit date (descending)' do
       # Remove duplicate timestamps because they make it harder to
       # assert that the commits are sorted as expected.
       commits = graph.commits.uniq(&:date)
@@ -29,5 +29,20 @@
       expect(commits).not_to be_empty
       expect(commits.map(&:id)).to eq(sorted_commits.map(&:id))
     end
+
+    it 'sorts children before parents for commits with the same timestamp' do
+      commits_by_time = graph.commits.group_by(&:date)
+
+      commits_by_time.each do |time, commits|
+        commit_ids = commits.map(&:id)
+
+        commits.each_with_index do |commit, index|
+          parent_indexes = commit.parent_ids.map { |parent_id| commit_ids.find_index(parent_id) }.compact
+
+          # All parents of the current commit should appear after it
+          expect(parent_indexes).to all( be > index )
+        end
+      end
+    end
   end
 end
diff --git a/spec/models/project_services/prometheus_service_spec.rb b/spec/models/project_services/prometheus_service_spec.rb
index d15079b686be9..f3126bc1e570b 100644
--- a/spec/models/project_services/prometheus_service_spec.rb
+++ b/spec/models/project_services/prometheus_service_spec.rb
@@ -94,7 +94,7 @@
     [404, 500].each do |status|
       context "when Prometheus responds with #{status}" do
         before do
-          stub_all_prometheus_requests(environment.slug, status: status, body: 'QUERY FAILED!')
+          stub_all_prometheus_requests(environment.slug, status: status, body: "QUERY FAILED!")
         end
 
         it { is_expected.to eq(success: false, result: %(#{status} - "QUERY FAILED!")) }
diff --git a/spec/models/user_spec.rb b/spec/models/user_spec.rb
index 0c01b5773e171..e09f3b4f6cd31 100644
--- a/spec/models/user_spec.rb
+++ b/spec/models/user_spec.rb
@@ -1930,4 +1930,12 @@ def add_user(access)
       expect(User.active.count).to eq(1)
     end
   end
+
+  describe 'preferred language' do
+    it 'is English by default' do
+      user = create(:user)
+
+      expect(user.preferred_language).to eq('en')
+    end
+  end
 end
diff --git a/spec/policies/issue_policy_spec.rb b/spec/policies/issue_policy_spec.rb
index 9a870b7fda12f..4a07c864428c6 100644
--- a/spec/policies/issue_policy_spec.rb
+++ b/spec/policies/issue_policy_spec.rb
@@ -15,7 +15,7 @@ def permissions(user, issue)
   context 'a private project' do
     let(:non_member) { create(:user) }
     let(:project) { create(:empty_project, :private) }
-    let(:issue) { create(:issue, project: project, assignee: assignee, author: author) }
+    let(:issue) { create(:issue, project: project, assignees: [assignee], author: author) }
     let(:issue_no_assignee) { create(:issue, project: project) }
 
     before do
@@ -69,7 +69,7 @@ def permissions(user, issue)
     end
 
     context 'with confidential issues' do
-      let(:confidential_issue) { create(:issue, :confidential, project: project, assignee: assignee, author: author) }
+      let(:confidential_issue) { create(:issue, :confidential, project: project, assignees: [assignee], author: author) }
       let(:confidential_issue_no_assignee) { create(:issue, :confidential, project: project) }
 
       it 'does not allow non-members to read confidential issues' do
@@ -110,7 +110,7 @@ def permissions(user, issue)
 
   context 'a public project' do
     let(:project) { create(:empty_project, :public) }
-    let(:issue) { create(:issue, project: project, assignee: assignee, author: author) }
+    let(:issue) { create(:issue, project: project, assignees: [assignee], author: author) }
     let(:issue_no_assignee) { create(:issue, project: project) }
 
     before do
@@ -157,7 +157,7 @@ def permissions(user, issue)
     end
 
     context 'with confidential issues' do
-      let(:confidential_issue) { create(:issue, :confidential, project: project, assignee: assignee, author: author) }
+      let(:confidential_issue) { create(:issue, :confidential, project: project, assignees: [assignee], author: author) }
       let(:confidential_issue_no_assignee) { create(:issue, :confidential, project: project) }
 
       it 'does not allow guests to read confidential issues' do
diff --git a/spec/requests/api/issues_spec.rb b/spec/requests/api/issues_spec.rb
index c28587a785b55..619680f211351 100644
--- a/spec/requests/api/issues_spec.rb
+++ b/spec/requests/api/issues_spec.rb
@@ -19,7 +19,7 @@
   let!(:closed_issue) do
     create :closed_issue,
            author: user,
-           assignee: user,
+           assignees: [user],
            project: project,
            state: :closed,
            milestone: milestone,
@@ -31,14 +31,14 @@
            :confidential,
            project: project,
            author: author,
-           assignee: assignee,
+           assignees: [assignee],
            created_at: generate(:past_time),
            updated_at: 2.hours.ago
   end
   let!(:issue) do
     create :issue,
            author: user,
-           assignee: user,
+           assignees: [user],
            project: project,
            milestone: milestone,
            created_at: generate(:past_time),
@@ -265,7 +265,7 @@
     let!(:group_closed_issue) do
       create :closed_issue,
              author: user,
-             assignee: user,
+             assignees: [user],
              project: group_project,
              state: :closed,
              milestone: group_milestone,
@@ -276,13 +276,13 @@
              :confidential,
              project: group_project,
              author: author,
-             assignee: assignee,
+             assignees: [assignee],
              updated_at: 2.hours.ago
     end
     let!(:group_issue) do
       create :issue,
              author: user,
-             assignee: user,
+             assignees: [user],
              project: group_project,
              milestone: group_milestone,
              updated_at: 1.hour.ago,
@@ -687,6 +687,7 @@
       expect(json_response['updated_at']).to be_present
       expect(json_response['labels']).to eq(issue.label_names)
       expect(json_response['milestone']).to be_a Hash
+      expect(json_response['assignees']).to be_a Array
       expect(json_response['assignee']).to be_a Hash
       expect(json_response['author']).to be_a Hash
       expect(json_response['confidential']).to be_falsy
@@ -760,9 +761,22 @@
   end
 
   describe "POST /projects/:id/issues" do
+    context 'support for deprecated assignee_id' do
+      it 'creates a new project issue' do
+        post api("/projects/#{project.id}/issues", user),
+          title: 'new issue', assignee_id: user2.id
+
+        expect(response).to have_http_status(201)
+        expect(json_response['title']).to eq('new issue')
+        expect(json_response['assignee']['name']).to eq(user2.name)
+        expect(json_response['assignees'].first['name']).to eq(user2.name)
+      end
+    end
+
     it 'creates a new project issue' do
       post api("/projects/#{project.id}/issues", user),
-        title: 'new issue', labels: 'label, label2', weight: 3
+        title: 'new issue', labels: 'label, label2', weight: 3,
+        assignee_ids: [user2.id]
 
       expect(response).to have_http_status(201)
       expect(json_response['title']).to eq('new issue')
@@ -770,6 +784,8 @@
       expect(json_response['labels']).to eq(%w(label label2))
       expect(json_response['confidential']).to be_falsy
       expect(json_response['weight']).to eq(3)
+      expect(json_response['assignee']['name']).to eq(user2.name)
+      expect(json_response['assignees'].first['name']).to eq(user2.name)
     end
 
     it 'creates a new confidential project issue' do
@@ -1059,6 +1075,46 @@
     end
   end
 
+  describe 'PUT /projects/:id/issues/:issue_iid to update assignee' do
+    context 'support for deprecated assignee_id' do
+      it 'removes assignee' do
+        put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+          assignee_id: 0
+
+        expect(response).to have_http_status(200)
+
+        expect(json_response['assignee']).to be_nil
+      end
+
+      it 'updates an issue with new assignee' do
+        put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+          assignee_id: user2.id
+
+        expect(response).to have_http_status(200)
+
+        expect(json_response['assignee']['name']).to eq(user2.name)
+      end
+    end
+
+    it 'removes assignee' do
+      put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+        assignee_ids: [0]
+
+      expect(response).to have_http_status(200)
+
+      expect(json_response['assignees']).to be_empty
+    end
+
+    it 'updates an issue with new assignee' do
+      put api("/projects/#{project.id}/issues/#{issue.iid}", user),
+        assignee_ids: [user2.id]
+
+      expect(response).to have_http_status(200)
+
+      expect(json_response['assignees'].first['name']).to eq(user2.name)
+    end
+  end
+
   describe 'PUT /projects/:id/issues/:issue_iid to update labels' do
     let!(:label) { create(:label, title: 'dummy', project: project) }
     let!(:label_link) { create(:label_link, label: label, target: issue) }
diff --git a/spec/requests/api/v3/issues_spec.rb b/spec/requests/api/v3/issues_spec.rb
index 2be1c8e29c705..c91a370722bed 100644
--- a/spec/requests/api/v3/issues_spec.rb
+++ b/spec/requests/api/v3/issues_spec.rb
@@ -14,7 +14,7 @@
   let!(:closed_issue) do
     create :closed_issue,
            author: user,
-           assignee: user,
+           assignees: [user],
            project: project,
            state: :closed,
            milestone: milestone,
@@ -26,14 +26,14 @@
            :confidential,
            project: project,
            author: author,
-           assignee: assignee,
+           assignees: [assignee],
            created_at: generate(:past_time),
            updated_at: 2.hours.ago
   end
   let!(:issue) do
     create :issue,
            author: user,
-           assignee: user,
+           assignees: [user],
            project: project,
            milestone: milestone,
            created_at: generate(:past_time),
@@ -247,7 +247,7 @@
     let!(:group_closed_issue) do
       create :closed_issue,
              author: user,
-             assignee: user,
+             assignees: [user],
              project: group_project,
              state: :closed,
              milestone: group_milestone,
@@ -258,13 +258,13 @@
              :confidential,
              project: group_project,
              author: author,
-             assignee: assignee,
+             assignees: [assignee],
              updated_at: 2.hours.ago
     end
     let!(:group_issue) do
       create :issue,
              author: user,
-             assignee: user,
+             assignees: [user],
              project: group_project,
              milestone: group_milestone,
              updated_at: 1.hour.ago
@@ -738,7 +738,7 @@
   describe "POST /projects/:id/issues" do
     it 'creates a new project issue' do
       post v3_api("/projects/#{project.id}/issues", user),
-        title: 'new issue', labels: 'label, label2', weight: 3
+        title: 'new issue', labels: 'label, label2', weight: 3, assignee_id: assignee.id
 
       expect(response).to have_http_status(201)
       expect(json_response['title']).to eq('new issue')
@@ -746,6 +746,7 @@
       expect(json_response['labels']).to eq(%w(label label2))
       expect(json_response['confidential']).to be_falsy
       expect(json_response['weight']).to eq(3)
+      expect(json_response['assignee']['name']).to eq(assignee.name)
     end
 
     it 'creates a new confidential project issue' do
@@ -1142,6 +1143,22 @@
     end
   end
 
+  describe 'PUT /projects/:id/issues/:issue_id to update assignee' do
+    it 'updates an issue with no assignee' do
+      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), assignee_id: 0
+
+      expect(response).to have_http_status(200)
+      expect(json_response['assignee']).to eq(nil)
+    end
+
+    it 'updates an issue with assignee' do
+      put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), assignee_id: user2.id
+
+      expect(response).to have_http_status(200)
+      expect(json_response['assignee']['name']).to eq(user2.name)
+    end
+  end
+
   describe 'PUT /projects/:id/issues/:issue_id to update weight' do
     it 'updates an issue with no weight' do
       put v3_api("/projects/#{project.id}/issues/#{issue.id}", user), weight: 5
diff --git a/spec/requests/projects/artifacts_controller_spec.rb b/spec/requests/projects/artifacts_controller_spec.rb
deleted file mode 100644
index d20866c0d44d4..0000000000000
--- a/spec/requests/projects/artifacts_controller_spec.rb
+++ /dev/null
@@ -1,117 +0,0 @@
-require 'spec_helper'
-
-describe Projects::ArtifactsController do
-  let(:user) { create(:user) }
-  let(:project) { create(:project, :repository) }
-
-  let(:pipeline) do
-    create(:ci_pipeline,
-            project: project,
-            sha: project.commit.sha,
-            ref: project.default_branch,
-            status: 'success')
-  end
-
-  let(:build) { create(:ci_build, :success, :artifacts, pipeline: pipeline) }
-
-  describe 'GET /:project/builds/artifacts/:ref_name/browse?job=name' do
-    before do
-      project.team << [user, :developer]
-
-      login_as(user)
-    end
-
-    def path_from_ref(
-      ref = pipeline.ref, job = build.name, path = 'browse')
-      latest_succeeded_namespace_project_artifacts_path(
-        project.namespace,
-        project,
-        [ref, path].join('/'),
-        job: job)
-    end
-
-    context 'cannot find the build' do
-      shared_examples 'not found' do
-        it { expect(response).to have_http_status(:not_found) }
-      end
-
-      context 'has no such ref' do
-        before do
-          get path_from_ref('TAIL', build.name)
-        end
-
-        it_behaves_like 'not found'
-      end
-
-      context 'has no such build' do
-        before do
-          get path_from_ref(pipeline.ref, 'NOBUILD')
-        end
-
-        it_behaves_like 'not found'
-      end
-
-      context 'has no path' do
-        before do
-          get path_from_ref(pipeline.sha, build.name, '')
-        end
-
-        it_behaves_like 'not found'
-      end
-    end
-
-    context 'found the build and redirect' do
-      shared_examples 'redirect to the build' do
-        it 'redirects' do
-          path = browse_namespace_project_build_artifacts_path(
-            project.namespace,
-            project,
-            build)
-
-          expect(response).to redirect_to(path)
-        end
-      end
-
-      context 'with regular branch' do
-        before do
-          pipeline.update(ref: 'master',
-                          sha: project.commit('master').sha)
-
-          get path_from_ref('master')
-        end
-
-        it_behaves_like 'redirect to the build'
-      end
-
-      context 'with branch name containing slash' do
-        before do
-          pipeline.update(ref: 'improve/awesome',
-                          sha: project.commit('improve/awesome').sha)
-
-          get path_from_ref('improve/awesome')
-        end
-
-        it_behaves_like 'redirect to the build'
-      end
-
-      context 'with branch name and path containing slashes' do
-        before do
-          pipeline.update(ref: 'improve/awesome',
-                          sha: project.commit('improve/awesome').sha)
-
-          get path_from_ref('improve/awesome', build.name, 'file/README.md')
-        end
-
-        it 'redirects' do
-          path = file_namespace_project_build_artifacts_path(
-            project.namespace,
-            project,
-            build,
-            'README.md')
-
-          expect(response).to redirect_to(path)
-        end
-      end
-    end
-  end
-end
diff --git a/spec/serializers/deploy_key_entity_spec.rb b/spec/serializers/deploy_key_entity_spec.rb
new file mode 100644
index 0000000000000..e73fbe190ca18
--- /dev/null
+++ b/spec/serializers/deploy_key_entity_spec.rb
@@ -0,0 +1,38 @@
+require 'spec_helper'
+
+describe DeployKeyEntity do
+  include RequestAwareEntity
+  
+  let(:user) { create(:user) }
+  let(:project) { create(:empty_project, :internal)}
+  let(:project_private) { create(:empty_project, :private)}
+  let(:deploy_key) { create(:deploy_key) }
+  let!(:deploy_key_internal) { create(:deploy_keys_project, project: project, deploy_key: deploy_key) }
+  let!(:deploy_key_private)  { create(:deploy_keys_project, project: project_private, deploy_key: deploy_key) }
+
+  let(:entity) { described_class.new(deploy_key, user: user) }
+
+  it 'returns deploy keys with projects a user can read' do
+    expected_result = {
+      id: deploy_key.id,
+      user_id: deploy_key.user_id,
+      title: deploy_key.title,
+      fingerprint: deploy_key.fingerprint,
+      can_push: deploy_key.can_push,
+      destroyed_when_orphaned: true,
+      almost_orphaned: false,
+      created_at: deploy_key.created_at,
+      updated_at: deploy_key.updated_at,
+      projects: [
+        {
+          id: project.id,
+          name: project.name,
+          full_path: namespace_project_path(project.namespace, project),
+          full_name: project.full_name
+        }
+      ]
+    }
+
+    expect(entity.as_json).to eq(expected_result)
+  end
+end
diff --git a/spec/serializers/label_serializer_spec.rb b/spec/serializers/label_serializer_spec.rb
new file mode 100644
index 0000000000000..c58c7da1f9ed5
--- /dev/null
+++ b/spec/serializers/label_serializer_spec.rb
@@ -0,0 +1,46 @@
+require 'spec_helper'
+
+describe LabelSerializer do
+  let(:user) { create(:user) }
+
+  let(:serializer) do
+    described_class.new(user: user)
+  end
+
+  subject { serializer.represent(resource) }
+
+  describe '#represent' do
+    context 'when a single object is being serialized' do
+      let(:resource) { create(:label) }
+
+      it 'serializes the label object' do
+        expect(subject[:id]).to eq resource.id
+      end
+    end
+
+    context 'when multiple objects are being serialized' do
+      let(:num_labels) { 2 }
+      let(:resource) { create_list(:label, num_labels) }
+
+      it 'serializes the array of labels' do
+        expect(subject.size).to eq(num_labels)
+      end
+    end
+  end
+
+  describe '#represent_appearance' do
+    context 'when represents only appearance' do
+      let(:resource) { create(:label) }
+
+      subject { serializer.represent_appearance(resource) }
+
+      it 'serializes only attributes used for appearance' do
+        expect(subject.keys).to eq([:id, :title, :color, :text_color])
+        expect(subject[:id]).to eq(resource.id)
+        expect(subject[:title]).to eq(resource.title)
+        expect(subject[:color]).to eq(resource.color)
+        expect(subject[:text_color]).to eq(resource.text_color)
+      end
+    end
+  end
+end
diff --git a/spec/services/issuable/bulk_update_service_spec.rb b/spec/services/issuable/bulk_update_service_spec.rb
index 7a1ac02731075..5b1639ca0d6e5 100644
--- a/spec/services/issuable/bulk_update_service_spec.rb
+++ b/spec/services/issuable/bulk_update_service_spec.rb
@@ -4,11 +4,12 @@
   let(:user)    { create(:user) }
   let(:project) { create(:empty_project, namespace: user.namespace) }
 
-  def bulk_update(issues, extra_params = {})
+  def bulk_update(issuables, extra_params = {})
     bulk_update_params = extra_params
-      .reverse_merge(issuable_ids: Array(issues).map(&:id).join(','))
+      .reverse_merge(issuable_ids: Array(issuables).map(&:id).join(','))
 
-    Issuable::BulkUpdateService.new(project, user, bulk_update_params).execute('issue')
+    type = Array(issuables).first.model_name.param_key
+    Issuable::BulkUpdateService.new(project, user, bulk_update_params).execute(type)
   end
 
   describe 'close issues' do
@@ -47,15 +48,15 @@ def bulk_update(issues, extra_params = {})
     end
   end
 
-  describe 'updating assignee' do
-    let(:issue) { create(:issue, project: project, assignee: user) }
+  describe 'updating merge request assignee' do
+    let(:merge_request) { create(:merge_request, target_project: project, source_project: project, assignee: user) }
 
     context 'when the new assignee ID is a valid user' do
       it 'succeeds' do
         new_assignee = create(:user)
         project.team << [new_assignee, :developer]
 
-        result = bulk_update(issue, assignee_id: new_assignee.id)
+        result = bulk_update(merge_request, assignee_id: new_assignee.id)
 
         expect(result[:success]).to be_truthy
         expect(result[:count]).to eq(1)
@@ -65,22 +66,59 @@ def bulk_update(issues, extra_params = {})
         assignee = create(:user)
         project.team << [assignee, :developer]
 
-        expect { bulk_update(issue, assignee_id: assignee.id) }
-          .to change { issue.reload.assignee }.from(user).to(assignee)
+        expect { bulk_update(merge_request, assignee_id: assignee.id) }
+          .to change { merge_request.reload.assignee }.from(user).to(assignee)
       end
     end
 
     context "when the new assignee ID is #{IssuableFinder::NONE}" do
       it "unassigns the issues" do
-        expect { bulk_update(issue, assignee_id: IssuableFinder::NONE) }
-          .to change { issue.reload.assignee }.to(nil)
+        expect { bulk_update(merge_request, assignee_id: IssuableFinder::NONE) }
+          .to change { merge_request.reload.assignee }.to(nil)
       end
     end
 
     context 'when the new assignee ID is not present' do
       it 'does not unassign' do
-        expect { bulk_update(issue, assignee_id: nil) }
-          .not_to change { issue.reload.assignee }
+        expect { bulk_update(merge_request, assignee_id: nil) }
+          .not_to change { merge_request.reload.assignee }
+      end
+    end
+  end
+
+  describe 'updating issue assignee' do
+    let(:issue) { create(:issue, project: project, assignees: [user]) }
+
+    context 'when the new assignee ID is a valid user' do
+      it 'succeeds' do
+        new_assignee = create(:user)
+        project.team << [new_assignee, :developer]
+
+        result = bulk_update(issue, assignee_ids: [new_assignee.id])
+
+        expect(result[:success]).to be_truthy
+        expect(result[:count]).to eq(1)
+      end
+
+      it 'updates the assignee to the use ID passed' do
+        assignee = create(:user)
+        project.team << [assignee, :developer]
+        expect { bulk_update(issue, assignee_ids: [assignee.id]) }
+          .to change { issue.reload.assignees.first }.from(user).to(assignee)
+      end
+    end
+
+    context "when the new assignee ID is #{IssuableFinder::NONE}" do
+      it "unassigns the issues" do
+        expect { bulk_update(issue, assignee_ids: [IssuableFinder::NONE.to_s]) }
+          .to change { issue.reload.assignees.count }.from(1).to(0)
+      end
+    end
+
+    context 'when the new assignee ID is not present' do
+      it 'does not unassign' do
+        expect { bulk_update(issue, assignee_ids: []) }
+          .not_to change{ issue.reload.assignees }
       end
     end
   end
diff --git a/spec/services/issues/close_service_spec.rb b/spec/services/issues/close_service_spec.rb
index 7a54373963e54..5184053171186 100644
--- a/spec/services/issues/close_service_spec.rb
+++ b/spec/services/issues/close_service_spec.rb
@@ -4,7 +4,7 @@
   let(:user) { create(:user) }
   let(:user2) { create(:user) }
   let(:guest) { create(:user) }
-  let(:issue) { create(:issue, assignee: user2) }
+  let(:issue) { create(:issue, assignees: [user2]) }
   let(:project) { issue.project }
   let!(:todo) { create(:todo, :assigned, user: user, project: project, target: issue, author: user2) }
 
diff --git a/spec/services/issues/create_service_spec.rb b/spec/services/issues/create_service_spec.rb
index 80bfb7315505a..01edc46496d97 100644
--- a/spec/services/issues/create_service_spec.rb
+++ b/spec/services/issues/create_service_spec.rb
@@ -6,10 +6,10 @@
 
   describe '#execute' do
     let(:issue) { described_class.new(project, user, opts).execute }
+    let(:assignee) { create(:user) }
+    let(:milestone) { create(:milestone, project: project) }
 
     context 'when params are valid' do
-      let(:assignee) { create(:user) }
-      let(:milestone) { create(:milestone, project: project) }
       let(:labels) { create_pair(:label, project: project) }
 
       before do
@@ -20,7 +20,7 @@
       let(:opts) do
         { title: 'Awesome issue',
           description: 'please fix',
-          assignee_id: assignee.id,
+          assignee_ids: [assignee.id],
           label_ids: labels.map(&:id),
           milestone_id: milestone.id,
           due_date: Date.tomorrow }
@@ -29,7 +29,7 @@
       it 'creates the issue with the given params' do
         expect(issue).to be_persisted
         expect(issue.title).to eq('Awesome issue')
-        expect(issue.assignee).to eq assignee
+        expect(issue.assignees).to eq [assignee]
         expect(issue.labels).to match_array labels
         expect(issue.milestone).to eq milestone
         expect(issue.due_date).to eq Date.tomorrow
@@ -37,6 +37,7 @@
 
       context 'when current user cannot admin issues in the project' do
         let(:guest) { create(:user) }
+
         before do
           project.team << [guest, :guest]
         end
@@ -47,7 +48,7 @@
           expect(issue).to be_persisted
           expect(issue.title).to eq('Awesome issue')
           expect(issue.description).to eq('please fix')
-          expect(issue.assignee).to be_nil
+          expect(issue.assignees).to be_empty
           expect(issue.labels).to be_empty
           expect(issue.milestone).to be_nil
           expect(issue.due_date).to be_nil
@@ -136,10 +137,83 @@
       end
     end
 
-    it_behaves_like 'issuable create service'
+    context 'issue create service' do
+      context 'assignees' do
+        before { project.team << [user, :master] }
+
+        it 'removes assignee when user id is invalid' do
+          opts = { title: 'Title', description: 'Description', assignee_ids: [-1] }
+
+          issue = described_class.new(project, user, opts).execute
+
+          expect(issue.assignees).to be_empty
+        end
+
+        it 'removes assignee when user id is 0' do
+          opts = { title: 'Title', description: 'Description',  assignee_ids: [0] }
+
+          issue = described_class.new(project, user, opts).execute
+
+          expect(issue.assignees).to be_empty
+        end
+
+        it 'saves assignee when user id is valid' do
+          project.team << [assignee, :master]
+          opts = { title: 'Title', description: 'Description', assignee_ids: [assignee.id] }
+
+          issue = described_class.new(project, user, opts).execute
+
+          expect(issue.assignees).to eq([assignee])
+        end
+
+        context "when issuable feature is private" do
+          before do
+            project.project_feature.update(issues_access_level: ProjectFeature::PRIVATE,
+                                           merge_requests_access_level: ProjectFeature::PRIVATE)
+          end
+
+          levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
+
+          levels.each do |level|
+            it "removes not authorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
+              project.update(visibility_level: level)
+              opts = { title: 'Title', description: 'Description', assignee_ids: [assignee.id] }
+
+              issue = described_class.new(project, user, opts).execute
+
+              expect(issue.assignees).to be_empty
+            end
+          end
+        end
+      end
+    end
 
     it_behaves_like 'new issuable record that supports slash commands'
 
+    context 'Slash commands' do
+      context 'with assignee and milestone in params and command' do
+        let(:opts) do
+          {
+            assignee_ids: [create(:user).id],
+            milestone_id: 1,
+            title: 'Title',
+            description: %(/assign @#{assignee.username}\n/milestone %"#{milestone.name}")
+          }
+        end
+
+        before do
+          project.team << [user, :master]
+          project.team << [assignee, :master]
+        end
+
+        it 'assigns and sets milestone to issuable from command' do
+          expect(issue).to be_persisted
+          expect(issue.assignees).to eq([assignee])
+          expect(issue.milestone).to eq(milestone)
+        end
+      end
+    end
+
     context 'resolving discussions' do
       let(:discussion) { create(:diff_note_on_merge_request).to_discussion }
       let(:merge_request) { discussion.noteable }
diff --git a/spec/services/issues/export_csv_service_spec.rb b/spec/services/issues/export_csv_service_spec.rb
index 6f8d6d87dc66d..f09c8b48a697f 100644
--- a/spec/services/issues/export_csv_service_spec.rb
+++ b/spec/services/issues/export_csv_service_spec.rb
@@ -33,7 +33,7 @@ def csv
 
     before do
       issue.update!(milestone: milestone,
-                    assignee: user,
+                    assignees: [user],
                     description: 'Issue with details',
                     state: :reopened,
                     due_date: DateTime.new(2014, 3, 2),
@@ -71,11 +71,11 @@ def csv
     end
 
     specify 'assignee name' do
-      expect(csv[0]['Assignee']).to eq issue.assignee_name
+      expect(csv[0]['Assignee']).to eq user.name
     end
 
     specify 'assignee username' do
-      expect(csv[0]['Assignee Username']).to eq issue.assignee.username
+      expect(csv[0]['Assignee Username']).to eq user.username
     end
 
     specify 'confidential' do
diff --git a/spec/services/issues/update_service_spec.rb b/spec/services/issues/update_service_spec.rb
index 5b324f3c706d6..1954d8739f69b 100644
--- a/spec/services/issues/update_service_spec.rb
+++ b/spec/services/issues/update_service_spec.rb
@@ -14,7 +14,7 @@
   let(:issue) do
     create(:issue, title: 'Old title',
                    description: "for #{user2.to_reference}",
-                   assignee_id: user3.id,
+                   assignee_ids: [user3.id],
                    project: project)
   end
 
@@ -40,7 +40,7 @@ def update_issue(opts)
         {
           title: 'New title',
           description: 'Also please fix',
-          assignee_id: user2.id,
+          assignee_ids: [user2.id],
           state_event: 'close',
           label_ids: [label.id],
           due_date: Date.tomorrow
@@ -53,15 +53,15 @@ def update_issue(opts)
         expect(issue).to be_valid
         expect(issue.title).to eq 'New title'
         expect(issue.description).to eq 'Also please fix'
-        expect(issue.assignee).to eq user2
+        expect(issue.assignees).to match_array([user2])
         expect(issue).to be_closed
         expect(issue.labels).to match_array [label]
         expect(issue.due_date).to eq Date.tomorrow
       end
 
       it 'sorts issues as specified by parameters' do
-        issue1 = create(:issue, project: project, assignee_id: user3.id)
-        issue2 = create(:issue, project: project, assignee_id: user3.id)
+        issue1 = create(:issue, project: project, assignees: [user3])
+        issue2 = create(:issue, project: project, assignees: [user3])
 
         [issue, issue1, issue2].each do |issue|
           issue.move_to_end
@@ -87,7 +87,7 @@ def update_issue(opts)
           expect(issue).to be_valid
           expect(issue.title).to eq 'New title'
           expect(issue.description).to eq 'Also please fix'
-          expect(issue.assignee).to eq user3
+          expect(issue.assignees).to match_array [user3]
           expect(issue.labels).to be_empty
           expect(issue.milestone).to be_nil
           expect(issue.due_date).to be_nil
@@ -132,12 +132,23 @@ def update_issue(opts)
       end
     end
 
+    context 'when description changed' do
+      it 'creates system note about description change' do
+        update_issue(description: 'Changed description')
+
+        note = find_note('changed the description')
+
+        expect(note).not_to be_nil
+        expect(note.note).to eq('changed the description')
+      end
+    end
+
     context 'when issue turns confidential' do
       let(:opts) do
         {
           title: 'New title',
           description: 'Also please fix',
-          assignee_id: user2.id,
+          assignee_ids: [user2],
           state_event: 'close',
           label_ids: [label.id],
           confidential: true
@@ -163,12 +174,12 @@ def update_issue(opts)
       it 'does not update assignee_id with unauthorized users' do
         project.update(visibility_level: Gitlab::VisibilityLevel::PUBLIC)
         update_issue(confidential: true)
-        non_member        = create(:user)
-        original_assignee = issue.assignee
+        non_member = create(:user)
+        original_assignees = issue.assignees
 
-        update_issue(assignee_id: non_member.id)
+        update_issue(assignee_ids: [non_member.id])
 
-        expect(issue.reload.assignee_id).to eq(original_assignee.id)
+        expect(issue.reload.assignees).to eq(original_assignees)
       end
     end
 
@@ -205,7 +216,7 @@ def update_issue(opts)
 
       context 'when is reassigned' do
         before do
-          update_issue(assignee: user2)
+          update_issue(assignees: [user2])
         end
 
         it 'marks previous assignee todos as done' do
@@ -408,6 +419,41 @@ def update_issue(opts)
       end
     end
 
+    context 'updating asssignee_id' do
+      it 'does not update assignee when assignee_id is invalid' do
+        update_issue(assignee_ids: [-1])
+
+        expect(issue.reload.assignees).to eq([user3])
+      end
+
+      it 'unassigns assignee when user id is 0' do
+        update_issue(assignee_ids: [0])
+
+        expect(issue.reload.assignees).to be_empty
+      end
+
+      it 'does not update assignee_id when user cannot read issue' do
+        update_issue(assignee_ids: [create(:user).id])
+
+        expect(issue.reload.assignees).to eq([user3])
+      end
+
+      context "when issuable feature is private" do
+        levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
+
+        levels.each do |level|
+          it "does not update with unauthorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
+            assignee = create(:user)
+            project.update(visibility_level: level)
+            feature_visibility_attr = :"#{issue.model_name.plural}_access_level"
+            project.project_feature.update_attribute(feature_visibility_attr, ProjectFeature::PRIVATE)
+
+            expect{ update_issue(assignee_ids: [assignee.id]) }.not_to change{ issue.assignees }
+          end
+        end
+      end
+    end
+
     context 'updating mentions' do
       let(:mentionable) { issue }
       include_examples 'updating mentions', Issues::UpdateService
diff --git a/spec/services/members/authorized_destroy_service_spec.rb b/spec/services/members/authorized_destroy_service_spec.rb
index 3b35a3b8e3a3a..ab440d18e9f68 100644
--- a/spec/services/members/authorized_destroy_service_spec.rb
+++ b/spec/services/members/authorized_destroy_service_spec.rb
@@ -14,8 +14,8 @@ def number_of_assigned_issuables(user)
     it "unassigns issues and merge requests" do
       group.add_developer(member_user)
 
-      issue = create :issue, project: group_project, assignee: member_user
-      create :issue, assignee: member_user
+      issue = create :issue, project: group_project, assignees: [member_user]
+      create :issue, assignees: [member_user]
       merge_request = create :merge_request, target_project: group_project, source_project: group_project, assignee: member_user
       create :merge_request, target_project: project, source_project: project, assignee: member_user
 
@@ -33,7 +33,7 @@ def number_of_assigned_issuables(user)
     it "unassigns issues and merge requests" do
       project.team << [member_user, :developer]
 
-      create :issue, project: project, assignee: member_user
+      create :issue, project: project, assignees: [member_user]
       create :merge_request, target_project: project, source_project: project, assignee: member_user
 
       member = project.members.find_by(user_id: member_user.id)
diff --git a/spec/services/merge_requests/assign_issues_service_spec.rb b/spec/services/merge_requests/assign_issues_service_spec.rb
index fe75757dd29c4..d3556020d4d06 100644
--- a/spec/services/merge_requests/assign_issues_service_spec.rb
+++ b/spec/services/merge_requests/assign_issues_service_spec.rb
@@ -15,14 +15,14 @@
     expect(service.assignable_issues.map(&:id)).to include(issue.id)
   end
 
-  it 'ignores issues already assigned to any user' do
-    issue.update!(assignee: create(:user))
+  it 'ignores issues the user cannot update assignee on' do
+    project.team.truncate
 
     expect(service.assignable_issues).to be_empty
   end
 
-  it 'ignores issues the user cannot update assignee on' do
-    project.team.truncate
+  it 'ignores issues already assigned to any user' do
+    issue.assignees = [create(:user)]
 
     expect(service.assignable_issues).to be_empty
   end
@@ -44,7 +44,7 @@
   end
 
   it 'assigns these to the merge request owner' do
-    expect { service.execute }.to change { issue.reload.assignee }.to(user)
+    expect { service.execute }.to change { issue.assignees.first }.to(user)
   end
 
   it 'ignores external issues' do
diff --git a/spec/services/merge_requests/create_service_spec.rb b/spec/services/merge_requests/create_service_spec.rb
index 0e16c7cc94bbd..ace82380cc999 100644
--- a/spec/services/merge_requests/create_service_spec.rb
+++ b/spec/services/merge_requests/create_service_spec.rb
@@ -84,7 +84,87 @@
       end
     end
 
-    it_behaves_like 'issuable create service'
+    context 'Slash commands' do
+      context 'with assignee and milestone in params and command' do
+        let(:merge_request) { described_class.new(project, user, opts).execute }
+        let(:milestone) { create(:milestone, project: project) }
+
+        let(:opts) do
+          {
+            assignee_id: create(:user).id,
+            milestone_id: 1,
+            title: 'Title',
+            description: %(/assign @#{assignee.username}\n/milestone %"#{milestone.name}"),
+            source_branch: 'feature',
+            target_branch: 'master'
+          }
+        end
+
+        before do
+          project.team << [user, :master]
+          project.team << [assignee, :master]
+        end
+
+        it 'assigns and sets milestone to issuable from command' do
+          expect(merge_request).to be_persisted
+          expect(merge_request.assignee).to eq(assignee)
+          expect(merge_request.milestone).to eq(milestone)
+        end
+      end
+    end
+
+    context 'merge request create service' do
+      context 'asssignee_id' do
+        let(:assignee) { create(:user) }
+
+        before { project.team << [user, :master] }
+
+        it 'removes assignee_id when user id is invalid' do
+          opts = { title: 'Title', description: 'Description', assignee_id: -1 }
+
+          merge_request = described_class.new(project, user, opts).execute
+
+          expect(merge_request.assignee_id).to be_nil
+        end
+
+        it 'removes assignee_id when user id is 0' do
+          opts = { title: 'Title', description: 'Description',  assignee_id: 0 }
+
+          merge_request = described_class.new(project, user, opts).execute
+
+          expect(merge_request.assignee_id).to be_nil
+        end
+
+        it 'saves assignee when user id is valid' do
+          project.team << [assignee, :master]
+          opts = { title: 'Title', description: 'Description', assignee_id: assignee.id }
+
+          merge_request = described_class.new(project, user, opts).execute
+
+          expect(merge_request.assignee).to eq(assignee)
+        end
+
+        context "when issuable feature is private" do
+          before do
+            project.project_feature.update(issues_access_level: ProjectFeature::PRIVATE,
+                                           merge_requests_access_level: ProjectFeature::PRIVATE)
+          end
+
+          levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
+
+          levels.each do |level|
+            it "removes not authorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
+              project.update(visibility_level: level)
+              opts = { title: 'Title', description: 'Description', assignee_id: assignee.id }
+
+              merge_request = described_class.new(project, user, opts).execute
+
+              expect(merge_request.assignee_id).to be_nil
+            end
+          end
+        end
+      end
+    end
 
     context 'while saving references to issues that the created merge request closes' do
       let(:first_issue) { create(:issue, project: project) }
diff --git a/spec/services/merge_requests/update_service_spec.rb b/spec/services/merge_requests/update_service_spec.rb
index b182d854ff077..77eca3be73364 100644
--- a/spec/services/merge_requests/update_service_spec.rb
+++ b/spec/services/merge_requests/update_service_spec.rb
@@ -102,6 +102,13 @@ def update_merge_request(opts)
         expect(note.note).to eq 'changed title from **{-Old-} title** to **{+New+} title**'
       end
 
+      it 'creates system note about description change' do
+        note = find_note('changed the description')
+
+        expect(note).not_to be_nil
+        expect(note.note).to eq('changed the description')
+      end
+
       it 'creates system note about branch change' do
         note = find_note('changed target')
 
@@ -475,6 +482,54 @@ def update_merge_request(opts)
       end
     end
 
+    context 'updating asssignee_id' do
+      it 'does not update assignee when assignee_id is invalid' do
+        merge_request.update(assignee_id: user.id)
+
+        update_merge_request(assignee_id: -1)
+
+        expect(merge_request.reload.assignee).to eq(user)
+      end
+
+      it 'unassigns assignee when user id is 0' do
+        merge_request.update(assignee_id: user.id)
+
+        update_merge_request(assignee_id: 0)
+
+        expect(merge_request.assignee_id).to be_nil
+      end
+
+      it 'saves assignee when user id is valid' do
+        update_merge_request(assignee_id: user.id)
+
+        expect(merge_request.assignee_id).to eq(user.id)
+      end
+
+      it 'does not update assignee_id when user cannot read issue' do
+        non_member        = create(:user)
+        original_assignee = merge_request.assignee
+
+        update_merge_request(assignee_id: non_member.id)
+
+        expect(merge_request.assignee_id).to eq(original_assignee.id)
+      end
+
+      context "when issuable feature is private" do
+        levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
+
+        levels.each do |level|
+          it "does not update with unauthorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
+            assignee = create(:user)
+            project.update(visibility_level: level)
+            feature_visibility_attr = :"#{merge_request.model_name.plural}_access_level"
+            project.project_feature.update_attribute(feature_visibility_attr, ProjectFeature::PRIVATE)
+
+            expect{ update_merge_request(assignee_id: assignee) }.not_to change{ merge_request.assignee }
+          end
+        end
+      end
+    end
+
     include_examples 'issuable update service' do
       let(:open_issuable) { merge_request }
       let(:closed_issuable) { create(:closed_merge_request, source_project: project) }
diff --git a/spec/services/notes/slash_commands_service_spec.rb b/spec/services/notes/slash_commands_service_spec.rb
index 1a64c8bbf00df..d837184382f5c 100644
--- a/spec/services/notes/slash_commands_service_spec.rb
+++ b/spec/services/notes/slash_commands_service_spec.rb
@@ -66,7 +66,7 @@
           expect(content).to eq ''
           expect(note.noteable).to be_closed
           expect(note.noteable.labels).to match_array(labels)
-          expect(note.noteable.assignee).to eq(assignee)
+          expect(note.noteable.assignees).to eq([assignee])
           expect(note.noteable.milestone).to eq(milestone)
         end
       end
@@ -113,7 +113,7 @@
           expect(content).to eq "HELLO\nWORLD"
           expect(note.noteable).to be_closed
           expect(note.noteable.labels).to match_array(labels)
-          expect(note.noteable.assignee).to eq(assignee)
+          expect(note.noteable.assignees).to eq([assignee])
           expect(note.noteable.milestone).to eq(milestone)
         end
       end
@@ -220,4 +220,31 @@
       let(:note) { build(:note_on_commit, project: project) }
     end
   end
+
+  context 'CE restriction for issue assignees' do
+    describe '/assign' do
+      let(:project) { create(:empty_project) }
+      let(:master) { create(:user).tap { |u| project.team << [u, :master] } }
+      let(:assignee) { create(:user) }
+      let(:master) { create(:user) }
+      let(:service) { described_class.new(project, master) }
+      let(:note) { create(:note_on_issue, note: note_text, project: project) }
+
+      let(:note_text) do
+        %(/assign @#{assignee.username} @#{master.username}\n")
+      end
+
+      before do
+        project.team << [master, :master]
+        project.team << [assignee, :master]
+      end
+
+      it 'adds only one assignee from the list' do
+        _, command_params = service.extract_commands(note)
+        service.execute(command_params, note)
+
+        expect(note.noteable.assignees.count).to eq(2)
+      end
+    end
+  end
 end
diff --git a/spec/services/notification_service_spec.rb b/spec/services/notification_service_spec.rb
index 817c31e72182f..5e8b54d2b6d4b 100644
--- a/spec/services/notification_service_spec.rb
+++ b/spec/services/notification_service_spec.rb
@@ -4,6 +4,7 @@
   include EmailHelpers
 
   let(:notification) { NotificationService.new }
+  let(:assignee) { create(:user) }
 
   around(:each) do |example|
     perform_enqueued_jobs do
@@ -52,7 +53,11 @@ def send_notifications(*new_mentions)
 
   shared_examples 'participating by assignee notification' do
     it 'emails the participant' do
-      issuable.update_attribute(:assignee, participant)
+      if issuable.is_a?(Issue)
+        issuable.assignees << participant
+      else
+        issuable.update_attribute(:assignee, participant)
+      end
 
       notification_trigger
 
@@ -103,14 +108,14 @@ def send_notifications(*new_mentions)
   describe 'Notes' do
     context 'issue note' do
       let(:project) { create(:empty_project, :private) }
-      let(:issue) { create(:issue, project: project, assignee: create(:user)) }
-      let(:mentioned_issue) { create(:issue, assignee: issue.assignee) }
+      let(:issue) { create(:issue, project: project, assignees: [assignee]) }
+      let(:mentioned_issue) { create(:issue, assignees: issue.assignees) }
       let(:note) { create(:note_on_issue, noteable: issue, project_id: issue.project_id, note: '@mention referenced, @outsider also') }
 
       before do
         build_team(note.project)
         project.add_master(issue.author)
-        project.add_master(issue.assignee)
+        project.add_master(assignee)
         project.add_master(note.author)
         create(:note_on_issue, noteable: issue, project_id: issue.project_id, note: '@subscribed_participant cc this guy')
         update_custom_notification(:new_note, @u_guest_custom, resource: project)
@@ -130,7 +135,7 @@ def send_notifications(*new_mentions)
 
           should_email(@u_watcher)
           should_email(note.noteable.author)
-          should_email(note.noteable.assignee)
+          should_email(note.noteable.assignees.first)
           should_email(@u_custom_global)
           should_email(@u_mentioned)
           should_email(@subscriber)
@@ -196,7 +201,7 @@ def send_notifications(*new_mentions)
           notification.new_note(note)
 
           should_email(note.noteable.author)
-          should_email(note.noteable.assignee)
+          should_email(note.noteable.assignees.first)
           should_email(@u_mentioned)
           should_email(@u_custom_global)
           should_not_email(@u_guest_custom)
@@ -218,7 +223,7 @@ def send_notifications(*new_mentions)
       let(:member) { create(:user) }
       let(:guest) { create(:user) }
       let(:admin) { create(:admin) }
-      let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee) }
+      let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee]) }
       let(:note) { create(:note_on_issue, noteable: confidential_issue, project: project, note: "#{author.to_reference} #{assignee.to_reference} #{non_member.to_reference} #{member.to_reference} #{admin.to_reference}") }
       let(:guest_watcher) { create_user_with_notification(:watch, "guest-watcher-confidential") }
 
@@ -244,8 +249,8 @@ def send_notifications(*new_mentions)
 
     context 'issue note mention' do
       let(:project) { create(:empty_project, :public) }
-      let(:issue) { create(:issue, project: project, assignee: create(:user)) }
-      let(:mentioned_issue) { create(:issue, assignee: issue.assignee) }
+      let(:issue) { create(:issue, project: project, assignees: [assignee]) }
+      let(:mentioned_issue) { create(:issue, assignees: issue.assignees) }
       let(:note) { create(:note_on_issue, noteable: issue, project_id: issue.project_id, note: '@all mentioned') }
 
       before do
@@ -269,7 +274,7 @@ def send_notifications(*new_mentions)
 
           should_email(@u_guest_watcher)
           should_email(note.noteable.author)
-          should_email(note.noteable.assignee)
+          should_email(note.noteable.assignees.first)
           should_not_email(note.author)
           should_email(@u_mentioned)
           should_not_email(@u_disabled)
@@ -449,7 +454,7 @@ def send_notifications(*new_mentions)
     let(:group) { create(:group) }
     let(:project) { create(:empty_project, :public, namespace: group) }
     let(:another_project) { create(:empty_project, :public, namespace: group) }
-    let(:issue) { create :issue, project: project, assignee: create(:user), description: 'cc @participant' }
+    let(:issue) { create :issue, project: project, assignees: [assignee], description: 'cc @participant' }
 
     before do
       build_team(issue.project)
@@ -465,7 +470,7 @@ def send_notifications(*new_mentions)
       it do
         notification.new_issue(issue, @u_disabled)
 
-        should_email(issue.assignee)
+        should_email(assignee)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
@@ -480,10 +485,10 @@ def send_notifications(*new_mentions)
       end
 
       it do
-        create_global_setting_for(issue.assignee, :mention)
+        create_global_setting_for(issue.assignees.first, :mention)
         notification.new_issue(issue, @u_disabled)
 
-        should_not_email(issue.assignee)
+        should_not_email(issue.assignees.first)
       end
 
       it "emails the author if they've opted into notifications about their activity" do
@@ -528,7 +533,7 @@ def send_notifications(*new_mentions)
         let(:member) { create(:user) }
         let(:guest) { create(:user) }
         let(:admin) { create(:admin) }
-        let(:confidential_issue) { create(:issue, :confidential, project: project, title: 'Confidential issue', author: author, assignee: assignee) }
+        let(:confidential_issue) { create(:issue, :confidential, project: project, title: 'Confidential issue', author: author, assignees: [assignee]) }
 
         it "emails subscribers of the issue's labels that can read the issue" do
           project.add_developer(member)
@@ -572,9 +577,9 @@ def send_notifications(*new_mentions)
       end
 
       it 'emails new assignee' do
-        notification.reassigned_issue(issue, @u_disabled)
+        notification.reassigned_issue(issue, @u_disabled, [assignee])
 
-        should_email(issue.assignee)
+        should_email(issue.assignees.first)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
@@ -588,9 +593,8 @@ def send_notifications(*new_mentions)
       end
 
       it 'emails previous assignee even if he has the "on mention" notif level' do
-        issue.update_attribute(:assignee, @u_mentioned)
-        issue.update_attributes(assignee: @u_watcher)
-        notification.reassigned_issue(issue, @u_disabled)
+        issue.assignees = [@u_mentioned]
+        notification.reassigned_issue(issue, @u_disabled, [@u_watcher])
 
         should_email(@u_mentioned)
         should_email(@u_watcher)
@@ -606,11 +610,11 @@ def send_notifications(*new_mentions)
       end
 
       it 'emails new assignee even if he has the "on mention" notif level' do
-        issue.update_attributes(assignee: @u_mentioned)
-        notification.reassigned_issue(issue, @u_disabled)
+        issue.assignees = [@u_mentioned]
+        notification.reassigned_issue(issue, @u_disabled, [@u_mentioned])
 
-        expect(issue.assignee).to be @u_mentioned
-        should_email(issue.assignee)
+        expect(issue.assignees.first).to be @u_mentioned
+        should_email(issue.assignees.first)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
@@ -624,11 +628,11 @@ def send_notifications(*new_mentions)
       end
 
       it 'emails new assignee' do
-        issue.update_attribute(:assignee, @u_mentioned)
-        notification.reassigned_issue(issue, @u_disabled)
+        issue.assignees = [@u_mentioned]
+        notification.reassigned_issue(issue, @u_disabled, [@u_mentioned])
 
-        expect(issue.assignee).to be @u_mentioned
-        should_email(issue.assignee)
+        expect(issue.assignees.first).to be @u_mentioned
+        should_email(issue.assignees.first)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
@@ -642,17 +646,17 @@ def send_notifications(*new_mentions)
       end
 
       it 'does not email new assignee if they are the current user' do
-        issue.update_attribute(:assignee, @u_mentioned)
-        notification.reassigned_issue(issue, @u_mentioned)
+        issue.assignees = [@u_mentioned]
+        notification.reassigned_issue(issue, @u_mentioned, [@u_mentioned])
 
-        expect(issue.assignee).to be @u_mentioned
+        expect(issue.assignees.first).to be @u_mentioned
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
         should_email(@u_guest_custom)
         should_email(@u_participant_mentioned)
         should_email(@subscriber)
         should_email(@u_custom_global)
-        should_not_email(issue.assignee)
+        should_not_email(issue.assignees.first)
         should_not_email(@unsubscriber)
         should_not_email(@u_participating)
         should_not_email(@u_disabled)
@@ -662,7 +666,7 @@ def send_notifications(*new_mentions)
       it_behaves_like 'participating notifications' do
         let(:participant) { create(:user, username: 'user-participant') }
         let(:issuable) { issue }
-        let(:notification_trigger) { notification.reassigned_issue(issue, @u_disabled) }
+        let(:notification_trigger) { notification.reassigned_issue(issue, @u_disabled, [assignee]) }
       end
     end
 
@@ -705,7 +709,7 @@ def send_notifications(*new_mentions)
       it "doesn't send email to anyone but subscribers of the given labels" do
         notification.relabeled_issue(issue, [group_label_2, label_2], @u_disabled)
 
-        should_not_email(issue.assignee)
+        should_not_email(issue.assignees.first)
         should_not_email(issue.author)
         should_not_email(@u_watcher)
         should_not_email(@u_guest_watcher)
@@ -729,7 +733,7 @@ def send_notifications(*new_mentions)
         let(:member) { create(:user) }
         let(:guest) { create(:user) }
         let(:admin) { create(:admin) }
-        let(:confidential_issue) { create(:issue, :confidential, project: project, title: 'Confidential issue', author: author, assignee: assignee) }
+        let(:confidential_issue) { create(:issue, :confidential, project: project, title: 'Confidential issue', author: author, assignees: [assignee]) }
         let!(:label_1) { create(:label, project: project, issues: [confidential_issue]) }
         let!(:label_2) { create(:label, project: project) }
 
@@ -767,7 +771,7 @@ def send_notifications(*new_mentions)
       it 'sends email to issue assignee and issue author' do
         notification.close_issue(issue, @u_disabled)
 
-        should_email(issue.assignee)
+        should_email(issue.assignees.first)
         should_email(issue.author)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
@@ -798,7 +802,7 @@ def send_notifications(*new_mentions)
       it 'sends email to issue notification recipients' do
         notification.reopen_issue(issue, @u_disabled)
 
-        should_email(issue.assignee)
+        should_email(issue.assignees.first)
         should_email(issue.author)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
@@ -826,7 +830,7 @@ def send_notifications(*new_mentions)
       it 'sends email to issue notification recipients' do
         notification.issue_moved(issue, new_issue, @u_disabled)
 
-        should_email(issue.assignee)
+        should_email(issue.assignees.first)
         should_email(issue.author)
         should_email(@u_watcher)
         should_email(@u_guest_watcher)
diff --git a/spec/services/projects/autocomplete_service_spec.rb b/spec/services/projects/autocomplete_service_spec.rb
index 7916c2d957cc0..c198c3eedfca2 100644
--- a/spec/services/projects/autocomplete_service_spec.rb
+++ b/spec/services/projects/autocomplete_service_spec.rb
@@ -11,7 +11,7 @@
       let(:project) { create(:empty_project, :public) }
       let!(:issue) { create(:issue, project: project, title: 'Issue 1') }
       let!(:security_issue_1) { create(:issue, :confidential, project: project, title: 'Security issue 1', author: author) }
-      let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignee: assignee) }
+      let!(:security_issue_2) { create(:issue, :confidential, title: 'Security issue 2', project: project, assignees: [assignee]) }
 
       it 'does not list project confidential issues for guests' do
         autocomplete = described_class.new(project, nil)
diff --git a/spec/services/projects/propagate_service_template_spec.rb b/spec/services/projects/propagate_service_template_spec.rb
new file mode 100644
index 0000000000000..90eff3bbc1e0b
--- /dev/null
+++ b/spec/services/projects/propagate_service_template_spec.rb
@@ -0,0 +1,103 @@
+require 'spec_helper'
+
+describe Projects::PropagateServiceTemplate, services: true do
+  describe '.propagate' do
+    let!(:service_template) do
+      PushoverService.create(
+        template: true,
+        active: true,
+        properties: {
+          device: 'MyDevice',
+          sound: 'mic',
+          priority: 4,
+          user_key: 'asdf',
+          api_key: '123456789'
+        })
+    end
+
+    let!(:project) { create(:empty_project) }
+
+    it 'creates services for projects' do
+      expect(project.pushover_service).to be_nil
+
+      described_class.propagate(service_template)
+
+      expect(project.reload.pushover_service).to be_present
+    end
+
+    it 'creates services for a project that has another service' do
+      BambooService.create(
+        template: true,
+        active: true,
+        project: project,
+        properties: {
+          bamboo_url: 'http://gitlab.com',
+          username: 'mic',
+          password: "password",
+          build_key: 'build'
+        }
+      )
+
+      expect(project.pushover_service).to be_nil
+
+      described_class.propagate(service_template)
+
+      expect(project.reload.pushover_service).to be_present
+    end
+
+    it 'does not create the service if it exists already' do
+      other_service = BambooService.create(
+        template: true,
+        active: true,
+        properties: {
+          bamboo_url: 'http://gitlab.com',
+          username: 'mic',
+          password: "password",
+          build_key: 'build'
+        }
+      )
+
+      Service.build_from_template(project.id, service_template).save!
+      Service.build_from_template(project.id, other_service).save!
+
+      expect { described_class.propagate(service_template) }.
+        not_to change { Service.count }
+    end
+
+    it 'creates the service containing the template attributes' do
+      described_class.propagate(service_template)
+
+      expect(project.pushover_service.properties).to eq(service_template.properties)
+    end
+
+    describe 'bulk update' do
+      it 'creates services for all projects' do
+        project_total = 5
+        stub_const 'Projects::PropagateServiceTemplate::BATCH_SIZE', 3
+
+        project_total.times { create(:empty_project) }
+
+        expect { described_class.propagate(service_template) }.
+          to change { Service.count }.by(project_total + 1)
+      end
+    end
+
+    describe 'external tracker' do
+      it 'updates the project external tracker' do
+        service_template.update!(category: 'issue_tracker', default: false)
+
+        expect { described_class.propagate(service_template) }.
+          to change { project.reload.has_external_issue_tracker }.to(true)
+      end
+    end
+
+    describe 'external wiki' do
+      it 'updates the project external tracker' do
+        service_template.update!(type: 'ExternalWikiService')
+
+        expect { described_class.propagate(service_template) }.
+          to change { project.reload.has_external_wiki }.to(true)
+      end
+    end
+  end
+end
diff --git a/spec/services/slash_commands/interpret_service_spec.rb b/spec/services/slash_commands/interpret_service_spec.rb
index bbe85f3127583..40427fc2173d9 100644
--- a/spec/services/slash_commands/interpret_service_spec.rb
+++ b/spec/services/slash_commands/interpret_service_spec.rb
@@ -3,6 +3,7 @@
 describe SlashCommands::InterpretService, services: true do
   let(:project) { create(:empty_project, :public) }
   let(:developer) { create(:user) }
+  let(:developer2) { create(:user) }
   let(:issue) { create(:issue, project: project) }
   let(:milestone) { create(:milestone, project: project, title: '9.10') }
   let(:inprogress) { create(:label, project: project, title: 'In Progress') }
@@ -42,23 +43,6 @@
       end
     end
 
-    shared_examples 'assign command' do
-      it 'fetches assignee and populates assignee_id if content contains /assign' do
-        _, updates = service.execute(content, issuable)
-
-        expect(updates).to eq(assignee_id: developer.id)
-      end
-    end
-
-    shared_examples 'unassign command' do
-      it 'populates assignee_id: nil if content contains /unassign' do
-        issuable.update!(assignee_id: developer.id)
-        _, updates = service.execute(content, issuable)
-
-        expect(updates).to eq(assignee_id: nil)
-      end
-    end
-
     shared_examples 'milestone command' do
       it 'fetches milestone and populates milestone_id if content contains /milestone' do
         milestone # populate the milestone
@@ -388,14 +372,46 @@
       let(:issuable) { issue }
     end
 
-    it_behaves_like 'assign command' do
+    context 'assign command' do
       let(:content) { "/assign @#{developer.username}" }
-      let(:issuable) { issue }
+
+      context 'Issue' do
+        it 'fetches assignee and populates assignee_id if content contains /assign' do
+          _, updates = service.execute(content, issue)
+
+          expect(updates).to eq(assignee_ids: [developer.id])
+        end
+      end
+
+      context 'Merge Request' do
+        it 'fetches assignee and populates assignee_id if content contains /assign' do
+          _, updates = service.execute(content, merge_request)
+
+          expect(updates).to eq(assignee_id: developer.id)
+        end
+      end
     end
 
-    it_behaves_like 'assign command' do
-      let(:content) { "/assign @#{developer.username}" }
-      let(:issuable) { merge_request }
+    context 'assign command with multiple assignees' do
+      let(:content) { "/assign @#{developer.username} @#{developer2.username}" }
+
+      before{ project.team << [developer2, :developer] }
+
+      context 'Issue' do
+        it 'fetches assignee and populates assignee_id if content contains /assign' do
+          _, updates = service.execute(content, issue)
+
+          expect(updates[:assignee_ids]).to match_array([developer.id, developer2.id])
+        end
+      end
+
+      context 'Merge Request' do
+        it 'fetches assignee and populates assignee_id if content contains /assign' do
+          _, updates = service.execute(content, merge_request)
+
+          expect(updates).to eq(assignee_id: developer.id)
+        end
+      end
     end
 
     it_behaves_like 'empty command' do
@@ -408,14 +424,26 @@
       let(:issuable) { issue }
     end
 
-    it_behaves_like 'unassign command' do
+    context 'unassign command' do
       let(:content) { '/unassign' }
-      let(:issuable) { issue }
-    end
 
-    it_behaves_like 'unassign command' do
-      let(:content) { '/unassign' }
-      let(:issuable) { merge_request }
+      context 'Issue' do
+        it 'populates assignee_ids: [] if content contains /unassign' do
+          issue.update(assignee_ids: [developer.id])
+          _, updates = service.execute(content, issue)
+
+          expect(updates).to eq(assignee_ids: [])
+        end
+      end
+
+      context 'Merge Request' do
+        it 'populates assignee_id: nil if content contains /unassign' do
+          merge_request.update(assignee_id: developer.id)
+          _, updates = service.execute(content, merge_request)
+
+          expect(updates).to eq(assignee_id: nil)
+        end
+      end
     end
 
     it_behaves_like 'milestone command' do
@@ -873,7 +901,7 @@
 
     describe 'unassign command' do
       let(:content) { '/unassign' }
-      let(:issue) { create(:issue, project: project, assignee: developer) }
+      let(:issue) { create(:issue, project: project, assignees: [developer]) }
 
       it 'includes current assignee reference' do
         _, explanations = service.explain(content, issue)
diff --git a/spec/services/system_note_service_spec.rb b/spec/services/system_note_service_spec.rb
index 7b883c636ae0b..2fdcd579f995c 100644
--- a/spec/services/system_note_service_spec.rb
+++ b/spec/services/system_note_service_spec.rb
@@ -6,6 +6,7 @@
   let(:project)  { create(:empty_project) }
   let(:author)   { create(:user) }
   let(:noteable) { create(:issue, project: project) }
+  let(:issue)    { noteable }
 
   shared_examples_for 'a system note' do
     let(:expected_noteable) { noteable }
@@ -155,6 +156,52 @@
     end
   end
 
+  describe '.change_issue_assignees' do
+    subject { described_class.change_issue_assignees(noteable, project, author, [assignee]) }
+
+    let(:assignee) { create(:user) }
+    let(:assignee1) { create(:user) }
+    let(:assignee2) { create(:user) }
+    let(:assignee3) { create(:user) }
+
+    it_behaves_like 'a system note' do
+      let(:action) { 'assignee' }
+    end
+
+    def build_note(old_assignees, new_assignees)
+      issue.assignees = new_assignees
+      described_class.change_issue_assignees(issue, project, author, old_assignees).note
+    end
+
+    it 'builds a correct phrase when an assignee is added to a non-assigned issue' do
+      expect(build_note([], [assignee1])).to eq "assigned to @#{assignee1.username}"
+    end
+
+    it 'builds a correct phrase when assignee removed' do
+      expect(build_note([assignee1], [])).to eq 'removed all assignees'
+    end
+
+    it 'builds a correct phrase when assignees changed' do
+      expect(build_note([assignee1], [assignee2])).to eq \
+        "assigned to @#{assignee2.username} and unassigned @#{assignee1.username}"
+    end
+
+    it 'builds a correct phrase when three assignees removed and one added' do
+      expect(build_note([assignee, assignee1, assignee2], [assignee3])).to eq \
+        "assigned to @#{assignee3.username} and unassigned @#{assignee.username}, @#{assignee1.username}, and @#{assignee2.username}"
+    end
+
+    it 'builds a correct phrase when one assignee changed from a set' do
+      expect(build_note([assignee, assignee1], [assignee, assignee2])).to eq \
+        "assigned to @#{assignee2.username} and unassigned @#{assignee1.username}"
+    end
+
+    it 'builds a correct phrase when one assignee removed from a set' do
+      expect(build_note([assignee, assignee1, assignee2], [assignee, assignee1])).to eq \
+        "unassigned @#{assignee2.username}"
+    end
+  end
+
   describe '.change_label' do
     subject { described_class.change_label(noteable, project, author, added, removed) }
 
@@ -292,6 +339,20 @@
     end
   end
 
+  describe '.change_description' do
+    subject { described_class.change_description(noteable, project, author) }
+
+    context 'when noteable responds to `description`' do
+      it_behaves_like 'a system note' do
+        let(:action) { 'description' }
+      end
+
+      it 'sets the note text' do
+        expect(subject.note).to eq('changed the description')
+      end
+    end
+  end
+
   describe '.change_issue_confidentiality' do
     subject { described_class.change_issue_confidentiality(noteable, project, author) }
 
diff --git a/spec/services/todo_service_spec.rb b/spec/services/todo_service_spec.rb
index 4639c93b441ae..09027e30bd3ea 100644
--- a/spec/services/todo_service_spec.rb
+++ b/spec/services/todo_service_spec.rb
@@ -25,11 +25,11 @@
   end
 
   describe 'Issues' do
-    let(:issue) { create(:issue, project: project, assignee: john_doe, author: author, description: "- [ ] Task 1\n- [ ] Task 2 #{mentions}") }
-    let(:addressed_issue) { create(:issue, project: project, assignee: john_doe, author: author, description: "#{directly_addressed}\n- [ ] Task 1\n- [ ] Task 2") }
-    let(:unassigned_issue) { create(:issue, project: project, assignee: nil) }
-    let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee, description: mentions) }
-    let(:addressed_confident_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee, description: directly_addressed) }
+    let(:issue) { create(:issue, project: project, assignees: [john_doe], author: author, description: "- [ ] Task 1\n- [ ] Task 2 #{mentions}") }
+    let(:addressed_issue) { create(:issue, project: project, assignees: [john_doe], author: author, description: "#{directly_addressed}\n- [ ] Task 1\n- [ ] Task 2") }
+    let(:unassigned_issue) { create(:issue, project: project, assignees: []) }
+    let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee], description: mentions) }
+    let(:addressed_confident_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee], description: directly_addressed) }
 
     describe '#new_issue' do
       it 'creates a todo if assigned' do
@@ -43,7 +43,7 @@
       end
 
       it 'creates a todo if assignee is the current user' do
-        unassigned_issue.update_attribute(:assignee, john_doe)
+        unassigned_issue.assignees = [john_doe]
         service.new_issue(unassigned_issue, john_doe)
 
         should_create_todo(user: john_doe, target: unassigned_issue, author: john_doe, action: Todo::ASSIGNED)
@@ -258,20 +258,20 @@
 
     describe '#reassigned_issue' do
       it 'creates a pending todo for new assignee' do
-        unassigned_issue.update_attribute(:assignee, john_doe)
+        unassigned_issue.assignees << john_doe
         service.reassigned_issue(unassigned_issue, author)
 
         should_create_todo(user: john_doe, target: unassigned_issue, action: Todo::ASSIGNED)
       end
 
       it 'does not create a todo if unassigned' do
-        issue.update_attribute(:assignee, nil)
+        issue.assignees.destroy_all
 
         should_not_create_any_todo { service.reassigned_issue(issue, author) }
       end
 
       it 'creates a todo if new assignee is the current user' do
-        unassigned_issue.update_attribute(:assignee, john_doe)
+        unassigned_issue.assignees << john_doe
         service.reassigned_issue(unassigned_issue, john_doe)
 
         should_create_todo(user: john_doe, target: unassigned_issue, author: john_doe, action: Todo::ASSIGNED)
@@ -361,7 +361,7 @@
     describe '#new_note' do
       let!(:first_todo) { create(:todo, :assigned, user: john_doe, project: project, target: issue, author: author) }
       let!(:second_todo) { create(:todo, :assigned, user: john_doe, project: project, target: issue, author: author) }
-      let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignee: assignee) }
+      let(:confidential_issue) { create(:issue, :confidential, project: project, author: author, assignees: [assignee]) }
       let(:note) { create(:note, project: project, noteable: issue, author: john_doe, note: mentions) }
       let(:addressed_note) { create(:note, project: project, noteable: issue, author: john_doe, note: directly_addressed) }
       let(:note_on_commit) { create(:note_on_commit, project: project, author: john_doe, note: mentions) }
@@ -884,7 +884,7 @@
   end
 
   it 'updates cached counts when a todo is created' do
-    issue = create(:issue, project: project, assignee: john_doe, author: author, description: mentions)
+    issue = create(:issue, project: project, assignees: [john_doe], author: author, description: mentions)
 
     expect(john_doe.todos_pending_count).to eq(0)
     expect(john_doe).to receive(:update_todos_count_cache).and_call_original
@@ -896,8 +896,8 @@
   end
 
   describe '#mark_todos_as_done' do
-    let(:issue) { create(:issue, project: project, author: author, assignee: john_doe) }
-    let(:another_issue) { create(:issue, project: project, author: author, assignee: john_doe) }
+    let(:issue) { create(:issue, project: project, author: author, assignees: [john_doe]) }
+    let(:another_issue) { create(:issue, project: project, author: author, assignees: [john_doe]) }
 
     it 'marks a relation of todos as done' do
       create(:todo, :mentioned, user: john_doe, target: issue, project: project)
diff --git a/spec/services/users/destroy_service_spec.rb b/spec/services/users/destroy_service_spec.rb
index 4bc30018ebd5d..de37a61e38880 100644
--- a/spec/services/users/destroy_service_spec.rb
+++ b/spec/services/users/destroy_service_spec.rb
@@ -47,7 +47,7 @@
       end
 
       context "for an issue the user was assigned to" do
-        let!(:issue) { create(:issue, project: project, assignee: user) }
+        let!(:issue) { create(:issue, project: project, assignees: [user]) }
 
         before do
           service.execute(user)
@@ -60,7 +60,7 @@
         it 'migrates the issue so that it is "Unassigned"' do
           migrated_issue = Issue.find_by_id(issue.id)
 
-          expect(migrated_issue.assignee).to be_nil
+          expect(migrated_issue.assignees).to be_empty
         end
       end
     end
diff --git a/spec/support/features/issuable_slash_commands_shared_examples.rb b/spec/support/features/issuable_slash_commands_shared_examples.rb
index 1d26c7baace36..ad46b163cd619 100644
--- a/spec/support/features/issuable_slash_commands_shared_examples.rb
+++ b/spec/support/features/issuable_slash_commands_shared_examples.rb
@@ -58,11 +58,12 @@
         expect(page).not_to have_content '/label ~bug'
         expect(page).not_to have_content '/milestone %"ASAP"'
 
+        wait_for_ajax
         issuable.reload
         note = issuable.notes.user.first
 
         expect(note.note).to eq "Awesome!"
-        expect(issuable.assignee).to eq assignee
+        expect(issuable.assignees).to eq [assignee]
         expect(issuable.labels).to eq [label_bug]
         expect(issuable.milestone).to eq milestone
       end
@@ -80,7 +81,7 @@
         issuable.reload
 
         expect(issuable.notes.user).to be_empty
-        expect(issuable.assignee).to eq assignee
+        expect(issuable.assignees).to eq [assignee]
         expect(issuable.labels).to eq [label_bug]
         expect(issuable.milestone).to eq milestone
       end
diff --git a/spec/support/import_export/export_file_helper.rb b/spec/support/import_export/export_file_helper.rb
index 944ea30656fc9..57b6abe12b7ca 100644
--- a/spec/support/import_export/export_file_helper.rb
+++ b/spec/support/import_export/export_file_helper.rb
@@ -10,7 +10,7 @@ def setup_project
 
     create(:release, project: project)
 
-    issue = create(:issue, assignee: user, project: project)
+    issue = create(:issue, assignees: [user], project: project)
     snippet = create(:project_snippet, project: project)
     label = create(:label, project: project)
     milestone = create(:milestone, project: project)
diff --git a/spec/support/prometheus_helpers.rb b/spec/support/prometheus_helpers.rb
index cc79b11616aca..a204365431b1e 100644
--- a/spec/support/prometheus_helpers.rb
+++ b/spec/support/prometheus_helpers.rb
@@ -33,6 +33,10 @@ def stub_prometheus_request(url, body: {}, status: 200)
       })
   end
 
+  def stub_prometheus_request_with_exception(url, exception_type)
+    WebMock.stub_request(:get, url).to_raise(exception_type)
+  end
+
   def stub_all_prometheus_requests(environment_slug, body: nil, status: 200)
     stub_prometheus_request(
       prometheus_query_url(prometheus_memory_query(environment_slug)),
diff --git a/spec/support/services/issuable_create_service_shared_examples.rb b/spec/support/services/issuable_create_service_shared_examples.rb
deleted file mode 100644
index 4f0c745b7ee48..0000000000000
--- a/spec/support/services/issuable_create_service_shared_examples.rb
+++ /dev/null
@@ -1,52 +0,0 @@
-shared_examples 'issuable create service' do
-  context 'asssignee_id' do
-    let(:assignee) { create(:user) }
-
-    before { project.team << [user, :master] }
-
-    it 'removes assignee_id when user id is invalid' do
-      opts = { title: 'Title', description: 'Description', assignee_id: -1 }
-
-      issuable = described_class.new(project, user, opts).execute
-
-      expect(issuable.assignee_id).to be_nil
-    end
-
-    it 'removes assignee_id when user id is 0' do
-      opts = { title: 'Title', description: 'Description',  assignee_id: 0 }
-
-      issuable = described_class.new(project, user, opts).execute
-
-      expect(issuable.assignee_id).to be_nil
-    end
-
-    it 'saves assignee when user id is valid' do
-      project.team << [assignee, :master]
-      opts = { title: 'Title', description: 'Description', assignee_id: assignee.id }
-
-      issuable = described_class.new(project, user, opts).execute
-
-      expect(issuable.assignee_id).to eq(assignee.id)
-    end
-
-    context "when issuable feature is private" do
-      before do
-        project.project_feature.update(issues_access_level: ProjectFeature::PRIVATE,
-                                       merge_requests_access_level: ProjectFeature::PRIVATE)
-      end
-
-      levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
-
-      levels.each do |level|
-        it "removes not authorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
-          project.update(visibility_level: level)
-          opts = { title: 'Title', description: 'Description', assignee_id: assignee.id }
-
-          issuable = described_class.new(project, user, opts).execute
-
-          expect(issuable.assignee_id).to be_nil
-        end
-      end
-    end
-  end
-end
diff --git a/spec/support/services/issuable_create_service_slash_commands_shared_examples.rb b/spec/support/services/issuable_create_service_slash_commands_shared_examples.rb
index 9e9cdf3e48b49..1dd3663b944fd 100644
--- a/spec/support/services/issuable_create_service_slash_commands_shared_examples.rb
+++ b/spec/support/services/issuable_create_service_slash_commands_shared_examples.rb
@@ -49,23 +49,7 @@
 
     it 'assigns and sets milestone to issuable' do
       expect(issuable).to be_persisted
-      expect(issuable.assignee).to eq(assignee)
-      expect(issuable.milestone).to eq(milestone)
-    end
-  end
-
-  context 'with assignee and milestone in params and command' do
-    let(:example_params) do
-      {
-        assignee: create(:user),
-        milestone_id: 1,
-        description: %(/assign @#{assignee.username}\n/milestone %"#{milestone.name}")
-      }
-    end
-
-    it 'assigns and sets milestone to issuable from command' do
-      expect(issuable).to be_persisted
-      expect(issuable.assignee).to eq(assignee)
+      expect(issuable.assignees).to eq([assignee])
       expect(issuable.milestone).to eq(milestone)
     end
   end
diff --git a/spec/support/services/issuable_update_service_shared_examples.rb b/spec/support/services/issuable_update_service_shared_examples.rb
index 49cea1e608cd9..8947f20562f8b 100644
--- a/spec/support/services/issuable_update_service_shared_examples.rb
+++ b/spec/support/services/issuable_update_service_shared_examples.rb
@@ -18,52 +18,4 @@ def update_issuable(opts)
       end
     end
   end
-
-  context 'asssignee_id' do
-    it 'does not update assignee when assignee_id is invalid' do
-      open_issuable.update(assignee_id: user.id)
-
-      update_issuable(assignee_id: -1)
-
-      expect(open_issuable.reload.assignee).to eq(user)
-    end
-
-    it 'unassigns assignee when user id is 0' do
-      open_issuable.update(assignee_id: user.id)
-
-      update_issuable(assignee_id: 0)
-
-      expect(open_issuable.assignee_id).to be_nil
-    end
-
-    it 'saves assignee when user id is valid' do
-      update_issuable(assignee_id: user.id)
-
-      expect(open_issuable.assignee_id).to eq(user.id)
-    end
-
-    it 'does not update assignee_id when user cannot read issue' do
-      non_member        = create(:user)
-      original_assignee = open_issuable.assignee
-
-      update_issuable(assignee_id: non_member.id)
-
-      expect(open_issuable.assignee_id).to eq(original_assignee.id)
-    end
-
-    context "when issuable feature is private" do
-      levels = [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]
-
-      levels.each do |level|
-        it "does not update with unauthorized assignee when project is #{Gitlab::VisibilityLevel.level_name(level)}" do
-          assignee = create(:user)
-          project.update(visibility_level: level)
-          feature_visibility_attr = :"#{open_issuable.model_name.plural}_access_level"
-          project.project_feature.update_attribute(feature_visibility_attr, ProjectFeature::PRIVATE)
-
-          expect{ update_issuable(assignee_id: assignee) }.not_to change{ open_issuable.assignee }
-        end
-      end
-    end
-  end
 end
diff --git a/spec/support/test_env.rb b/spec/support/test_env.rb
index a5f034461b50f..8dc513be495fb 100644
--- a/spec/support/test_env.rb
+++ b/spec/support/test_env.rb
@@ -28,6 +28,7 @@ module TestEnv
     'expand-collapse-files'              => '025db92',
     'expand-collapse-lines'              => '238e82d',
     'video'                              => '8879059',
+    'add-balsamiq-file'                  => 'b89b56d',
     'crlf-diff'                          => '5938907',
     'conflict-start'                     => '824be60',
     'conflict-resolvable'                => '1450cd6',
diff --git a/spec/support/time_tracking_shared_examples.rb b/spec/support/time_tracking_shared_examples.rb
index 01bc80f957e24..84ef46ffa27bf 100644
--- a/spec/support/time_tracking_shared_examples.rb
+++ b/spec/support/time_tracking_shared_examples.rb
@@ -8,6 +8,7 @@
   it 'updates the sidebar component when estimate is added' do
     submit_time('/estimate 3w 1d 1h')
 
+    wait_for_ajax
     page.within '.time-tracking-estimate-only-pane' do
       expect(page).to have_content '3w 1d 1h'
     end
@@ -16,6 +17,7 @@
   it 'updates the sidebar component when spent is added' do
     submit_time('/spend 3w 1d 1h')
 
+    wait_for_ajax
     page.within '.time-tracking-spend-only-pane' do
       expect(page).to have_content '3w 1d 1h'
     end
@@ -25,6 +27,7 @@
     submit_time('/estimate 3w 1d 1h')
     submit_time('/spend 3w 1d 1h')
 
+    wait_for_ajax
     page.within '.time-tracking-comparison-pane' do
       expect(page).to have_content '3w 1d 1h'
     end
@@ -34,7 +37,7 @@
     submit_time('/estimate 3w 1d 1h')
     submit_time('/remove_estimate')
 
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       expect(page).to have_content 'No estimate or time spent'
     end
   end
@@ -43,13 +46,13 @@
     submit_time('/spend 3w 1d 1h')
     submit_time('/remove_time_spent')
 
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       expect(page).to have_content 'No estimate or time spent'
     end
   end
 
   it 'shows the help state when icon is clicked' do
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       find('.help-button').click
       expect(page).to have_content 'Track time with slash commands'
       expect(page).to have_content 'Learn more'
@@ -57,7 +60,7 @@
   end
 
   it 'hides the help state when close icon is clicked' do
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       find('.help-button').click
       find('.close-help-button').click
 
@@ -67,7 +70,7 @@
   end
 
   it 'displays the correct help url' do
-    page.within '#issuable-time-tracker' do
+    page.within '.time-tracking-component-wrap' do
       find('.help-button').click
 
       expect(find_link('Learn more')[:href]).to have_content('/help/workflow/time_tracking.md')
diff --git a/spec/views/projects/tags/index.html.haml_spec.rb b/spec/views/projects/tags/index.html.haml_spec.rb
new file mode 100644
index 0000000000000..33122365e9ab7
--- /dev/null
+++ b/spec/views/projects/tags/index.html.haml_spec.rb
@@ -0,0 +1,20 @@
+require 'spec_helper'
+
+describe 'projects/tags/index', :view do
+  let(:project) { create(:project) }
+
+  before do
+    assign(:project, project)
+    assign(:repository, project.repository)
+    assign(:tags, [])
+
+    allow(view).to receive(:current_ref).and_return('master')
+    allow(view).to receive(:can?).and_return(false)
+  end
+
+  it 'defaults sort dropdown toggle to last updated' do
+    render
+
+    expect(rendered).to have_button('Last updated')
+  end
+end
diff --git a/spec/workers/elastic_commit_indexer_worker_spec.rb b/spec/workers/elastic_commit_indexer_worker_spec.rb
index d60f6deb3b118..0227b9c88a540 100644
--- a/spec/workers/elastic_commit_indexer_worker_spec.rb
+++ b/spec/workers/elastic_commit_indexer_worker_spec.rb
@@ -1,7 +1,7 @@
 require 'spec_helper'
 
 describe ElasticCommitIndexerWorker do
-  let(:project) { create(:project) }
+  let!(:project) { create(:project) }
 
   subject { described_class.new }
 
@@ -12,6 +12,7 @@
 
     it 'runs indexer' do
       expect_any_instance_of(Gitlab::Elastic::Indexer).to receive(:run)
+
       subject.perform(project.id, '0000', '0000')
     end
 
diff --git a/spec/workers/propagate_service_template_worker_spec.rb b/spec/workers/propagate_service_template_worker_spec.rb
new file mode 100644
index 0000000000000..7040d5ef81c97
--- /dev/null
+++ b/spec/workers/propagate_service_template_worker_spec.rb
@@ -0,0 +1,29 @@
+require 'spec_helper'
+
+describe PropagateServiceTemplateWorker do
+  let!(:service_template) do
+    PushoverService.create(
+      template: true,
+      active: true,
+      properties: {
+        device: 'MyDevice',
+        sound: 'mic',
+        priority: 4,
+        user_key: 'asdf',
+        api_key: '123456789'
+      })
+  end
+
+  before do
+    allow_any_instance_of(Gitlab::ExclusiveLease).to receive(:try_obtain).
+      and_return(true)
+  end
+
+  describe '#perform' do
+    it 'calls the propagate service with the template' do
+      expect(Projects::PropagateServiceTemplate).to receive(:propagate).with(service_template)
+
+      subject.perform(service_template.id)
+    end
+  end
+end
diff --git a/yarn.lock b/yarn.lock
index 2042d60036f9d..2024f7ce30c0b 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -2159,6 +2159,13 @@ expand-range@^1.8.1:
   dependencies:
     fill-range "^2.1.0"
 
+exports-loader@^0.6.4:
+  version "0.6.4"
+  resolved "https://registry.yarnpkg.com/exports-loader/-/exports-loader-0.6.4.tgz#d70fc6121975b35fc12830cf52754be2740fc886"
+  dependencies:
+    loader-utils "^1.0.2"
+    source-map "0.5.x"
+
 express@^4.13.3, express@^4.14.1:
   version "4.14.1"
   resolved "https://registry.yarnpkg.com/express/-/express-4.14.1.tgz#646c237f766f148c2120aff073817b9e4d7e0d33"
@@ -3102,6 +3109,10 @@ jasmine-jquery@^2.1.1:
   version "2.1.1"
   resolved "https://registry.yarnpkg.com/jasmine-jquery/-/jasmine-jquery-2.1.1.tgz#d4095e646944a26763235769ab018d9f30f0d47b"
 
+jed@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/jed/-/jed-1.1.1.tgz#7a549bbd9ffe1585b0cd0a191e203055bee574b4"
+
 jodid25519@^1.0.0:
   version "1.0.2"
   resolved "https://registry.yarnpkg.com/jodid25519/-/jodid25519-1.0.2.tgz#06d4912255093419477d425633606e0e90782967"
@@ -3179,7 +3190,7 @@ json-stable-stringify@^1.0.0, json-stable-stringify@^1.0.1:
   dependencies:
     jsonify "~0.0.0"
 
-json-stringify-safe@~5.0.1:
+json-stringify-safe@^5.0.1, json-stringify-safe@~5.0.1:
   version "5.0.1"
   resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb"
 
@@ -4573,6 +4584,12 @@ raphael@^2.2.7:
   dependencies:
     eve-raphael "0.5.0"
 
+raven-js@^3.14.0:
+  version "3.14.0"
+  resolved "https://registry.yarnpkg.com/raven-js/-/raven-js-3.14.0.tgz#94dda81d975fdc4a42f193db437cf70021d654e0"
+  dependencies:
+    json-stringify-safe "^5.0.1"
+
 raw-body@~2.2.0:
   version "2.2.0"
   resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.2.0.tgz#994976cf6a5096a41162840492f0bdc5d6e7fb96"
@@ -5118,6 +5135,10 @@ source-map-support@^0.4.2:
   dependencies:
     source-map "^0.5.3"
 
+source-map@0.5.x, source-map@^0.5.0, source-map@^0.5.3, source-map@^0.5.6, source-map@~0.5.1, source-map@~0.5.3:
+  version "0.5.6"
+  resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.6.tgz#75ce38f52bf0733c5a7f0c118d81334a2bb5f412"
+
 source-map@^0.1.41:
   version "0.1.43"
   resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.1.43.tgz#c24bc146ca517c1471f5dacbe2571b2b7f9e3346"
@@ -5130,10 +5151,6 @@ source-map@^0.4.4:
   dependencies:
     amdefine ">=0.0.4"
 
-source-map@^0.5.0, source-map@^0.5.3, source-map@^0.5.6, source-map@~0.5.1, source-map@~0.5.3:
-  version "0.5.6"
-  resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.6.tgz#75ce38f52bf0733c5a7f0c118d81334a2bb5f412"
-
 source-map@~0.2.0:
   version "0.2.0"
   resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.2.0.tgz#dab73fbcfc2ba819b4de03bd6f6eaa48164b3f9d"
@@ -5184,6 +5201,10 @@ sprintf-js@~1.0.2:
   version "1.0.3"
   resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c"
 
+sql.js@^0.4.0:
+  version "0.4.0"
+  resolved "https://registry.yarnpkg.com/sql.js/-/sql.js-0.4.0.tgz#23be9635520eb0ff43a741e7e830397266e88445"
+
 sshpk@^1.7.0:
   version "1.10.2"
   resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.10.2.tgz#d5a804ce22695515638e798dbe23273de070a5fa"
-- 
GitLab