您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Adds more convenient pull request links to Jira tickets.
// ==UserScript== // @name Jira: Pull Request Link Improver // @namespace https://github.com/rybak/atlassian-tweaks // @version 11 // @license MIT // @description Adds more convenient pull request links to Jira tickets. // @author Andrei Rybak // @match https://jira.example.com/browse/* // @include https://*jira*/browse/* // @icon https://jira.atlassian.com/favicon.ico // @homepageURL https://github.com/rybak/atlassian-tweaks // @grant none // ==/UserScript== /* * Copyright (c) 2022-2025 Andrei Rybak * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ (function() { 'use strict'; const LOG_PREFIX = '[PR Link Improver]'; const PANEL_ID = 'PrLinksImproverPanel'; const LIST_ID = 'PrLinksImproverList'; var loadInProgress = false; const PARSE_PATH = /[/](projects|users)[/]([^/]*)[/]repos[/]([^/]*)[/].*/; function log(...toLog) { console.log(LOG_PREFIX, ...toLog); } function info(...toLog) { console.info(LOG_PREFIX, ...toLog); } function warn(...toLog) { console.warn(LOG_PREFIX, ...toLog); } function error(...toLog) { console.error(LOG_PREFIX, ...toLog); } function getJiraMajorVersion() { return document.querySelector('meta[name="application-name"]').attributes.getNamedItem("data-version").value.split(".")[0]; } function createPanel() { var header; const jiraMajorVersion = getJiraMajorVersion(); switch (jiraMajorVersion) { case "7": header = '<div class="mod-header"><h2 class="toggle-title">Pull requests</h2></div>'; break; case "8": case "9": header = '<div class="mod-header"><h4 class="toggle-title">Pull requests</h4></div>'; break; default: warn("Jira v" + jiraMajorVersion + " is not supported"); header = '<div class="mod-header"><h4 class="toggle-title">Pull requests</h4></div>'; break; } $('#viewissue-devstatus-panel') .prepend($(header + `<div class="mod-content" id="${PANEL_ID}" style="margin-bottom:1rem;"></div>`)); } function addError(errors) { for (const e of errors) { error("Error: " + e); } $(`#${PANEL_ID}`).append($(`<p>Could not load from Bitbucket. Got: ${errors}</p>`)); } function extractProjectRepoSlugsFromPr(pr) { const url = pr.url; const path = new URL(url).pathname; const matching = path.match(PARSE_PATH); const project = matching[2]; const repository = matching[3]; return project + '/' + repository; } function addPrLinks(pullRequests) { $(`#${PANEL_ID}`).append($(`<ul id="${LIST_ID}" class="item-details status-panels devstatus-entry"></ul>`)); const list = $(`#${LIST_ID}`); for (const pr of pullRequests) { const url = pr.url; const prId = pr.id; const li = $('<li/>').appendTo(list); const slugsPrefix = extractProjectRepoSlugsFromPr(pr) + ': '; const link = document.createElement('a'); link.href = url; link.appendChild(document.createTextNode(`${prId}: ${pr.name}`)); info("NAME='" + pr.name + "' issue='" + JIRA.Issue.getIssueKey() + "'"); if (pr.name.includes(JIRA.Issue.getIssueKey())) { link.style.fontWeight = 'bold'; } if (pr.status == "DECLINED") { link.style.textDecoration = 'line-through'; } $(li).append(slugsPrefix).append(link); } } function addPrLinksPanel() { if ($(`#${PANEL_ID}`).length) { // the PR links panel has already been created loadInProgress = false; return; } const issueId = JIRA.Issue.getIssueId(); // https://community.atlassian.com/t5/Jira-questions/JIRA-REST-API-to-get-list-of-branches-related-to-a-issue/qaq-p/800389 const pullRequestsUrl = `/rest/dev-status/1.0/issue/detail?issueId=${issueId}&applicationType=stash&dataType=pullrequest`; log("Loading: " + pullRequestsUrl); $.getJSON(pullRequestsUrl, data => { if ($(`#${PANEL_ID}`).length) { // the PR links panel has been created while we were getting the PR data loadInProgress = false; return; } createPanel(); if (data.detail.length == 0) { addError(data.errors); loadInProgress = false; return; } const pullRequests = data.detail[0].pullRequests; addPrLinks(pullRequests); loadInProgress = false; }); } function startPrLoading() { if (!loadInProgress) { loadInProgress = true; addPrLinksPanel(); } else { log("Already loading. Skipping..."); } } $(document).ready(startPrLoading); JIRA.bind(JIRA.Events.NEW_CONTENT_ADDED, startPrLoading); })();