blob: 88f358a34f365a23dcdc2aadec3eedf5067ea203 [file] [log] [blame]
Zack Williams27cd3e52018-09-18 16:44:50 -07001#!/usr/bin/env bash
Joey Armstrong28eddda2023-01-10 03:09:34 -05002# -----------------------------------------------------------------------
3# Copyright 2018-2023 Open Networking Foundation (ONF) and the ONF Contributors
Zack Williams27cd3e52018-09-18 16:44:50 -07004#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
Joey Armstrong28eddda2023-01-10 03:09:34 -050016#
Zack Williams27cd3e52018-09-18 16:44:50 -070017# github-release.sh
18# builds (with make) and uploads release artifacts (binaries, etc.) to github
19# given a tag also create checksums files and release notes from the commit
20# message
Joey Armstrong28eddda2023-01-10 03:09:34 -050021# -----------------------------------------------------------------------
Zack Williams27cd3e52018-09-18 16:44:50 -070022
Joey Armstrong2097d3e2023-03-26 10:32:03 -040023set -euo pipefail
24
25##-------------------##
26##---] GLOBALS [---##
27##-------------------##
28declare -g WORKSPACE
29declare -g GERRIT_PROJECT
30declare -g __githost='github.com'
31
32## -----------------------------------------------------------------------
33## Uncomment to activate
34## -----------------------------------------------------------------------
35declare -i -g draft_release=1
Joey Armstrong41923cc2023-01-30 14:38:16 -050036# declare -g TRACE=0 # uncomment to set -x
37
38# shellcheck disable=SC2015
39[[ -v TRACE ]] && { set -x; } || { set +x; } # SC2015 (shellcheck -x)
40
Joey Armstrong2097d3e2023-03-26 10:32:03 -040041declare -a -g ARGV=() # Capture args to minimize globals and arg passing
42[[ $# -gt 0 ]] && ARGV=("$@")
Joey Armstrongd99e3d22023-01-29 12:35:43 -050043
44declare -g scratch # temp workspace for downloads
45declare -g gh_cmd # path to gh command
46
Joey Armstrong2097d3e2023-03-26 10:32:03 -040047declare -g SCRIPT_VERSION='1.2' # git changeset needed
Joey Armstrong26707f02023-01-26 12:41:12 -050048
Joey Armstrong7f382ef2023-01-25 12:00:08 -050049##--------------------##
50##---] INCLUDES [---##
Joey Armstrong1962bcf2023-01-27 13:53:18 -050051##--------------------#
Joey Armstrong2097d3e2023-03-26 10:32:03 -040052
53## -----------------------------------------------------------------------
54## Intent: Register an interrupt handler to display a stack trace on error
55## -----------------------------------------------------------------------
56function errexit()
57{
58 local err=$?
59 set +o xtrace
60 local code="${1:-1}"
61
62 local prefix="${BASH_SOURCE[1]}:${BASH_LINENO[0]}"
63 echo -e "\nOFFENDER: ${prefix}"
64 if [ $# -gt 0 ] && [ "$1" == '--stacktrace-quiet' ]; then
65 code=1
66 else
67 echo "ERROR: '${BASH_COMMAND}' exited with status $err"
68 fi
69
70 # Print out the stack trace described by $function_stack
71 if [ ${#FUNCNAME[@]} -gt 2 ]
72 then
73 echo "Call tree:"
74 for ((i=1;i<${#FUNCNAME[@]}-1;i++))
75 do
76 echo " $i: ${BASH_SOURCE[$i+1]}:${BASH_LINENO[$i]} ${FUNCNAME[$i]}(...)"
77 done
78 fi
79
80 echo "Exiting with status ${code}"
81 echo
82 exit "${code}"
83 # return
84}
85
86# trap ERR to provide an error handler whenever a command exits nonzero
87# this is a more verbose version of set -o errexit
88trap errexit ERR
89
90# setting errtrace allows our ERR trap handler to be propagated to functions,
91# expansions and subshells
92set -o errtrace
Joey Armstrong7f382ef2023-01-25 12:00:08 -050093
94## -----------------------------------------------------------------------
Joey Armstrongd99e3d22023-01-29 12:35:43 -050095## Intent: Cleanup scratch area on exit
96## -----------------------------------------------------------------------
97function sigtrap()
98{
99 ## Prevent mishaps
100 local is_read_only
101 is_read_only="$(declare -p scratch)"
102 if [[ $is_read_only != *"declare -r scratch="* ]]; then
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400103 echo "ERROR: variable scratch is not read-only, cleanup skipped"
104 exit 1
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500105 fi
106
107 if [ -d "$scratch" ]; then
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400108 /bin/rm -fr "$scratch"
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500109 fi
110
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400111 do_logout
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500112 return
113}
114trap sigtrap EXIT
115
116## -----------------------------------------------------------------------
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400117## Intent: Return a release version for queries
118## Note: Do not use in production, function is intended for interactive use
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400119## -----------------------------------------------------------------------
120function get_version()
121{
122 declare -n ref="$1"
123
124 banner
125 declare -a rev=()
126 rev+=("$(( RANDOM % 10 + 1 ))")
127 rev+=("$(( RANDOM % 256 + 1 ))")
Joey Armstrongeef8c2c2023-03-27 17:27:43 -0400128 rev+=("$(( RANDOM % 10000 + 1 ))")
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400129 ver="v${rev[0]}.${rev[1]}.${rev[2]}"
130
131 func_echo "version: $ver"
132 ref="$ver"
133 return
134}
135
136## -----------------------------------------------------------------------
137## Intent: Provide defaults for environment variables
138## -----------------------------------------------------------------------
139function initEnvVars()
140{
141 # when not running under Jenkins, use current dir as workspace and a blank
142 # project name
143 declare -g WORKSPACE=${WORKSPACE:-.}
144 declare -g GERRIT_PROJECT=${GERRIT_PROJECT:-}
145
146 # Github organization (or user) this project is published on. Project name should
147 # be the same on both Gerrit and GitHub
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400148 declare -g GITHUB_ORGANIZATION=${GITHUB_ORGANIZATION:-}
149
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400150 # glob pattern relative to project dir matching release artifacts
151 # ARTIFACT_GLOB=${ARTIFACT_GLOB:-"release/*"} # stat -- release/* not found, literal string (?)
152 declare -g ARTIFACT_GLOB=${ARTIFACT_GLOB:-"release/."}
153
154 # Use "release" as the default makefile target, can be a space separated list
155 declare -g RELEASE_TARGETS=${RELEASE_TARGETS:-release}
156
157 # Set and handle GOPATH and PATH
158 export GOPATH=${GOPATH:-$WORKSPACE/go}
159 export PATH=$PATH:/usr/lib/go-1.12/bin:/usr/local/go/bin:$GOPATH/bin
160 return
161}
162
163## -----------------------------------------------------------------------
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500164## Intent: Create a scratch area for downloads and transient tools
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400165## temp directory will be automatically removed upon exit.
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500166## -----------------------------------------------------------------------
167function init()
168{
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500169 local pkgbase="${0##*/}" # basename
170 local pkgname="${pkgbase%.*}"
171
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400172 # initEnvVars # moved to full_banner()
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400173
174 ## Create a temp directory for auto-cleanup
175 declare -g scratch
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500176 scratch="$(mktemp -d -t "${pkgname}.XXXXXXXXXX")"
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400177 readonly scratch
178 declare -p scratch
179
180 ## prime the stream
181 local work
182 get_release_dir work
183 declare -p work
184 return
185}
186
187## -----------------------------------------------------------------------
188## Intent: Verbose output for logging
189## -----------------------------------------------------------------------
190function banner()
191{
192 local iam="${0##*/}"
193 cat <<EOB
194
195** -----------------------------------------------------------------------
196** ${iam}::${FUNCNAME[1]}: $*
197** -----------------------------------------------------------------------
198EOB
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500199}
200
201## -----------------------------------------------------------------------
Joey Armstrong7f382ef2023-01-25 12:00:08 -0500202## Intent: Output a log banner to identify the running script/version.
203## -----------------------------------------------------------------------
204## TODO:
205## o SCRIPT_VERSION => git changeset for repo:ci-managment
206## o echo "library version: ${env."library.libName.version"}"
207# -----------------------------------------------------------------------
208# 14:18:38 > git fetch --no-tags --progress -- https://gerrit.opencord.org/ci-management.git +refs/heads/*:refs/remotes/origin/* # timeout=10
209# 14:18:39 Checking out Revision 50f6e0b97f449b32d32ec0e02d59642000351847 (master)
210# -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400211function full_banner()
Joey Armstrong7f382ef2023-01-25 12:00:08 -0500212{
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400213 local iam="${0##*/}"
214
215 initEnvVars # set defaults
Joey Armstrong7f382ef2023-01-25 12:00:08 -0500216
217cat <<EOH
218
219** -----------------------------------------------------------------------
220** IAM: ${iam} :: ${FUNCNAME[0]}
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400221** ARGV: ${ARGV[@]}
Joey Armstrong7f382ef2023-01-25 12:00:08 -0500222** PWD: $(/bin/pwd)
223** NOW: $(date '+%Y/%m/%d %H:%M:%S')
224** VER: ${SCRIPT_VERSION:-'unknown'}
225** -----------------------------------------------------------------------
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400226** GERRIT_PROJECT: $(declare -p GERRIT_PROJECT)
227** GITHUB_ORGANIZATION: $(declare -p GITHUB_ORGANIZATION)
228** RELEASE_TARGETS: $(declare -p RELEASE_TARGETS)
229** GOPATH: $(declare -p GOPATH)
230** -----------------------------------------------------------------------
231** PATH += /usr/lib/go-1.12/bin:/usr/local/go/bin:GOPATH/bin
232** -----------------------------------------------------------------------
Joey Armstrong7f382ef2023-01-25 12:00:08 -0500233EOH
234
235 return
236}
237
Joey Armstrongaf577ab2022-12-15 14:43:33 -0500238## -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400239## Intent: Display a message with 'iam' identifier for logging
Joey Armstrong50f6e0b2023-01-24 14:14:08 -0500240## -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400241function func_echo()
Joey Armstrong1962bcf2023-01-27 13:53:18 -0500242{
243 local iam="${0##*/}"
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400244 echo "** ${iam} :: ${FUNCNAME[1]}: $*"
Joey Armstrong1962bcf2023-01-27 13:53:18 -0500245 return
246}
247
248## -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400249## Intent: Display an error message then exit with status.
250## -----------------------------------------------------------------------
251function error()
252{
253 local iam="${0##*/}"
254 echo "ERROR ${iam} :: ${FUNCNAME[1]}: $*"
255 exit 1
256}
257
258## -----------------------------------------------------------------------
259## Intent: Verify sandbox/build is versioned for release.
260## -----------------------------------------------------------------------
261function getGitVersion()
262{
263 declare -n varname="$1"; shift
264 local ver
265
266 banner
267 ver="$(git tag -l --points-at HEAD)"
268 declare -p ver
269
270 get_version 'ver'
271 declare -p ver
272
273 # ------------------------------------------------------
274 # match bare versions or v-prefixed golang style version
275 # Critical failure for new/yet-to-be-released repo ?
276 # ------------------------------------------------------
277 if [[ "$ver" =~ ^v?([0-9]+)\.([0-9]+)\.([0-9]+)$ ]]
278 then
279 echo "git has a SemVer released version tag: '$ver'"
280 echo "Building artifacts for GitHub release."
281 else
282 error "No SemVer released version tag found, exiting..."
283 fi
284
285 varname="$ver"
286 func_echo "GIT_VERSION is [$GIT_VERSION] from $(/bin/pwd)"
287 return
288}
289
290## -----------------------------------------------------------------------
291## Intent:
292## Note: Release description is sanitized version of the log message
293## -----------------------------------------------------------------------
294function getReleaseDescription()
295{
296 declare -n varname="$1"; shift
297
298 local msg
299 msg="$(git log -1 --pretty=%B)"
300
301 local val
302 val="$(tr -dc "[:alnum:]\n\r\.\[\]\:\-\\\/\`\' " <<< "$msg")"
303
304 [[ ${#val} -eq 0 ]] && error "Release Description is empty ($msg)"
305 varname="$val"
306 return
307}
308
309## -----------------------------------------------------------------------
310## Intent: Retrieve value of the release temp directory.
311## -----------------------------------------------------------------------
312## Note: Limit use of globals so logic can be isolated in function calls.
313## -----------------------------------------------------------------------
314# declare -g RELEASE_TEMP
315function get_release_dir()
316{
317 declare -n varname=$1; shift
318
319 # Temporary staging directory to copy artifacts to
320 varname="$scratch/release"
321 mkdir -p "$varname"
322 return
323}
324
325## -----------------------------------------------------------------------
326## Intent: Retrieve github server hostname
327## -----------------------------------------------------------------------
328function get_gh_hostname()
329{
330 declare -n varname=$1; shift
331 varname+=('--hostname' "${__githost}")
332 return
333}
334
335## -----------------------------------------------------------------------
336## Intent: Retrieve repository organizaiton name
337## -----------------------------------------------------------------------
338function get_gh_repo_org()
339{
340 declare -n varname=$1; shift
341 declare -g __repo_org
342
343 local org
344 if [[ -v __repo_org ]]; then
345 org="$__repo_org"
346 elif [[ ! -v GITHUB_ORGANIZATION ]]; then
347 error "--repo-org or GITHUB_ORGANIZATION= are required"
348 else
349 org="${GITHUB_ORGANIZATION}"
350 fi
351
352 # shellcheck disable=SC2178
353 varname="$org"
354 return
355}
356
357## -----------------------------------------------------------------------
358## Intent: Retrieve repository organizaiton name
359## -----------------------------------------------------------------------
360function get_gh_repo_name()
361{
362 declare -n varname=$1; shift
363 declare -g __repo_name
364
365 local name
366 if [[ -v __repo_name ]]; then
367 name="$__repo_name"
368 elif [[ ! -v GITHUB_PROJECT ]]; then
369 error "--repo-name or GITHUB_PROJECT= are required"
370 else
371 name="${GITHUB_PROJECT}"
372 fi
373
374 varname="$name"
375 return
376}
377
378## -----------------------------------------------------------------------
379## Intent: Return path for the gh release query
380## -----------------------------------------------------------------------
381function get_gh_releases()
382{
383 declare -n ref="$1"
384
385 local repo_org
386 get_gh_repo_org repo_org
387
388 local repo_name
389 get_gh_repo_name repo_name
390
391 ref="repos/${repo_org}/${repo_name}/releases"
392 func_echo "ref=$ref"
393 return
394}
395
396## -----------------------------------------------------------------------
397## Intent: Retrieve repository path argument
398## -----------------------------------------------------------------------
399function get_argv_repo()
400{
401 declare -n varname=$1; shift
402
403 local repo_org
404 get_gh_repo_org repo_org
405
406 local repo_name
407 get_gh_repo_name repo_name
408
409 varname="${repo_org}/${repo_name}"
410 func_echo "VARNAME=$varname"
411 return
412}
413
414## -----------------------------------------------------------------------
415## Intent: Retrieve release name string "project - version"
416## -----------------------------------------------------------------------
417function get_argv_name()
418{
419 declare -n varname=$1; shift
420
421 local repo_name
422 get_gh_repo_name repo_name
423 varname="${repo_name} - $GIT_VERSION"
424 func_echo "varname=$varname"
425 return
426}
427
428## -----------------------------------------------------------------------
429## Intent: Retrieve tag version
430## -----------------------------------------------------------------------
431function get_argv_tag()
432{
433 declare -n varname=$1; shift
434
435 # cached_argv_tag='v3.41.3204'
436 if [[ ! -v cached_argv_tag ]]; then
437 declare -g cached_argv_tag
438 if [[ -v GIT_VERSION ]]; then # hello jenkins
439 cached_argv_tag="$GIT_VERSION"
440 fi
441 fi
442
443 [[ ${#cached_argv_tag} -eq 0 ]] && error "Unable to determine GIT_VERSION="
444 varname="$cached_argv_tag"
445 func_echo "varname=$varname"
446 return
447}
448
449## -----------------------------------------------------------------------
450## Intent:
451## -----------------------------------------------------------------------
452# To support golang projects that require GOPATH to be set and code checked out there
453# If $DEST_GOPATH is not an empty string:
454# - create GOPATH within WORKSPACE, and destination directory within
455# - set PATH to include $GOPATH/bin and the system go binaries
456# - move project from $WORKSPACE/$GERRIT_PROJECT to new location in $GOPATH
457# - start release process within that directory
458## -----------------------------------------------------------------------
459function get_release_path()
460{
461 declare -n varname=$1; shift
462
463 DEST_GOPATH=${DEST_GOPATH:-}
464 if [ -n "$DEST_GOPATH" ]; then
465 # if [[ -v DEST_GOPATH ]] && [[ -n DEST_GOPATH ]]; then
466 ## [jenkins] Suspect this will taint the golang installation.
467 ## [jenkins] Install succeeds, release fails, next job affected due to corruption.
468 ## [jenkins] Copy golang to temp then augment ?
469 ## [jenkins] Worst case (problematic) backup then restore
470 mkdir -p "$GOPATH/src/$DEST_GOPATH"
471 varname="$GOPATH/src/$DEST_GOPATH/$GERRIT_PROJECT"
472 mv "$WORKSPACE/$GERRIT_PROJECT" "$varname"
473
474 ## [TODO] - support local dev use
475 # elif [[ ! -v WORKSPACE ]] && [[ -d '.git' ]]; then
476 # project=$(git remote -v show | awk -F' ' '{print $2}' | xargs basename)
477 # project=voltctl.git
478
479 else
480 varname="$WORKSPACE/$GERRIT_PROJECT"
481 fi
482
483 if [ ! -f "$varname/Makefile" ]; then
484 :
485 elif [ ! -f "$varname/makefile" ]; then
486 :
487 else
488 error "Makefile not found at $varname!"
489 fi
490
491 return
492}
493
494## -----------------------------------------------------------------------
495## Intent: Display future enhancements
496## -----------------------------------------------------------------------
497function todo()
498{
499 local iam="${0##*/}"
500
501cat <<EOT
502
503** -----------------------------------------------------------------------
504** IAM: ${iam} :: ${FUNCNAME[0]}
505** -----------------------------------------------------------------------
506 o get_release_path()
507 - refactor redundant paths into local vars.
508 - see comments, do we have a one-off failure condition ?
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400509 o PATH += golang appended 3 times, release needs a single, reliable answer.
510 o do_login, do_logout and api calls do not use the my_gh wrapper:
511 - Add a lookup function to cache and retrieve path to downloaded gh command.
512
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400513EOT
514
515 return
516}
517
518## -----------------------------------------------------------------------
519## Intent: Copy files from the build directory into the release staging
520## directory for publishing to github releases/ endpoint.
Joey Armstrong26707f02023-01-26 12:41:12 -0500521## -----------------------------------------------------------------------
522function copyToRelease()
523{
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400524 func_echo "ENTER"
Joey Armstrong26707f02023-01-26 12:41:12 -0500525
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400526 local artifact_glob="${ARTIFACT_GLOB%/*}"
527 func_echo "$(declare -p artifact_glob)"
528
529 local work
530 get_release_dir work
531
532 ## Flatten filesystem, should we recurse here to release subdirs ?
533 # cp $(which ls) "$work"
534 # readarray -t arts < <(find "$artifact_glob" -type f)
535 readarray -t arts < <(find "$work" -type f)
Joey Armstrongeef8c2c2023-03-27 17:27:43 -0400536 declare -p args
537 # [[ ${#arts[@]} -eq 0 ]] && error "Artifact dir is empty, check for build failures in $artifact_glob"
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400538
Joey Armstrong26707f02023-01-26 12:41:12 -0500539 # Copy artifacts into the release temp dir
540 # shellcheck disable=SC2086
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400541 echo "rsync -rv --checksum \"$artifact_glob/.\" \"$work/.\""
542 rsync -rv --checksum "$artifact_glob/." "$work/."
Joey Armstrong1962bcf2023-01-27 13:53:18 -0500543
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400544 func_echo "LEAVE"
Joey Armstrong26707f02023-01-26 12:41:12 -0500545
Joey Armstrong26707f02023-01-26 12:41:12 -0500546 return
547}
548
549## -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400550## https://docs.github.com/en/rest/releases?apiVersion=2022-11-28
551 # https://cli.github.com/manual/gh_release_create
552 # --target <branch> or commit SHA
553 # --title
554 # --generate-notes
555 # --release-notes (notes file)
556 # --release
557 # release create dist/*.tgz
558 # --discussion-category "General"
Joey Armstrongf085d872023-01-28 17:52:29 -0500559## -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400560# https://cli.github.com/manual/gh_release_create
561## -----------------------------------------------------------------------
562function gh_release_create()
Joey Armstrong38bfeea2023-02-06 18:01:29 -0500563{
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400564 banner ''
Joey Armstrong38bfeea2023-02-06 18:01:29 -0500565
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400566 local version="$GIT_VERSION"
567# get_version 'version'
568# create_release_by_version "$version"
Joey Armstrong38bfeea2023-02-06 18:01:29 -0500569
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400570 declare -a args=()
571 args+=('--host-repo')
572 args+=('--notes' "'Testing release create -- ignore'")
573 args+=('--title')
574 if [[ -v draft_release ]]; then
575 args+=('--draft')
576 else
577 args+=('--discussion-category' 'Announcements')
578 fi
Joey Armstrong38bfeea2023-02-06 18:01:29 -0500579
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400580 local work
581 get_release_dir work
582
583 # pushd "$work" >/dev/null
584 readarray -t payload < <(find "$work" ! -type d -print)
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400585 func_echo "$gh_cmd release create ${version} ${args[@]}" "${payload[@]}"
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400586 my_gh 'release' 'create' "'$version'" "${args[@]}" "${payload[@]}"
587 # popd >/dev/null
588
Joey Armstrong38bfeea2023-02-06 18:01:29 -0500589 return
590}
591
592## -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400593## Intent: Authenticate credentials for a github gh session
Joey Armstrong38bfeea2023-02-06 18:01:29 -0500594## -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400595## NOTE: my_gh currently unused due to --with-token < "$pac"
596## -----------------------------------------------------------------------
597function do_login()
Joey Armstrongf085d872023-01-28 17:52:29 -0500598{
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400599 declare -g pac
Joey Armstrong485081f2023-03-27 13:34:08 -0400600 declare -a login_args=()
601 [[ $# -gt 0 ]] && login_args+=("$@")
Joey Armstrongf085d872023-01-28 17:52:29 -0500602
Joey Armstronge6e18eb2023-03-27 14:19:21 -0400603 func_echo "$(declare -p WORKSPACE)"
604
Joey Armstrong485081f2023-03-27 13:34:08 -0400605 # https://github.com/cli/cli/issues/2922#issuecomment-775027762
606 # (sigh) why not quietly return VS forcing a special case
Joey Armstronge6e18eb2023-03-27 14:19:21 -0400607 # [[ -v WORKSPACE ]] && [[ -v GITHUB_TOKEN ]] && return
608
609# 12:58:36 ** -----------------------------------------------------------------------
610# 12:58:36 ** jenkins582353203049151829.sh::do_login: --hostname github.com
611# 12:58:36 ** --------------------------------------------------------------------# ---
612# 12:58:36 ** jenkins582353203049151829.sh :: do_login: Detected ENV{GITHUB_TOKEN}=
613# 12:58:36 The value of the GITHUB_TOKEN environment variable is being used for authentication.
614# 12:58:36 To have GitHub CLI store credentials instead, first clear the value from the environment.
615 return
Joey Armstrong485081f2023-03-27 13:34:08 -0400616
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400617 # bridge to my_gh()
Joey Armstrong485081f2023-03-27 13:34:08 -0400618 get_gh_hostname login_args
Joey Armstrongf085d872023-01-28 17:52:29 -0500619
Joey Armstrong485081f2023-03-27 13:34:08 -0400620 banner "${login_args[@]}"
Joey Armstrongf085d872023-01-28 17:52:29 -0500621
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400622 ## Read from disk is safer than export GITHUB_TOKEN=
623 if [[ -v pac ]] && [[ ${#pac} -gt 0 ]]; then # interactive/debugging
624 [ ! -f "$pac" ] && error "PAC token file $pac does not exist"
625 # func_echo "--token file is $pac"
Joey Armstrong485081f2023-03-27 13:34:08 -0400626 "$gh_cmd" auth login "${login_args[@]}" --with-token < "$pac"
Joey Armstrongf085d872023-01-28 17:52:29 -0500627
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400628 elif [[ ! -v GITHUB_TOKEN ]]; then
629 error "--token [t] or GITHUB_TOKEN= are required"
Joey Armstrong41923cc2023-01-30 14:38:16 -0500630
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400631 else # jenkins
632 func_echo 'Detected ENV{GITHUB_TOKEN}='
Joey Armstrong485081f2023-03-27 13:34:08 -0400633 "$gh_cmd" auth login "${login_args[@]}"
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400634 fi
Joey Armstrongf085d872023-01-28 17:52:29 -0500635
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400636 declare -i -g active_login=1 # signal logout
Joey Armstrongf085d872023-01-28 17:52:29 -0500637
Joey Armstrongf085d872023-01-28 17:52:29 -0500638 return
639}
640
641## -----------------------------------------------------------------------
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400642## Intent: Destroy credentials/gh session authenticated by do_login
643## -----------------------------------------------------------------------
644## NOTE: my_gh currently unused due to "<<< 'y'"
645## -----------------------------------------------------------------------
646function do_logout()
647{
648 declare -i -g active_login
649 [[ ! -v active_login ]] && return
650
651 declare -a out_args=()
652 [[ $# -gt 0 ]] && out_args+=("$@")
653
654 # bridge to my_gh()
655
656 get_gh_hostname in_args
657
658 banner "${out_args[@]}"
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400659 "$gh_cmd" auth logout "${out_args[@]}" <<< 'Y'
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400660
661 unset active_login
662 return
663}
664
665## -----------------------------------------------------------------------
666## Intent: Query for repository version strings
667## -----------------------------------------------------------------------
668function get_releases()
669{
670 declare -n ref="$1"; shift
671 local func="${FUNCNAME[0]}"
672
673 banner ""
674 pushd "$scratch" >/dev/null
675
676 # gh api repos/{owner}/{repo}/releases
677 local releases_uri
678 get_gh_releases releases_uri
679 declare -p releases_uri
680
681 ref=()
Joey Armstrongad7b1e02023-03-27 11:55:48 -0400682 "$gh_cmd" api "$releases_uri" "${common[@]}" | jq . > 'release.raw'
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400683 readarray -t __tmp < <(jq '.[] | "\(.tag_name)"' 'release.raw')
684
685 local release
686 for release in "${__tmp[@]}";
687 do
688 release="${release//\"/}"
689 ref+=("$release")
690 done
691
692 popd >/dev/null
693 return
694}
695
696## -----------------------------------------------------------------------
697## Intent: Display repository query strings.
698## Indirect: verify authentication and API
699## -----------------------------------------------------------------------
700function showReleases()
701{
702 declare -a releases=()
703 get_releases releases
704
705 local release
706 for release in "${releases[@]}";
707 do
708 func_echo "$release"
709 done
710 return
711}
712
713## -----------------------------------------------------------------------
714## Intent: Install the gh command line tool locally
Joey Armstrong26707f02023-01-26 12:41:12 -0500715## -----------------------------------------------------------------------
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500716function install_gh_binary()
Joey Armstrong26707f02023-01-26 12:41:12 -0500717{
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400718 banner
Joey Armstrong26707f02023-01-26 12:41:12 -0500719
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500720 pushd "$scratch"
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400721 func_echo "Retrieve latest gh download URLs"
Joey Armstrongf085d872023-01-28 17:52:29 -0500722
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500723 local latest="https://github.com/cli/cli/releases/latest"
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400724 local tarball="gh.tar.tgz"
725
726 readarray -t latest < <(curl --silent -qI "$latest" \
727 | awk -F '/' '/^location/ {print substr($NF, 1, length($NF)-1)}')
728 declare -p latest
729 if [ ${#latest[@]} -ne 1 ]; then
730 error "Unable to determine latest gh package version"
731 fi
732
733 local VER="${latest[0]}"
734
735 func_echo "Download latest gh binary"
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500736 local url="https://github.com/cli/cli/releases/download/${VER}/gh_${VER#v}_linux_amd64.tar.gz"
737 wget --quiet --output-document="$tarball" "$url"
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400738
739 func_echo "Unpack tarball"
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500740 tar zxf "$tarball"
741
742 gh_cmd="$(find "${scratch}" -name 'gh' -print)"
743 readonly gh_cmd
744
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400745 func_echo "Command: ${gh_cmd}"
746 func_echo "Version: $("$gh_cmd" --version)"
Joey Armstrongd99e3d22023-01-29 12:35:43 -0500747 popd
748
Joey Armstrong2097d3e2023-03-26 10:32:03 -0400749 return
750}
751
752## -----------------------------------------------------------------------
753## Intent: Danger Will Robinson
754## -----------------------------------------------------------------------
755function releaseDelete()
756{
757 local version="$1"; shift
758
759 banner "${in_args[@]}"
760 declare -a args=()
761 args+=('--host-repo')
762 args+=('--yes')
763 # args+=('--cleanup-tag')
764
765# ** github-release.sh :: get_argv_repo: VARNAME=opencord/voltctl
766#* Running: /tmp/github-release.R7geZo7Ywo/gh_2.25.1_linux_amd64/bin/gh release delete v4.175.710 --repo 'github.com/opencord/voltctl' --yes --cleanup-tag
767#rror connecting to 'github.com
768#heck your internet connection or https://githubstatus.com
769
770 echo
771 echo "==========================================================================="
772 my_gh 'release' 'delete' "$version" "${args[@]}"
773 echo "==========================================================================="
774 echo
775
776 showReleases
777 return
778}
779
780## -----------------------------------------------------------------------
781## Intent: Copy binaries into temp/release, checksum then publish
782## -----------------------------------------------------------------------
783function release_staging()
784{
785 local release_temp
786 get_release_dir release_temp
787
788 banner ''
789 func_echo "Packaging release files"
790 # pushd "$RELEASE_TEMP"
791 pushd "$release_temp" >/dev/null || error "pushd failed: dir is [$release_temp]"
792
793 readarray -t to_release < <(find . -mindepth 1 -maxdepth 1 -type f -print)
794 func_echo "Files to release: $(declare -p to_release)"
795
796 # Generate and check checksums
797 sha256sum -- * > checksum.SHA256
798 sha256sum -c < checksum.SHA256
799
800 echo
801 func_echo "Checksums(checksum.SHA256):"
802 cat checksum.SHA256
803
804 ## ARGS NEEDED
805 gh_release_create
806
807 popd >/dev/null || error "pushd failed: dir is [$release_temp]"
808
809 return
810}
811
812## -----------------------------------------------------------------------
813## Intent: Display program usage
814## -----------------------------------------------------------------------
815function usage()
816{
817 [[ $# -gt 0 ]] && func_echo "$*"
818
819 cat <<EOH
820Usage: github-release.sh [options] [target] ...
821
822[Github CLI (gh) arguments]
823 --login Perform authentication using a PAC
824 --logout
825 --host [h] Specify github server for connection.
826
827[Options]
828 --token Login debugging, alternative to env var use.
829
830[Modes]
831 --debug Enable script debug mode
832 --verbose Enable script verbose mode
833
834All remaining arguments are passthrough to the gh command.
835EOH
836
837 exit 0
838}
839
840## -----------------------------------------------------------------------
841## Intent: Provide common arguments and access to the gh command.
842## o Cache path to the gh command
843## o Construct a gh command line from given args
844## o Command wrapper can provide defaults (--hostname github.com)
845## -----------------------------------------------------------------------
846## Given:
847## scalar Array variable name (declare -n is a reference)
848## Return:
849## ref gh command line passed back to caller
850## Switches:
851## --host Pass default github hostname
852## --verbose Enable verbose mode#
853## --version Display command version
854## @array Remaining arguments passed as command switches.
855## -----------------------------------------------------------------------
856## See Also:
857## o https://cli.github.com/manual
858## -----------------------------------------------------------------------
859function my_gh()
860{
861 ## ------------------------
862 ## Cache path to gh command
863 ## ------------------------
864 if [[ ! -v gh_cmd ]]; then
865 readarray -t cmds < <(which -a gh)
866 declare -p cmds
867 if [ ${#cmds} -eq 0 ]; then
868 error "Unable to locate the gh command"
869 fi
870 gh_cmd="${cmds[0]}"
871 readonly gh_cmd
872 fi
873
874 declare -a cmd=()
875 cmd+=("$gh_cmd")
876
877 ## ----------------------
878 ## Construct command line
879 ## ----------------------
880 # shellcheck disable=SC2034
881 declare -A action=() # pending operations
882 declare -a args=() # common gh command line args
883
884 while [ $# -gt 0 ]; do
885 local arg="$1"; shift
886 case "$arg" in
887
888 # Modes
889 -*debug) declare -i -g debug=1 ;;
890 -*help) show_help ;;
891 -*verbose) args+=('--verbose') ;;
892
893 -*hostname)
894 get_gh_hostname in_args
895 ;;
896
897 --host-repo)
898 local val
899 get_argv_repo val
900
901 # --repo <[HOST/]OWNER/REPO>
902 args+=('--repo' "'${__githost}/${val}'")
903 ;;
904
905 --repo)
906 local val
907 get_argv_repo val
908 args+=('--repo' "$val")
909 ;;
910
911 --tag)
912 local val
913 get_argv_tag val
914 args+=("'$val'") # No switch, pass inline
915 ;;
916
917 --title)
918 local val
919 get_argv_name val
920 args+=('--title' "'$val'")
921 ;;
922
923 # --draft
924 # --latest
925 *) args+=("$arg") ;;
926 esac
927 done
928
929 cmd+=("${args[@]}")
930 echo "** Running: ${cmd[*]}"
931 "${cmd[@]}"
932 local status=$?
933
934 [[ $status -eq 0 ]] && { true; } || { false; }
935 return
936}
937
938## ---------------------------------------------------------------------------
939## Intent:
940## ---------------------------------------------------------------------------
941function usage()
942{
943 cat <<EOH
944
945Usage: $0
946Usage: make [options] [target] ...
947 --help This mesage
948 --pac Personal Access Token (file)
949
950[DEBUG]
951 --gen-version Generate a random release version string.
952 --git-hostname Git server hostname (default=github.com)
953
954EOH
955 return
956}
957
958## ---------------------------------------------------------------------------
959## Intent: Parse script command line arguments
960## ---------------------------------------------------------------------------
961function parse_args()
962{
963 # declare -n ref="$1"; shift
964
965 [[ -v DEBUG ]] && func_echo "ENTER"
966
967# ref="repos/${__repo_user}/${__repo_name}/releases"
968
969 while [ $# -gt 0 ]; do
970 local arg="$1"; shift
971 case "$arg" in
972 -*gen-version)
973 get_version GIT_VERSION
974 ;;
975
976 -*git-hostname)
977 __githost="$1"; shift
978 ;;
979
980 -*repo-name)
981 __repo_name="$1"; shift
982 ;;
983
984 -*repo-org)
985 __repo_org="$1"; shift
986 ;;
987
988 -*pac)
989 declare -g pac="$1"; shift
990 readonly pac
991 [[ ! -f "$pac" ]] && error "--token= does not exist ($pac)"
992 : # nop/true
993 ;;
994
995 -*help)
996 usage
997 exit 0
998 ;;
999 *) error "Detected unknown argument $arg" ;;
1000 esac
1001 done
1002
Joey Armstrong26707f02023-01-26 12:41:12 -05001003 return
1004}
1005
Joey Armstrongaf577ab2022-12-15 14:43:33 -05001006##----------------##
1007##---] MAIN [---##
1008##----------------##
Joey Armstrong1962bcf2023-01-27 13:53:18 -05001009iam="${0##*/}"
1010
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001011full_banner
1012parse_args $@
Joey Armstrongd99e3d22023-01-29 12:35:43 -05001013init
1014install_gh_binary
Joey Armstrong7f382ef2023-01-25 12:00:08 -05001015
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001016do_login
Zack Williams27cd3e52018-09-18 16:44:50 -07001017
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001018release_path='/dev/null'
1019get_release_path release_path
1020declare -p release_path
Zack Williams27cd3e52018-09-18 16:44:50 -07001021
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001022pushd "$release_path" || error "pushd failed: dir is [$release_path]"
Zack Williams27cd3e52018-09-18 16:44:50 -07001023
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001024 # legacy: getGitVersion "$GERRIT_PROJECT" GIT_VERSION
1025 getGitVersion GIT_VERSION
1026 getReleaseDescription RELEASE_DESCRIPTION
Zack Williams27cd3e52018-09-18 16:44:50 -07001027
Joey Armstrongeef8c2c2023-03-27 17:27:43 -04001028 cat <<EOM
1029
1030** -----------------------------------------------------------------------
1031** GIT VERSION: $(declare -p GIT_VERSION)
1032** RELEASE_DESCRIPTION: $(declare -p RELEASE_DESCRIPTION)
1033** RELEASE_TARGETS: $(declare -p RELEASE_TARGETS)
1034** make release
1035** -----------------------------------------------------------------------
1036EOM
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001037 # build the release, can be multiple space separated targets
1038 # -----------------------------------------------------------------------
1039 # % go build command-line-arguments:
1040 # copying /tmp/go-build4212845548/b001/exe/a.out:
1041 # open release/voltctl-1.8.25-linux-amd64: permission denied
1042 # missing: docker run mkdir
1043 # -----------------------------------------------------------------------
1044 # shellcheck disable=SC2086
Joey Armstrongeef8c2c2023-03-27 17:27:43 -04001045 make "$RELEASE_TARGETS"
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001046 copyToRelease
Joey Armstrong50f6e0b2023-01-24 14:14:08 -05001047
Joey Armstrong26707f02023-01-26 12:41:12 -05001048 cat <<EOM
Joey Armstrong1962bcf2023-01-27 13:53:18 -05001049
1050** -----------------------------------------------------------------------
1051** Create the release:
1052** 1) Create initial github release with download area.
1053** 2) Generate checksum.SHA256 for all released files.
1054** 3) Upload files to complete the release.
1055** 4) Display released info from github.
1056** -----------------------------------------------------------------------
Joey Armstrong26707f02023-01-26 12:41:12 -05001057EOM
1058
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001059 showReleases
1060 # releaseDelete 'v4.175.710'
1061 release_staging
1062 popd || error "pushd failed: dir is [$release_path]"
Zack Williams27cd3e52018-09-18 16:44:50 -07001063fi
Joey Armstrong28eddda2023-01-10 03:09:34 -05001064
Joey Armstrong2097d3e2023-03-26 10:32:03 -04001065do_logout
1066
Joey Armstrong28eddda2023-01-10 03:09:34 -05001067# [SEE ALSO]
1068# -----------------------------------------------------------------------
1069# https://www.shellcheck.net/wiki/SC2236
Joey Armstrong26707f02023-01-26 12:41:12 -05001070# https://docs.github.com/en/rest/releases/releases?apiVersion=2022-11-28#create-a-release
Joey Armstrong28eddda2023-01-10 03:09:34 -05001071# -----------------------------------------------------------------------
Joey Armstrongf085d872023-01-28 17:52:29 -05001072# https://cli.github.com/manual/gh_help_reference
1073# https://cli.github.com/manual/gh_release
1074# -----------------------------------------------------------------------
Joey Armstrong28eddda2023-01-10 03:09:34 -05001075
1076# [EOF]