Simple Makefile target helper https://pagure.io/mkit

facts.sh 5.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. #!/bin/bash
  2. . "$MKIT_DIR/include/ini.sh" || die "cannot import ini.sh"
  3. git_fact() {
  4. #
  5. # Get git fact $1
  6. #
  7. local fact_name="$1"
  8. git_present || warn "can't give fact outside git repo: $fact_name"
  9. case "$fact_name" in
  10. latest_tag)
  11. git log --format="%D" \
  12. | sed 's/,/\n/g' \
  13. | sed 's/^[[:blank:]]*//; ' \
  14. | grep -E '^tag: v[[:digit:]]+\.' \
  15. | cut -d' ' -f2 \
  16. | head -1
  17. ;;
  18. latest_version)
  19. git_fact latest_tag | git_tag2ver
  20. ;;
  21. current_branch)
  22. git rev-parse --abbrev-ref HEAD
  23. ;;
  24. reldiff)
  25. git log --oneline "$(git_fact latest_tag)..HEAD" --name-only
  26. ;;
  27. latest_sha)
  28. git log -1 --pretty=format:%h HEAD
  29. ;;
  30. *)
  31. warn "unknown git fact asked: $fact_name"
  32. ;;
  33. esac
  34. }
  35. git_present() {
  36. #
  37. # True if we're in a git repo
  38. #
  39. git rev-parse HEAD >&/dev/null
  40. }
  41. git_tag2ver() {
  42. #
  43. # Convert tag to version
  44. #
  45. sed s/^v//
  46. }
  47. git_ver2tag() {
  48. #
  49. # Convert version to tag
  50. #
  51. sed s/^/v/
  52. }
  53. git_lasthash() {
  54. #
  55. # Show last commit hash (with .dirty suffix if needed)
  56. #
  57. # If outside git repo, get it from .mkit/git_lasthash, which
  58. # should have been put there by dist target. (I.e., this won't
  59. # work if you got outside the git repo in other way than dist
  60. # target, but that's actually expected.)
  61. #
  62. if git_present;
  63. then # we are in git repo
  64. local last_hash=$(git rev-parse HEAD)
  65. if test "$(git diff --shortstat 2>/dev/null)" != "";
  66. then # the tree is "dirty", i.e. has been edited
  67. local suffix=.dirty
  68. fi
  69. local suffix=""
  70. echo -n "$last_hash$suffix"
  71. else # we are outside (eg. distributor's build dir')
  72. grep . .mkit/git_lasthash || {
  73. echo UNKNOWN
  74. warn "malformed source, could not determine git hash"
  75. }
  76. fi
  77. }
  78. semver() {
  79. #
  80. # Build semver version string with build metadata
  81. #
  82. # Build version string from available info using following
  83. # logic:
  84. #
  85. # 1. use project.version (from mkit.ini)
  86. # 2. if we are in git, override the version with last tag
  87. # 3. if set, add project:prerl (from mkit.ini) as pre-release ID
  88. # (afer dash)
  89. # 4. if we are at a later commit than the last tag, add branch
  90. # name and commit sha1 to build metadata (after plus sign)
  91. # 5. if the tree is "dirty", i.e. has uncommited changes,
  92. # add "dirty" to build metadata
  93. #
  94. # The version is compatible with SemVer 2.0.0.
  95. #
  96. # Examples:
  97. #
  98. # myprog v1.0.7 # all clear
  99. # myprog v1.0.7-alpha # mkit.ini: project:prerl="alpha"
  100. # myprog v1.0.7-alpha+g1aef811.master # ^^ + some commits after
  101. # myprog v1.0.7-alpha+gf14fc4f.api2 # ^^ + on a feature branch
  102. # myprog v1.0.7-alpha+gf14fc4f.api2.dirty # ^^ + tree edited
  103. # myprog v1.0.7-alpha+dirty # tag OK but tree edited
  104. # myprog v1.0.7+dirty # ^^ but no pre-release id
  105. #
  106. # Note that versions with "dirty" should be perceived as kind of
  107. # dangerous outside developer's own machine. Versions with sha1 are
  108. # safer but must not be released.
  109. #
  110. # I have considered decorating the git commit refs to make them
  111. # sort of sortable (e.g. "r1.g1aef811"), but on second thought,
  112. # I don't think it's good idea to give *any* semantics to meta-data
  113. # at all. First, there is no rule that r1<r2<r3; a commit can be
  114. # removing what other just added and one change can be split to
  115. # multiple commits. Also, the whole thing breaks anyway once you
  116. # rebase your branch (no, it's not a sin). The sole purpose of
  117. # meta-data is to *identify* the code, and provide safe path back
  118. # to tree; commit refs are already perfect for that.
  119. #
  120. # FIXME: Using project:prerl for release IDs may not be compatible with
  121. # release strategy implemented in release.sh
  122. #
  123. local version=$(ini 1value project:version)
  124. local prerl=$(ini 1value project:prerl)
  125. grep ":" <<<"$prerl" && warn "colon in project:prerl may corrupt version data: $prerl"
  126. if git_present;
  127. then # we are in git repo... so we can get smart
  128. local latest_tag=$(git_fact latest_tag)
  129. if ! git describe --tags --exact-match HEAD >&/dev/null;
  130. then # we are at a later commit than the last tag
  131. local commit="$(git_fact current_branch).g$(git_fact latest_sha)"
  132. fi
  133. if test "$(git diff --shortstat 2>/dev/null)" != "";
  134. then # the tree is "dirty", i.e. has been edited
  135. local dirty=dirty
  136. fi
  137. test -n "$latest_tag" && version=${latest_tag:1}
  138. local suffix=""
  139. case "$commit:$dirty" in
  140. :) suffix="" ;;
  141. :dirty) suffix="+$dirty" ;;
  142. *:) suffix="+$commit" ;;
  143. *:dirty) suffix="+$commit.$dirty" ;;
  144. esac
  145. test -b "$prerl" && suffix="-$prerl$suffix"
  146. version="$version$suffix"
  147. fi
  148. echo "$version"
  149. }