aboutsummaryrefslogtreecommitdiffstats
path: root/web_src
diff options
context:
space:
mode:
authoryp05327 <576951401@qq.com>2023-08-22 11:30:02 +0900
committerGitHub <noreply@github.com>2023-08-22 10:30:02 +0800
commita4a567f29f69e39d57a9fa7008d708f7fd080e58 (patch)
treea33773d1860087d7ec4c766c2117e405b03be335 /web_src
parentb3f713717407fcb66515a7a702e81b2028800f76 (diff)
downloadgitea-a4a567f29f69e39d57a9fa7008d708f7fd080e58.tar.gz
gitea-a4a567f29f69e39d57a9fa7008d708f7fd080e58.zip
Check disabled workflow when rerun jobs (#26535)
In GitHub, we can not rerun jobs if the workflow is disabled. --------- Co-authored-by: silverwind <me@silverwind.io> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
Diffstat (limited to 'web_src')
-rw-r--r--web_src/js/bootstrap.js10
-rw-r--r--web_src/js/components/RepoActionView.vue15
-rw-r--r--web_src/js/features/common-global.js116
3 files changed, 55 insertions, 86 deletions
diff --git a/web_src/js/bootstrap.js b/web_src/js/bootstrap.js
index f0b020ce1c..43075ab241 100644
--- a/web_src/js/bootstrap.js
+++ b/web_src/js/bootstrap.js
@@ -20,6 +20,10 @@ export function showGlobalErrorMessage(msg) {
* @param {ErrorEvent} e
*/
function processWindowErrorEvent(e) {
+ if (e.type === 'unhandledrejection') {
+ showGlobalErrorMessage(`JavaScript promise rejection: ${e.reason}. Open browser console to see more details.`);
+ return;
+ }
if (!e.error && e.lineno === 0 && e.colno === 0 && e.filename === '' && window.navigator.userAgent.includes('FxiOS/')) {
// At the moment, Firefox (iOS) (10x) has an engine bug. See https://github.com/go-gitea/gitea/issues/20240
// If a script inserts a newly created (and content changed) element into DOM, there will be a nonsense error event reporting: Script error: line 0, col 0.
@@ -30,6 +34,10 @@ function processWindowErrorEvent(e) {
}
function initGlobalErrorHandler() {
+ if (window._globalHandlerErrors?._inited) {
+ showGlobalErrorMessage(`The global error handler has been initialized, do not initialize it again`);
+ return;
+ }
if (!window.config) {
showGlobalErrorMessage(`Gitea JavaScript code couldn't run correctly, please check your custom templates`);
}
@@ -40,7 +48,7 @@ function initGlobalErrorHandler() {
processWindowErrorEvent(e);
}
// then, change _globalHandlerErrors to an object with push method, to process further error events directly
- window._globalHandlerErrors = {'push': (e) => processWindowErrorEvent(e)};
+ window._globalHandlerErrors = {_inited: true, push: (e) => processWindowErrorEvent(e)};
}
initGlobalErrorHandler();
diff --git a/web_src/js/components/RepoActionView.vue b/web_src/js/components/RepoActionView.vue
index 90823b986c..8b899ac2fe 100644
--- a/web_src/js/components/RepoActionView.vue
+++ b/web_src/js/components/RepoActionView.vue
@@ -14,7 +14,7 @@
<button class="ui basic small compact button red" @click="cancelRun()" v-else-if="run.canCancel">
{{ locale.cancel }}
</button>
- <button class="ui basic small compact button gt-mr-0" @click="rerun()" v-else-if="run.canRerun">
+ <button class="ui basic small compact button gt-mr-0 link-action" :data-url="`${run.link}/rerun`" v-else-if="run.canRerun">
{{ locale.rerun_all }}
</button>
</div>
@@ -38,7 +38,7 @@
<span class="job-brief-name gt-mx-3 gt-ellipsis">{{ job.name }}</span>
</div>
<span class="job-brief-item-right">
- <SvgIcon name="octicon-sync" role="button" :data-tooltip-content="locale.rerun" class="job-brief-rerun gt-mx-3" @click="rerunJob(index)" v-if="job.canRerun && onHoverRerunIndex === job.id"/>
+ <SvgIcon name="octicon-sync" role="button" :data-tooltip-content="locale.rerun" class="job-brief-rerun gt-mx-3 link-action" :data-url="`${run.link}/jobs/${index}/rerun`" v-if="job.canRerun && onHoverRerunIndex === job.id"/>
<span class="step-summary-duration">{{ job.duration }}</span>
</span>
</a>
@@ -264,17 +264,6 @@ const sfc = {
this.loadJob(); // try to load the data immediately instead of waiting for next timer interval
}
},
- // rerun a job
- async rerunJob(idx) {
- const jobLink = `${this.run.link}/jobs/${idx}`;
- await this.fetchPost(`${jobLink}/rerun`);
- window.location.href = jobLink;
- },
- // rerun workflow
- async rerun() {
- await this.fetchPost(`${this.run.link}/rerun`);
- window.location.href = this.run.link;
- },
// cancel a run
cancelRun() {
this.fetchPost(`${this.run.link}/cancel`);
diff --git a/web_src/js/features/common-global.js b/web_src/js/features/common-global.js
index be337ee903..7291410c1a 100644
--- a/web_src/js/features/common-global.js
+++ b/web_src/js/features/common-global.js
@@ -8,7 +8,7 @@ import {handleGlobalEnterQuickSubmit} from './comp/QuickSubmit.js';
import {svg} from '../svg.js';
import {hideElem, showElem, toggleElem} from '../utils/dom.js';
import {htmlEscape} from 'escape-goat';
-import {createTippy, showTemporaryTooltip} from '../modules/tippy.js';
+import {showTemporaryTooltip} from '../modules/tippy.js';
import {confirmModal} from './comp/ConfirmModal.js';
import {showErrorToast} from '../modules/toast.js';
@@ -64,9 +64,9 @@ export function initGlobalButtonClickOnEnter() {
});
}
-// doRedirect does real redirection to bypass the browser's limitations of "location"
+// fetchActionDoRedirect does real redirection to bypass the browser's limitations of "location"
// more details are in the backend's fetch-redirect handler
-function doRedirect(redirect) {
+function fetchActionDoRedirect(redirect) {
const form = document.createElement('form');
const input = document.createElement('input');
form.method = 'post';
@@ -79,6 +79,33 @@ function doRedirect(redirect) {
form.submit();
}
+async function fetchActionDoRequest(actionElem, url, opt) {
+ try {
+ const resp = await fetch(url, opt);
+ if (resp.status === 200) {
+ let {redirect} = await resp.json();
+ redirect = redirect || actionElem.getAttribute('data-redirect');
+ actionElem.classList.remove('dirty'); // remove the areYouSure check before reloading
+ if (redirect) {
+ fetchActionDoRedirect(redirect);
+ } else {
+ window.location.reload();
+ }
+ } else if (resp.status >= 400 && resp.status < 500) {
+ const data = await resp.json();
+ // the code was quite messy, sometimes the backend uses "err", sometimes it uses "error", and even "user_error"
+ // but at the moment, as a new approach, we only use "errorMessage" here, backend can use JSONError() to respond.
+ await showErrorToast(data.errorMessage || `server error: ${resp.status}`);
+ } else {
+ await showErrorToast(`server error: ${resp.status}`);
+ }
+ } catch (e) {
+ console.error('error when doRequest', e);
+ actionElem.classList.remove('is-loading', 'small-loading-icon');
+ await showErrorToast(i18n.network_error);
+ }
+}
+
async function formFetchAction(e) {
if (!e.target.classList.contains('form-fetch-action')) return;
@@ -115,50 +142,7 @@ async function formFetchAction(e) {
reqOpt.body = formData;
}
- let errorTippy;
- const onError = (msg) => {
- formEl.classList.remove('is-loading', 'small-loading-icon');
- if (errorTippy) errorTippy.destroy();
- // TODO: use a better toast UI instead of the tippy. If the form height is large, the tippy position is not good
- errorTippy = createTippy(formEl, {
- content: msg,
- interactive: true,
- showOnCreate: true,
- hideOnClick: true,
- role: 'alert',
- theme: 'form-fetch-error',
- trigger: 'manual',
- arrow: false,
- });
- };
-
- const doRequest = async () => {
- try {
- const resp = await fetch(reqUrl, reqOpt);
- if (resp.status === 200) {
- const {redirect} = await resp.json();
- formEl.classList.remove('dirty'); // remove the areYouSure check before reloading
- if (redirect) {
- doRedirect(redirect);
- } else {
- window.location.reload();
- }
- } else if (resp.status >= 400 && resp.status < 500) {
- const data = await resp.json();
- // the code was quite messy, sometimes the backend uses "err", sometimes it uses "error", and even "user_error"
- // but at the moment, as a new approach, we only use "errorMessage" here, backend can use JSONError() to respond.
- onError(data.errorMessage || `server error: ${resp.status}`);
- } else {
- onError(`server error: ${resp.status}`);
- }
- } catch (e) {
- console.error('error when doRequest', e);
- onError(i18n.network_error);
- }
- };
-
- // TODO: add "confirm" support like "link-action" in the future
- await doRequest();
+ await fetchActionDoRequest(formEl, reqUrl, reqOpt);
}
export function initGlobalCommon() {
@@ -209,6 +193,7 @@ export function initGlobalCommon() {
$('.tabular.menu .item').tab();
document.addEventListener('submit', formFetchAction);
+ document.addEventListener('click', linkAction);
}
export function initGlobalDropzone() {
@@ -269,41 +254,29 @@ export function initGlobalDropzone() {
}
async function linkAction(e) {
- e.preventDefault();
-
// A "link-action" can post AJAX request to its "data-url"
// Then the browser is redirected to: the "redirect" in response, or "data-redirect" attribute, or current URL by reloading.
// If the "link-action" has "data-modal-confirm" attribute, a confirm modal dialog will be shown before taking action.
+ const el = e.target.closest('.link-action');
+ if (!el) return;
- const $this = $(this);
- const redirect = $this.attr('data-redirect');
-
- const doRequest = () => {
- $this.prop('disabled', true);
- $.post($this.attr('data-url'), {
- _csrf: csrfToken
- }).done((data) => {
- if (data && data.redirect) {
- window.location.href = data.redirect;
- } else if (redirect) {
- window.location.href = redirect;
- } else {
- window.location.reload();
- }
- }).always(() => {
- $this.prop('disabled', false);
- });
+ e.preventDefault();
+ const url = el.getAttribute('data-url');
+ const doRequest = async () => {
+ el.disabled = true;
+ await fetchActionDoRequest(el, url, {method: 'POST', headers: {'X-Csrf-Token': csrfToken}});
+ el.disabled = false;
};
- const modalConfirmContent = htmlEscape($this.attr('data-modal-confirm') || '');
+ const modalConfirmContent = htmlEscape(el.getAttribute('data-modal-confirm') || '');
if (!modalConfirmContent) {
- doRequest();
+ await doRequest();
return;
}
- const isRisky = $this.hasClass('red') || $this.hasClass('yellow') || $this.hasClass('orange') || $this.hasClass('negative');
+ const isRisky = el.classList.contains('red') || el.classList.contains('yellow') || el.classList.contains('orange') || el.classList.contains('negative');
if (await confirmModal({content: modalConfirmContent, buttonColor: isRisky ? 'orange' : 'green'})) {
- doRequest();
+ await doRequest();
}
}
@@ -354,7 +327,6 @@ export function initGlobalLinkActions() {
// Helpers.
$('.delete-button').on('click', showDeletePopup);
- $('.link-action').on('click', linkAction);
}
function initGlobalShowModal() {