-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbuild_info.sh
executable file
·225 lines (197 loc) · 9.13 KB
/
build_info.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
#!/bin/bash
#set -uo pipefail
#DEBUG_SCRIPT=1
# contains(string, substring)
#
# Returns 0 if the specified string contains the specified substring,
# otherwise returns 1.
contains() {
string="$1"
substring="$2"
if test "${string#*$substring}" != "$string"
then
return 0 # $substring is in $string
else
return 1 # $substring is not in $string
fi
}
## Создаёт или редактирует файл build_info.h
## Переписывает файл только если что-то изменилось, благодаря чему модули C/C++, включающие build_info.h не будут перекопилированы.
## TODO Ignore new submodules. Актуально при возврате на предыдущие коммиты, чтоб не числились как Dirty
## TODO Стоит ли писать detached?
scriptName=$(basename "${0}")
if [[ "$1" == "-h" ]]; then
echo -e "Usage:\n"\
" $scriptName [GitRepo] [TargetFile] \n" \
" GitRepo Any path under Git VCS. If not specified, uses current folder \n" \
" TargetFile Optional, default is $GitRepoTopLevelPath/src/build_info.h "
exit 1
fi
## Set GitRepo
if [[ -n "$1" ]]; then
GitRepo=$1
if [ ! -d "$GitRepo" ]; then
echo "GitRepo $GitRepo is not a directory" >> /dev/stderr
exit 1
fi
else
GitRepo=$(pwd)
fi
## git command with directory prefix. Todo replace all `git -C "$GitRepo"` to `$GIT` and test.
GIT="git -C "$GitRepo
DEBUG_SCRIPT=${DEBUG_SCRIPT:=0}
if (( $DEBUG_SCRIPT )); then
echo "GIT : $GIT"
fi
## Check if $GitRepo is inside Git repository ###--is-inside-git-dir
$($GIT rev-parse)
if [ $? != 0 ]; then
echo "GitRepo \"${GitRepo}\" is not valid" >> /dev/stderr
exit $?;
fi
## Set TargetFile
if [ -n "$2" ]; then
TargetFile=$2
else
## Default TargetFile.
GitRepoTopLevelPath=$($GIT rev-parse --show-toplevel)
contains "$GitRepoTopLevelPath" "fatal" && \
echo "$GitRepoTopLevelPath" && exit
TargetFile="$GitRepoTopLevelPath/src/build_info.h"
fi
if (( $DEBUG_SCRIPT )); then
echo GitRepo: $GitRepo
echo GitRepoTopLevelPath: $GitRepoTopLevelPath
echo TargetFile: $TargetFile
else
StdErrMod="2>/dev/null" ## To be used. Do nothing when debug and hide messages when working normally.
fi
temppath=`mktemp` #$(mktemp -p /dev/shm/) ## Файл по идее в ОЗУ http://unix.stackexchange.com/a/188537/156608
short=$($GIT rev-parse --short HEAD)
long=$($GIT rev-parse HEAD) #git -C $GitRepo show-ref -h HEAD
count=$($GIT rev-list --count --first-parent ${BuildInfo_RevName:=HEAD})
## Checking for a dirty index or untracked files with Git
## [1] https://stackoverflow.com/questions/2657935/checking-for-a-dirty-index-or-untracked-files-with-git#2659808
## [2] https://unix.stackexchange.com/questions/155046/determine-if-git-working-directory-is-clean-from-a-script
#$GIT diff-index --quiet HEAD --; ## This somehow gives different results under Linux and Windows on the same working tree. It does not see newly created files.
$GIT diff --exit-code >/dev/null ## Does not see new (untracked) files.
isDirty=$?
if (( $DEBUG_SCRIPT )); then
echo isDirty : $isDirty
fi
[[ $isDirty -ne 0 ]] && dirty="dirty" || dirty=""
[[ $isDirty -ne 0 ]] && _dirty="-dirty" || _dirty=""
#if [[ -n $($GIT status --porcelain) ]]; then dirty="dirty"; else dirty=""; fi
#if [ $isDirty -ne 0 ] ; then dirty="dirty" ; else dirty=""; fi
#if [ $isDirty -ne 0 ] ; then _dirty="-dirty"; else _dirty=""; fi
## Note: ${$dirty:+-$dirty} expands to nothing when dirty is empty or undefined, and prepends '-' else.
## Записать тэг.
tag=$($GIT tag --list --points-at HEAD)
if [ "$tag" ] ; then
#tag_=$tag${dirty:+-$dirty}
if [ "$dirty" ] ; then
tag_=$tag
else
tag_=$tag-$dirty
fi
fi
## Записать ветку.
## Check detached
if [ -z $($GIT symbolic-ref HEAD -q) ]; then
branch="DETACHED"
else
#branch=$($GIT branch --list --points-at HEAD | grep "^* .*")
#branch=${branch:2} ## Текущая ветка отмечена *. Предполагается результат "* branch", убрать первые 2 символа.
#branch=$($GIT name-rev --name-only HEAD) ## Возвращает первую попавшую ветку.
branch=$($GIT rev-parse --abbrev-ref HEAD) ## Show only the current branch, no parsing required
fi
if [ "$branch" ] ; then
#branch_=$branch${dirty:+-$dirty}
if [ "$dirty" ] ; then
branch_=$branch-$dirty
else
branch_=$branch
fi
fi
## Header
echo -e "/**\n * This file was created automatically by script build_info.sh.\n * DO NOT EDIT! \n */\n" > $temppath
echo -e "#pragma once\n" >> $temppath
echo -e "#define BUILD_GIT_SHORT \"$short\"" >> $temppath
echo -e "#define BUILD_GIT_LONG \"$long\"" >> $temppath
echo -e "#define BUILD_GIT_COUNT \"$count\"" >> $temppath
## if result is not "" (working tree is not clean) then dirty.
echo -e "#define BUILD_GIT_DIRTY \"$dirty\"" >> $temppath
echo -e "#define BUILD_GIT_DIRTY_ \"$_dirty\"" >> $temppath
## Записать тэг.
echo -e "#define BUILD_GIT_TAG \"$tag_\"" >> $temppath
## Записать ветку.
echo -e "#define BUILD_GIT_BRANCH \"$branch_\"" >> $temppath
## Результирующие
if [ "$tag" ] ; then
build_git="$tag-c$count($short)$_dirty"
echo -e "#define BUILD_GIT_ BUILD_GIT_TAG \"-c\" BUILD_GIT_COUNT \"(\" BUILD_GIT_SHORT \")\" BUILD_GIT_DIRTY_" >> $temppath
echo -e "#define BUILD_GIT \"$build_git\"" >> $temppath
echo "$build_git, branch:$branch" ## Сообщить результат в консоль
else
build_git="$branch-c$count($short)$_dirty"
echo -e "#define BUILD_GIT_ BUILD_GIT_BRANCH \"-c\" BUILD_GIT_COUNT \"(\" BUILD_GIT_SHORT \")\" BUILD_GIT_DIRTY_" >> $temppath
echo -e "#define BUILD_GIT \"$build_git\"" >> $temppath
echo "$build_git" ## Сообщить результат в консоль
fi
echo -e "//#define BUILD_INFO \"Build \" __DATE__ \" \" __TIME__ \" Git \" BUILD_GIT" >> $temppath
echo -e "#define BUILD_INFO \"Build \" BUILD_DATE_ISO8601 \" Git \" BUILD_GIT" >> $temppath
## Отдельный файл с текущими датой и времемем
## Separate file with current date and time
## The output should be like:
## ```
## #define BUILD_DATE_ISO8601 "2018-02-06T18:16:51+02:00"
## #define BUILD_EPOCH1970_SEC 1517933811
## #define BUILD_DATE "2018-02-06"
## #define BUILD_TIME "18:16:51"
## #define BUILD_DATE_TIME "2018-02-06 18:16:51"
## #define BUILD_YEAR 2018
## #define BUILD_MONTH 02
## #define BUILD_DAY 06
## #define BUILD_HOUR 18
## #define BUILD_MIN 16
## #define BUILD_SEC 51
## #define BUILD_NANOSEC 135301337
## ```
temp_datetime=`mktemp` #"$(mktemp -p /dev/shm/)" ## Файл по идее в ОЗУ http://unix.stackexchange.com/a/188537/156608
epoch1970sec=$(date +%s)
echo -e "" > $temp_datetime
echo -e "#define BUILD_DATE_ISO8601 \"$(date --iso-8601=seconds)\"" >> $temp_datetime
echo -e "#define BUILD_DATE_RFC3339 \"$(date --rfc-3339=seconds)\"" >> $temp_datetime
echo -e "#define BUILD_EPOCH1970_SEC $epoch1970sec " >> $temp_datetime
echo -e "#define BUILD_DATE \"$(date +%F)\"" >> $temp_datetime
echo -e "#define BUILD_TIME \"$(date +%T)\"" >> $temp_datetime
echo -e "#define BUILD_DATE_TIME \"$(date +'%F %T')\"" >> $temp_datetime
echo -e "#define BUILD_YEAR $(date +%Y)" >> $temp_datetime
echo -e "#define BUILD_MONTH $(date +%m)" >> $temp_datetime
echo -e "#define BUILD_DAY $(date +%d)" >> $temp_datetime
echo -e "#define BUILD_HOUR $(date +%H)" >> $temp_datetime
echo -e "#define BUILD_MIN $(date +%M)" >> $temp_datetime
echo -e "#define BUILD_SEC $(date +%S)" >> $temp_datetime
echo -e "#define BUILD_NANOSEC $(date +%N)" >> $temp_datetime
# extract linux epoch timestamp from prev file. `-F` is field-separator, `2>` is stderr redirection.
epoch1970sec_prev=`grep BUILD_EPOCH1970 $TargetFile 2>/dev/null | awk -F " " '{print $3}'`
if (( $DEBUG_SCRIPT )); then
echo BUILD_EPOCH1970_SEC $epoch1970sec
echo BUILD_EPOCH1970_SEC PREV $epoch1970sec_prev
fi
## Использовать предыдующие дату и время, если разница во времени меньше 10 минут
RequireTimeDiffSeconds=${RequireTimeDiffSeconds:-600} ## Use default value 600s = 10 mims
if (( epoch1970sec - epoch1970sec_prev < $RequireTimeDiffSeconds )); then
echo -e "" > $temp_datetime
# extract date-time block from prev file
awk -v RS="" '/#define BUILD_DATE.*/' $TargetFile >> $temp_datetime
fi
cat $temp_datetime >> $temppath
## ToDo Всё равно обновлять время если файлы были изменены. Например, учитывать вывод `git status`.
## Копировать файл если есть изменения. &>/dev/null для вывода stdout и stderr в никуда.
if diff --brief "$temppath" "$TargetFile" &>/dev/null ; then
echo Nothing to change
else
cp "$temppath" "$TargetFile" && echo Written to "$TargetFile" || (echo Failed writing to "$TargetFile" >> /dev/stderr; exit $?)
fi
rm "$temppath"