environment.vue 7.84 KB
Newer Older
1
<script>
2
/* global Flash */
3
import Visibility from 'visibilityjs';
4
import EnvironmentsService from '../services/environments_service';
5
import environmentTable from './environments_table.vue';
6
import EnvironmentsStore from '../stores/environments_store';
7
import loadingIcon from '../../vue_shared/components/loading_icon.vue';
8
import tablePagination from '../../vue_shared/components/table_pagination.vue';
9
import '../../lib/utils/common_utils';
10
import eventHub from '../event_hub';
11 12
import Poll from '../../lib/utils/poll';
import environmentsMixin from '../mixins/environments_mixin';
13

14
export default {
15 16

  components: {
17 18
    environmentTable,
    tablePagination,
19
    loadingIcon,
20 21
  },

22 23 24 25
  mixins: [
    environmentsMixin,
  ],

26 27
  data() {
    const environmentsData = document.querySelector('#environments-list-view').dataset;
Filipa Lacerda's avatar
Filipa Lacerda committed
28
    const store = new EnvironmentsStore();
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43

    return {
      store,
      state: store.state,
      visibility: 'available',
      isLoading: false,
      cssContainerClass: environmentsData.cssClass,
      endpoint: environmentsData.environmentsDataEndpoint,
      canCreateDeployment: environmentsData.canCreateDeployment,
      canReadEnvironment: environmentsData.canReadEnvironment,
      canCreateEnvironment: environmentsData.canCreateEnvironment,
      projectEnvironmentsPath: environmentsData.projectEnvironmentsPath,
      projectStoppedEnvironmentsPath: environmentsData.projectStoppedEnvironmentsPath,
      newEnvironmentPath: environmentsData.newEnvironmentPath,
      helpPagePath: environmentsData.helpPagePath,
44
      isMakingRequest: false,
Filipa Lacerda's avatar
Filipa Lacerda committed
45 46 47 48

      // Pagination Properties,
      paginationInformation: {},
      pageNumber: 1,
49 50 51 52 53
    };
  },

  computed: {
    scope() {
54
      return gl.utils.getParameterByName('scope');
Filipa Lacerda's avatar
Filipa Lacerda committed
55 56
    },

57
    canReadEnvironmentParsed() {
58
      return gl.utils.convertPermissionToBoolean(this.canReadEnvironment);
Filipa Lacerda's avatar
Filipa Lacerda committed
59 60
    },

61
    canCreateDeploymentParsed() {
62
      return gl.utils.convertPermissionToBoolean(this.canCreateDeployment);
Filipa Lacerda's avatar
Filipa Lacerda committed
63 64
    },

65
    canCreateEnvironmentParsed() {
66
      return gl.utils.convertPermissionToBoolean(this.canCreateEnvironment);
Filipa Lacerda's avatar
Filipa Lacerda committed
67
    },
68 69 70 71 72 73 74
  },

  /**
   * Fetches all the environments and stores them.
   * Toggles loading property.
   */
  created() {
75 76 77
    const scope = gl.utils.getParameterByName('scope') || this.visibility;
    const page = gl.utils.getParameterByName('page') || this.pageNumber;

78 79
    this.service = new EnvironmentsService(this.endpoint);

80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
    const poll = new Poll({
      resource: this.service,
      method: 'get',
      data: { scope, page },
      successCallback: this.successCallback,
      errorCallback: this.errorCallback,
      notificationCallback: (isMakingRequest) => {
        this.isMakingRequest = isMakingRequest;
      },
    });

    if (!Visibility.hidden()) {
      this.isLoading = true;
      poll.makeRequest();
    }

    Visibility.change(() => {
      if (!Visibility.hidden()) {
        poll.restart();
      } else {
        poll.stop();
      }
    });
103

104
    eventHub.$on('toggleFolder', this.toggleFolder);
105
    eventHub.$on('postAction', this.postAction);
106 107
  },

108
  beforeDestroy() {
109
    eventHub.$off('toggleFolder');
Filipa Lacerda's avatar
Filipa Lacerda committed
110
    eventHub.$off('postAction');
111 112 113
  },

  methods: {
Filipa Lacerda's avatar
Filipa Lacerda committed
114
    toggleFolder(folder, folderUrl) {
115 116 117
      this.store.toggleFolder(folder);

      if (!folder.isOpen) {
118
        this.fetchChildEnvironments(folder, folderUrl, true);
119
      }
Filipa Lacerda's avatar
Filipa Lacerda committed
120
    },
Filipa Lacerda's avatar
Filipa Lacerda committed
121

122 123 124 125
    /**
     * Will change the page number and update the URL.
     *
     * @param  {Number} pageNumber desired page to go to.
126
     * @return {String}
127 128
     */
    changePage(pageNumber) {
129
      const param = gl.utils.setParamInURL('page', pageNumber);
130

131
      gl.utils.visitUrl(param);
132
      return param;
Filipa Lacerda's avatar
Filipa Lacerda committed
133
    },
134 135 136

    fetchEnvironments() {
      const scope = gl.utils.getParameterByName('scope') || this.visibility;
137
      const page = gl.utils.getParameterByName('page') || this.pageNumber;
138 139 140

      this.isLoading = true;

141 142 143
      return this.service.get({ scope, page })
        .then(this.successCallback)
        .catch(this.errorCallback);
144
    },
145

146 147
    fetchChildEnvironments(folder, folderUrl, showLoader = false) {
      this.store.updateEnvironmentProp(folder, 'isLoadingFolderContent', showLoader);
148

Filipa Lacerda's avatar
Filipa Lacerda committed
149
      this.service.getFolderContent(folderUrl)
150
        .then(resp => resp.json())
151 152
        .then(response => this.store.setfolderContent(folder, response.environments))
        .then(() => this.store.updateEnvironmentProp(folder, 'isLoadingFolderContent', false))
153
        .catch(() => {
154
          // eslint-disable-next-line no-new
155
          new Flash('An error occurred while fetching the environments.');
156
          this.store.updateEnvironmentProp(folder, 'isLoadingFolderContent', false);
157 158
        });
    },
159 160

    postAction(endpoint) {
161 162 163 164 165 166 167 168 169 170 171 172
      if (!this.isMakingRequest) {
        this.isLoading = true;

        this.service.postAction(endpoint)
          .then(() => this.fetchEnvironments())
          .catch(() => new Flash('An error occured while making the request.'));
      }
    },

    successCallback(resp) {
      this.saveData(resp);

173 174 175 176
      // We need to verify if any folder is open to also update it
      const openFolders = this.store.getOpenFolders();
      if (openFolders.length) {
        openFolders.forEach((folder) => {
177 178 179 180 181 182 183 184 185 186 187 188
          // TODO - Move this to the backend
          const folderUrl = `${window.location.pathname}/folders/${folder.folderName}`;

          return this.fetchChildEnvironments(folder, folderUrl);
        });
      }
    },

    errorCallback() {
      this.isLoading = false;
      // eslint-disable-next-line no-new
      new Flash('An error occurred while fetching the environments.');
189
    },
190
  },
191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212
};
</script>
<template>
  <div :class="cssContainerClass">
    <div class="top-area">
      <ul
        v-if="!isLoading"
        class="nav-links">
        <li :class="{ active: scope === null || scope === 'available' }">
          <a :href="projectEnvironmentsPath">
            Available
            <span class="badge js-available-environments-count">
              {{state.availableCounter}}
            </span>
          </a>
        </li>
        <li :class="{ active : scope === 'stopped' }">
          <a :href="projectStoppedEnvironmentsPath">
            Stopped
            <span class="badge js-stopped-environments-count">
              {{state.stoppedCounter}}
            </span>
