diff --git a/.github/workflows/jirabot-merge.yml b/.github/workflows/jirabot-merge.yml new file mode 100644 index 00000000000..57aa9b02879 --- /dev/null +++ b/.github/workflows/jirabot-merge.yml @@ -0,0 +1,299 @@ +name: Jirabot - Merge + +on: + pull_request_target: + types: [closed] + branches: + - "master" + - "candidate-*" + +jobs: + jirabot: + runs-on: ubuntu-latest + if: github.event.pull_request.merged == true + steps: + - name: "Debug Vars" + env: + JIRA_URL : ${{ vars.JIRA_URL }} + PULL_REQUEST_NUMBER : ${{ github.event.pull_request.number }} + PULL_REQUEST_TITLE : ${{ github.event.pull_request.title }} + PULL_REQUEST_AUTHOR_NAME : ${{ github.event.pull_request.user.login }} + PULL_URL: ${{ github.event.pull_request.html_url }} + COMMENTS_URL: ${{ github.event.pull_request.comments_url }} + BRANCH_NAME: ${{ github.ref_name }} + run: | + echo "JIRA_URL: $JIRA_URL" + echo "Pull Request Number: $PULL_REQUEST_NUMBER" + echo "Pull Request Title: $PULL_REQUEST_TITLE" + echo "Pull Request Author Name: $PULL_REQUEST_AUTHOR_NAME" + echo "Pull Request URL: $PULL_URL" + echo "Comments URL: $COMMENTS_URL" + echo "Branch Name: $BRANCH_NAME" + - uses: "actions/setup-python@v5" + with: + python-version: "3.8" + - name: "Install dependencies" + run: | + set -xe + python -VV + python -m site + python -m pip install --upgrade pip setuptools wheel + python -m pip install --upgrade atlassian-python-api + python -m pip install --upgrade jira + - name: "Checkout" + uses: actions/checkout@v4 + with: + ref: ${{ github.event.pull_request.base.ref }} + fetch-depth: 0 + fetch-tags: true + - name: "Run" + env: + JIRABOT_USERNAME : ${{ secrets.JIRABOT_USERNAME }} + JIRABOT_PASSWORD : ${{ secrets.JIRABOT_PASSWORD }} + JIRA_URL : ${{ vars.JIRA_URL }} + PULL_REQUEST_NUMBER : ${{ github.event.pull_request.number }} + PULL_REQUEST_TITLE : ${{ github.event.pull_request.title }} + PULL_REQUEST_AUTHOR_NAME : ${{ github.event.pull_request.user.login }} + PULL_URL: ${{ github.event.pull_request.html_url }} + PROJECT_CONFIG: ${{ vars.PROJECT_CONFIG}} + COMMENTS_URL: ${{ github.event.pull_request.comments_url }} + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + BRANCH_NAME: ${{ github.ref_name }} + shell: python + run: | + import os + import re + import subprocess + import time + import sys + import json + from atlassian.jira import Jira + + def extractVersion(versionStr): + parts = versionStr.split('.') + if len(parts) != 3: + print('Invalid version: ' + versionStr) + sys.exit(1) + if parts[2].lower() == 'x': + parts[2] = '0' + + major, minor, point = map(int, parts) + return [major, minor, point] + + def getTagVersionForCmd(cmd): + versionPattern = re.compile(r".*([0-9]+\.[0-9]+\.[0-9]+).*") + + # Get latest release version + gitTagProcess = subprocess.Popen(cmd, stdout=subprocess.PIPE, shell=True) + (output, err) = gitTagProcess.communicate() + gitTagProcessStatus = gitTagProcess.wait() + + if gitTagProcessStatus != 0: + print('Unable to retrieve latest git tag.') + sys.exit(1) + + latestGitTag = str(output) + + versionMatch = versionPattern.match(latestGitTag) + if versionMatch: + return extractVersion(versionMatch.group(1)) + else: + print('Unable to extract version from git tag.') + sys.exit(2) + + def buildVersionString(version): + major, minor, point = map(int, version) + return f"{major}.{minor}.{point}" + + def createReleaseTagPattern(projectConfig, major = None, minor = None, point = None): + releaseTagPrefix = projectConfig.get('tagPrefix') + releaseTagPostfix = projectConfig.get('tagPostfix') + + if releaseTagPrefix is None or releaseTagPostfix is None: + print('Error: PROJECT_CONFIG is missing required fields: tagPrefix and/or tagPostfix') + sys.exit(1) + + releaseTagPattern = releaseTagPrefix + if major is not None: + releaseTagPattern += str(major) + '\\.' + else: + releaseTagPattern += '[0-9]+\\.' + + if minor is not None: + releaseTagPattern += str(minor) + '\\.' + else: + releaseTagPattern += '[0-9]+\\.' + + if point is not None: + releaseTagPattern += str(point) + '(-[0-9]+)?' + else: + releaseTagPattern += '[0-9]+(-[0-9]+)?' + + releaseTagPattern += releaseTagPostfix + '$' + + return releaseTagPattern + + def getLatestSemVer(projectConfig, major = None, minor = None, point = None): + cmd = "git tag --list --sort=-v:refname | grep -E '" + createReleaseTagPattern(projectConfig, major, minor, point) + "' | head -n 1" + return getTagVersionForCmd(cmd) + + def generateFixVersionList(jira, projectConfig, projectName, branchName): + latestVersion = getLatestSemVer(projectConfig) + + # If we are merging into master we assume it is going into the next minor release + fixVersions = [] + if branchName == "master": + fixVersions = [buildVersionString([latestVersion[0], latestVersion[1] + 2, 0])] + else: + # Extract candidate branch major / minor version + candidateBranchPattern = re.compile(r"candidate-([0-9]+\.[0-9]+\.([0-9]+|x)).*") + branchVersionMatch = candidateBranchPattern.match(branchName) + branchVersion = extractVersion(branchVersionMatch.group(1)) + + # Get latest release in branch + latestBranchVer = getLatestSemVer(projectConfig, branchVersion[0], branchVersion[1]) + + curMajor = branchVersion[0] + latestMajor = latestVersion[0] + while curMajor <= latestMajor: + latestVersionInMajor = getLatestSemVer(projectConfig, curMajor) + + curMinor = 0 + if curMajor == branchVersion[0]: + curMinor = branchVersion[1] + + latestMinor = latestVersionInMajor[1] + + while curMinor <= latestMinor: + latestPointInMinor = getLatestSemVer(projectConfig, curMajor, curMinor) + fixVersions.append(buildVersionString([latestPointInMinor[0], latestPointInMinor[1], latestPointInMinor[2] + 2])) + curMinor += 2 + curMajor += 1 + + for fixVersion in fixVersions: + try: + alreadyHasFixVersion = False + versions = jira.get_project_versions(projectName) + for v in versions: + if v['name'] == fixVersion: + alreadyHasFixVersion = True + + if not alreadyHasFixVersion: + project = jira.get_project(projectName) + projectId = project['id'] + jira.add_version(projectName, projectId, fixVersion) + except Exception as error: + print('Error: Unable to add fix version: ' + fixVersion + ' with: ' + str(error)) + sys.exit(1) + + return fixVersions + + def resolveIssue(jira, projectName, issue, fixVersions) -> str: + result = '' + + versionsToAdd = [] + + issueName = issue['key'] + issueFields = issue['fields'] + + for addedVersion in fixVersions: + alreadyHasFixVersion = False + for v in issueFields['fixVersions']: + if v['name'] == addedVersion: + alreadyHasFixVersion = True + break + if not alreadyHasFixVersion: + versionsToAdd.append(addedVersion) + + versions = jira.get_project_versions(projectName) + updatedVersionList = [] + for v in issueFields['fixVersions']: + updatedVersionList.append({'id' : v['id']}) + + for fixVersionName in versionsToAdd: + fixVersion = None + for v in versions: + if v['name'] == fixVersionName: + fixVersion = v + break + + if fixVersion: + updatedVersionList.append({'id' : fixVersion['id']}) + result += "Added fix version: " + fixVersionName + "\n" + else: + result += "Error: Unable to find fix version: " + fixVersionName + "\n" + + if len(versionsToAdd) > 0: + try: + jira.update_issue_field(issueName, {'fixVersions': updatedVersionList}) + except Exception as error: + result += 'Error: Updating fix versions failed with: "' + str(error) + '\n' + else: + result += "Fix versions already added.\n" + + statusName = str(issueFields['status']['name']) + if statusName != 'Resolved': + try: + transitionId = jira.get_transition_id_to_status_name(issueName, 'Resolved') + jira.set_issue_status_by_transition_id(issueName, transitionId) + result += "Workflow Transition: 'Resolve issue'\n" + except Exception as error: + result += 'Error: Transitioning to: "Resolved" failed with: "' + str(error) + '\n' + + return result + + jirabot_user = os.environ['JIRABOT_USERNAME'] + jirabot_pass = os.environ['JIRABOT_PASSWORD'] + jira_url = os.environ['JIRA_URL'] + + pr = os.environ['PULL_REQUEST_NUMBER'] + title = os.environ['PULL_REQUEST_TITLE'] + user = os.environ['PULL_REQUEST_AUTHOR_NAME'] + pull_url = os.environ['PULL_URL'] + github_token = os.environ['GITHUB_TOKEN'] + branch_name = os.environ['BRANCH_NAME'] + comments_url = os.environ['COMMENTS_URL'] + + projectConfig = json.loads(os.environ['PROJECT_CONFIG']) + if not isinstance(projectConfig, dict): + print('Error: PROJECT_CONFIG is not a valid JSON object, aborting.') + sys.exit(1) + + if 'tagPrefix' not in projectConfig or 'tagPostfix' not in projectConfig: + print('Error: PROJECT_CONFIG is missing required fields: tagPrefix and/or tagPostfix') + sys.exit(1) + + project_prefixes = projectConfig.get('projectPrefixes') + if project_prefixes is None: + print('Error: PROJECT_CONFIG is missing required field: projectPrefixes') + sys.exit(1) + + project_list_regex = '|'.join(project_prefixes) + + result = '' + issuem = re.search("(" + project_list_regex + ")-[0-9]+", title, re.IGNORECASE) + if issuem: + project_name = issuem.group(1) + issue_name = issuem.group() + + jira = Jira(url=jira_url, username=jirabot_user, password=jirabot_pass, cloud=True) + + if not jira.issue_exists(issue_name): + sys.exit('Error: Unable to find Jira issue: ' + issue_name) + else: + issue = jira.issue(issue_name) + + result = 'Jirabot Action Result:\n' + + fixVersions = generateFixVersionList(jira, projectConfig, project_name, branch_name) + result += resolveIssue(jira, project_name, issue, fixVersions) + jira.issue_add_comment(issue_name, result) + + # Escape the result for JSON + result = json.dumps(result) + + subprocess.run(['curl', '-X', 'POST', comments_url, '-H', 'Content-Type: application/json', '-H', f'Authorization: token {github_token}', '--data', f'{{ "body": {result} }}'], check=True) + else: + print('Unable to find Jira issue name in title') + + print(result) diff --git a/dali/base/dadfs.cpp b/dali/base/dadfs.cpp index 937f38e64db..dcf10aaf171 100644 --- a/dali/base/dadfs.cpp +++ b/dali/base/dadfs.cpp @@ -118,6 +118,19 @@ inline unsigned groupDistance(IGroup *grp1,IGroup *grp2) return grp1->distance(grp2); } +inline StringBuffer &appendEnsurePathSepChar(StringBuffer &dest, StringBuffer &newPart, char psc) +{ + addPathSepChar(dest, psc); + if (newPart.length() > 0) + { + if (isPathSepChar(newPart.charAt(0))) + dest.append(newPart.str()+1); + else + dest.append(newPart); + } + return dest; +} + static StringBuffer &normalizeFormat(StringBuffer &in) { @@ -4474,7 +4487,8 @@ protected: friend class CDistributedFilePart; if (isPathSepChar(newPath.charAt(newPath.length()-1))) newPath.setLength(newPath.length()-1); newPath.remove(0, myBase.length()); - newdir.append(baseDir).append(newPath); + newdir.append(baseDir); + appendEnsurePathSepChar(newdir, newPath, psc); StringBuffer fullname; CIArrayOf newNames; unsigned i; @@ -4493,7 +4507,8 @@ protected: friend class CDistributedFilePart; StringBuffer copyDir(baseDir); adjustClusterDir(i, copy, copyDir); - fullname.clear().append(copyDir).append(newPath); + fullname.clear().append(copyDir); + appendEnsurePathSepChar(fullname, newPath, psc); newNames.item(i).append(fullname); } } diff --git a/esp/src/src-react/components/Logs.tsx b/esp/src/src-react/components/Logs.tsx index a0ef17ee26b..a1fad7bab0d 100644 --- a/esp/src/src-react/components/Logs.tsx +++ b/esp/src/src-react/components/Logs.tsx @@ -1,10 +1,10 @@ import * as React from "react"; import { CommandBar, ContextualMenuItemType, ICommandBarItemProps } from "@fluentui/react"; -import { GetLogsExRequest, LogaccessService, TargetAudience, LogType } from "@hpcc-js/comms"; +import { GetLogsExRequest, LogaccessService, LogType, TargetAudience, WsLogaccess } from "@hpcc-js/comms"; import { Level, scopedLogger } from "@hpcc-js/util"; import nlsHPCC from "src/nlsHPCC"; -import { logColor, timestampToDate, wuidToDate, wuidToTime } from "src/Utility"; -import { useLoggingEngine } from "../hooks/platform"; +import { logColor, removeAllExcept, wuidToDate, wuidToTime } from "src/Utility"; +import { useLogAccessInfo } from "../hooks/platform"; import { HolyGrail } from "../layouts/HolyGrail"; import { pushParams } from "../util/history"; import { FluentGrid, useCopyButtons, useFluentStoreState, FluentColumns } from "./controls/Grid"; @@ -94,6 +94,8 @@ const levelMap = (level) => { } }; +const columnOrder: string[] = [WsLogaccess.LogColumnType.timestamp, WsLogaccess.LogColumnType.message]; + export const Logs: React.FunctionComponent = ({ wuid, filter = defaultFilter, @@ -111,49 +113,48 @@ export const Logs: React.FunctionComponent = ({ const now = React.useMemo(() => new Date(), []); - const loggingEngine = useLoggingEngine(); + const { columns: logColumns } = useLogAccessInfo(); // Grid --- const columns = React.useMemo((): FluentColumns => { - let retVal = { - timestamp: { - label: nlsHPCC.TimeStamp, width: 140, sortable: false, - formatter: ts => { - if (ts) { - if (ts.indexOf(":") < 0) { - return timestampToDate(ts).toISOString(); - } - return new Date(ts).toISOString(); - } - }, - }, + // we've defined the columnOrder array above to ensure specific columns will + // appear on the left-most side of the grid, eg timestamps and log messages + const cols = logColumns?.sort((a, b) => { + const logTypeA = columnOrder.indexOf(a.LogType); + const logTypeB = columnOrder.indexOf(b.LogType); + + if (logTypeA >= 0) { + if (logTypeB >= 0) { return logTypeA - logTypeB; } + return -1; + } else if (logTypeB >= 0) { + return 1; + } else { + return 0; + } + }); + const retVal = { + timestamp: { label: nlsHPCC.TimeStamp, width: 140, sortable: false, }, message: { label: nlsHPCC.Message, width: 600, sortable: false, }, + components: { label: nlsHPCC.ContainerName, width: 150, sortable: false }, + instance: { label: nlsHPCC.PodName, width: 150, sortable: false }, + audience: { label: nlsHPCC.Audience, width: 60, sortable: false, }, + class: { + label: nlsHPCC.Class, width: 40, sortable: false, + formatter: level => { + const colors = logColor(levelMap(level)); + const styles = { backgroundColor: colors.background, padding: "2px 6px", color: colors.foreground }; + return {level}; + } + }, + workunits: { label: nlsHPCC.JobID, width: 50, sortable: false, hidden: wuid !== undefined, }, + processid: { label: nlsHPCC.ProcessID, width: 75, sortable: false, }, + logid: { label: nlsHPCC.Sequence, width: 70, sortable: false, }, + threadid: { label: nlsHPCC.ThreadID, width: 60, sortable: false, }, }; - if (loggingEngine === "grafanacurl") { - retVal = Object.assign(retVal, { - pod: { label: nlsHPCC.PodName, width: 150, sortable: false }, - }); - } else { - retVal = Object.assign(retVal, { - instance: { label: nlsHPCC.PodName, width: 150, sortable: false }, - components: { label: nlsHPCC.ContainerName, width: 150, sortable: false }, - audience: { label: nlsHPCC.Audience, width: 60, sortable: false, }, - class: { - label: nlsHPCC.Class, width: 40, sortable: false, - formatter: level => { - const colors = logColor(levelMap(level)); - const styles = { backgroundColor: colors.background, padding: "2px 6px", color: colors.foreground }; - return {level}; - } - }, - workunits: { label: nlsHPCC.JobID, width: 50, sortable: false, hidden: wuid !== undefined, }, - processid: { label: nlsHPCC.ProcessID, width: 75, sortable: false, }, - logid: { label: nlsHPCC.Sequence, width: 70, sortable: false, }, - threadid: { label: nlsHPCC.ThreadID, width: 60, sortable: false, }, - }); - } + const colTypes = cols?.map(c => c.LogType.toString()) ?? []; + removeAllExcept(retVal, colTypes); return retVal; - }, [loggingEngine, wuid]); + }, [logColumns, wuid]); const copyButtons = useCopyButtons(columns, selection, "logaccess"); @@ -218,8 +219,10 @@ export const Logs: React.FunctionComponent = ({ delete retVal.jobId; } } + const colTypes = logColumns?.map(c => c.LogType.toString()) ?? []; + removeAllExcept(retVal, colTypes); return retVal; - }, [filter, wuid]); + }, [filter, logColumns, wuid]); return } diff --git a/esp/src/src-react/hooks/platform.ts b/esp/src/src-react/hooks/platform.ts index 4bc86caa2c4..33f2089c260 100644 --- a/esp/src/src-react/hooks/platform.ts +++ b/esp/src/src-react/hooks/platform.ts @@ -2,7 +2,7 @@ import * as React from "react"; import { Octokit } from "octokit"; import { useConst } from "@fluentui/react-hooks"; import { scopedLogger } from "@hpcc-js/util"; -import { LogaccessService, Topology, WsTopology, WorkunitsServiceEx } from "@hpcc-js/comms"; +import { LogaccessService, Topology, WsLogaccess, WsTopology, WorkunitsServiceEx } from "@hpcc-js/comms"; import { getBuildInfo, BuildInfo, fetchModernMode } from "src/Session"; import { cmake_build_type, containerized, ModernMode } from "src/BuildInfo"; import { sessionKeyValStore, userKeyValStore } from "src/KeyValStore"; @@ -209,12 +209,19 @@ export function useModernMode(): { return { modernMode, setModernMode }; } -export function useLoggingEngine(): string { - const [loggingEngine, setLoggingEngine] = React.useState(""); +export function useLogAccessInfo(): { + managerType: string; + columns: WsLogaccess.Column[] +} { + const [managerType, setManagerType] = React.useState(""); + const [columns, setColumns] = React.useState(); React.useEffect(() => { - service.GetLogAccessInfo({}).then(response => setLoggingEngine(response.RemoteLogManagerType ?? "")); + service.GetLogAccessInfo({}).then(response => { + setManagerType(response.RemoteLogManagerType ?? ""); + setColumns(response?.Columns?.Column); + }); }, []); - return loggingEngine; + return { managerType, columns }; } \ No newline at end of file diff --git a/esp/src/src/Utility.ts b/esp/src/src/Utility.ts index 4b70d873129..2d6686a4a40 100644 --- a/esp/src/src/Utility.ts +++ b/esp/src/src/Utility.ts @@ -1322,4 +1322,12 @@ export function wuidToTime(wuid: string): string { export function wuidToDateTime(wuid: string): Date { return new Date(`${wuidToDate(wuid)}T${wuidToTime(wuid)}Z`); +} + +export function removeAllExcept(arr: any, keysToKeep: string[]): void { + for (const key of Object.keys(arr)) { + if (keysToKeep.indexOf(key) < 0) { + delete arr[key]; + } + } } \ No newline at end of file