213
          </a>
214 215 216 217 218 219 220 221 222 223
        </li>
      </ul>
      <div
        v-if="canCreateEnvironmentParsed && !isLoading"
        class="nav-controls">
        <a
          :href="newEnvironmentPath"
          class="btn btn-create">
          New environment
        </a>
224
      </div>
225 226
    </div>

227
    <div class="environments-container">
228 229 230 231 232
      <loading-icon
        label="Loading environments"
        size="3"
        v-if="isLoading"
        />
233 234 235 236 237 238 239 240 241 242 243 244

      <div
        class="blank-state blank-state-no-icon"
        v-if="!isLoading && state.environments.length === 0">
        <h2 class="blank-state-title js-blank-state-title">
          You don't have any environments right now.
        </h2>
        <p class="blank-state-text">
          Environments are places where code gets deployed, such as staging or production.
          <br />
          <a :href="helpPagePath">
            Read more about environments
245
          </a>
246 247 248 249 250 251
        </p>

        <a
          v-if="canCreateEnvironmentParsed"
          :href="newEnvironmentPath"
          class="btn btn-create js-new-environment-button">
252
          New environment
253
        </a>
Filipa Lacerda's avatar
Filipa Lacerda committed
254
      </div>
255 256 257 258 259 260 261 262 263

      <div
        class="table-holder"
        v-if="!isLoading && state.environments.length > 0">

        <environment-table
          :environments="state.environments"
          :can-create-deployment="canCreateDeploymentParsed"
          :can-read-environment="canReadEnvironmentParsed"
264
          />
265 266 267 268 269 270
      </div>

      <table-pagination
        v-if="state.paginationInformation && state.paginationInformation.totalPages > 1"
        :change="changePage"
        :pageInfo="state.paginationInformation" />
271
    </div>
272 273
  </div>
</template>