Skip to content

Commit b3c00e4

Browse files
authored
Docker: Build and use musl-based binaries in alpine images to resolve glibc incompatibility issues (grafana#19798)
* build: Install musl cross compilers as part of build Docker image * build: Build also musl packages in scripts/build/build.sh * scripts/build/build-all.sh: Build musl Linux targets * build: Upgrade build-container to 1.2.11 * build.go: De-duplicate code * build: Base Docker images on musl binaries
1 parent 63b561b commit b3c00e4

File tree

11 files changed

+99
-42
lines changed

11 files changed

+99
-42
lines changed

.circleci/config.yml

+8-8
Original file line numberDiff line numberDiff line change
@@ -195,7 +195,7 @@ jobs:
195195

196196
build-all:
197197
docker:
198-
- image: grafana/build-container:1.2.10
198+
- image: grafana/build-container:1.2.11
199199
working_directory: /go/src/github.com/grafana/grafana
200200
steps:
201201
- checkout
@@ -239,7 +239,7 @@ jobs:
239239

240240
build:
241241
docker:
242-
- image: grafana/build-container:1.2.10
242+
- image: grafana/build-container:1.2.11
243243
working_directory: /go/src/github.com/grafana/grafana
244244
steps:
245245
- checkout
@@ -268,7 +268,7 @@ jobs:
268268

269269
build-fast-backend:
270270
docker:
271-
- image: grafana/build-container:1.2.10
271+
- image: grafana/build-container:1.2.11
272272
working_directory: /go/src/github.com/grafana/grafana
273273
steps:
274274
- checkout
@@ -285,7 +285,7 @@ jobs:
285285

286286
build-fast-frontend:
287287
docker:
288-
- image: grafana/build-container:1.2.10
288+
- image: grafana/build-container:1.2.11
289289
working_directory: /go/src/github.com/grafana/grafana
290290
steps:
291291
- checkout
@@ -309,7 +309,7 @@ jobs:
309309

310310
build-fast-package:
311311
docker:
312-
- image: grafana/build-container:1.2.10
312+
- image: grafana/build-container:1.2.11
313313
working_directory: /go/src/github.com/grafana/grafana
314314
steps:
315315
- checkout
@@ -336,7 +336,7 @@ jobs:
336336

337337
build-fast-save:
338338
docker:
339-
- image: grafana/build-container:1.2.10
339+
- image: grafana/build-container:1.2.11
340340
working_directory: /go/src/github.com/grafana/grafana
341341
steps:
342342
- checkout
@@ -425,7 +425,7 @@ jobs:
425425

426426
build-enterprise:
427427
docker:
428-
- image: grafana/build-container:1.2.10
428+
- image: grafana/build-container:1.2.11
429429
working_directory: /go/src/github.com/grafana/grafana
430430
steps:
431431
- checkout
@@ -460,7 +460,7 @@ jobs:
460460

461461
build-all-enterprise:
462462
docker:
463-
- image: grafana/build-container:1.2.10
463+
- image: grafana/build-container:1.2.11
464464
working_directory: /go/src/github.com/grafana/grafana
465465
steps:
466466
- checkout

Gruntfile.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ module.exports = function (grunt) {
1010
tempDir: 'tmp',
1111
platform: process.platform.replace('win32', 'windows'),
1212
enterprise: false,
13+
libc: null,
1314
};
1415

1516
if (grunt.option('platform')) {
@@ -30,6 +31,10 @@ module.exports = function (grunt) {
3031
}
3132
}
3233

34+
if (grunt.option('libc')) {
35+
config.libc = grunt.option('libc');
36+
}
37+
3338
config.phjs = grunt.option('phjsToRelease');
3439
config.pkg.version = grunt.option('pkgVer') || config.pkg.version;
3540

@@ -42,7 +47,7 @@ module.exports = function (grunt) {
4247
grunt.loadTasks('./scripts/grunt');
4348

4449
// Utility function to load plugin settings into config
45-
function loadConfig(config,path) {
50+
function loadConfig(config, path) {
4651
require('glob').sync('*', {cwd: path}).forEach(function(option) {
4752
var key = option.replace(/\.js$/,'');
4853
// If key already exists, extend it. It is your responsibility to avoid naming collisions

build.go

+26-14
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@ var (
3333
goos string
3434
gocc string
3535
cgo bool
36+
libc string
3637
pkgArch string
3738
version string = "v1"
3839
// deb & rpm does not support semver so have to handle their version a little differently
@@ -65,6 +66,7 @@ func main() {
6566
flag.StringVar(&goarch, "goarch", runtime.GOARCH, "GOARCH")
6667
flag.StringVar(&goos, "goos", runtime.GOOS, "GOOS")
6768
flag.StringVar(&gocc, "cc", "", "CC")
69+
flag.StringVar(&libc, "libc", "", "LIBC")
6870
flag.BoolVar(&cgo, "cgo-enabled", cgo, "Enable cgo")
6971
flag.StringVar(&pkgArch, "pkg-arch", "", "PKG ARCH")
7072
flag.StringVar(&phjsToRelease, "phjs", "", "PhantomJS binary")
@@ -106,18 +108,14 @@ func main() {
106108
case "setup":
107109
setup()
108110

109-
case "build-srv":
111+
case "build-srv", "build-server":
110112
clean()
111113
build("grafana-server", "./pkg/cmd/grafana-server", []string{})
112114

113115
case "build-cli":
114116
clean()
115117
build("grafana-cli", "./pkg/cmd/grafana-cli", []string{})
116118

117-
case "build-server":
118-
clean()
119-
build("grafana-server", "./pkg/cmd/grafana-server", []string{})
120-
121119
case "build":
122120
//clean()
123121
for _, binary := range binaries {
@@ -176,12 +174,15 @@ func makeLatestDistCopies() {
176174
}
177175

178176
latestMapping := map[string]string{
179-
"_amd64.deb": "dist/grafana_latest_amd64.deb",
180-
".x86_64.rpm": "dist/grafana-latest-1.x86_64.rpm",
181-
".linux-amd64.tar.gz": "dist/grafana-latest.linux-x64.tar.gz",
182-
".linux-armv7.tar.gz": "dist/grafana-latest.linux-armv7.tar.gz",
183-
".linux-armv6.tar.gz": "dist/grafana-latest.linux-armv6.tar.gz",
184-
".linux-arm64.tar.gz": "dist/grafana-latest.linux-arm64.tar.gz",
177+
"_amd64.deb": "dist/grafana_latest_amd64.deb",
178+
".x86_64.rpm": "dist/grafana-latest-1.x86_64.rpm",
179+
".linux-amd64.tar.gz": "dist/grafana-latest.linux-x64.tar.gz",
180+
".linux-amd64-musl.tar.gz": "dist/grafana-latest.linux-x64-musl.tar.gz",
181+
".linux-armv7.tar.gz": "dist/grafana-latest.linux-armv7.tar.gz",
182+
".linux-armv7-musl.tar.gz": "dist/grafana-latest.linux-armv7-musl.tar.gz",
183+
".linux-armv6.tar.gz": "dist/grafana-latest.linux-armv6.tar.gz",
184+
".linux-arm64.tar.gz": "dist/grafana-latest.linux-arm64.tar.gz",
185+
".linux-arm64-musl.tar.gz": "dist/grafana-latest.linux-arm64-musl.tar.gz",
185186
}
186187

187188
for _, file := range files {
@@ -455,6 +456,9 @@ func gruntBuildArg(task string) []string {
455456
if pkgArch != "" {
456457
args = append(args, fmt.Sprintf("--arch=%v", pkgArch))
457458
}
459+
if libc != "" {
460+
args = append(args, fmt.Sprintf("--libc=%s", libc))
461+
}
458462
if phjsToRelease != "" {
459463
args = append(args, fmt.Sprintf("--phjsToRelease=%v", phjsToRelease))
460464
}
@@ -481,9 +485,13 @@ func test(pkg string) {
481485
}
482486

483487
func build(binaryName, pkg string, tags []string) {
484-
binary := fmt.Sprintf("./bin/%s-%s/%s", goos, goarch, binaryName)
488+
libcPart := ""
489+
if libc != "" {
490+
libcPart = fmt.Sprintf("-%s", libc)
491+
}
492+
binary := fmt.Sprintf("./bin/%s-%s%s/%s", goos, goarch, libcPart, binaryName)
485493
if isDev {
486-
//don't include os and arch in output path in dev environment
494+
//don't include os/arch/libc in output path in dev environment
487495
binary = fmt.Sprintf("./bin/%s", binaryName)
488496
}
489497

@@ -511,7 +519,11 @@ func build(binaryName, pkg string, tags []string) {
511519
if !isDev {
512520
setBuildEnv()
513521
runPrint("go", "version")
514-
fmt.Printf("Targeting %s/%s\n", goos, goarch)
522+
libcPart := ""
523+
if libc != "" {
524+
libcPart = fmt.Sprintf("/%s", libc)
525+
}
526+
fmt.Printf("Targeting %s/%s%s\n", goos, goarch, libcPart)
515527
}
516528

517529
runPrint("go", args...)

packaging/docker/Dockerfile

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
ARG BASE_IMAGE=alpine:3.10
22
FROM ${BASE_IMAGE}
33

4-
ARG GRAFANA_TGZ="grafana-latest.linux-x64.tar.gz"
4+
ARG GRAFANA_TGZ="grafana-latest.linux-x64-musl.tar.gz"
55

66
COPY ${GRAFANA_TGZ} /tmp/grafana.tar.gz
77

packaging/docker/build.sh

+3-3
Original file line numberDiff line numberDiff line change
@@ -59,10 +59,10 @@ docker_tag_all () {
5959
fi
6060
}
6161

62-
docker_build "alpine:3.10" "grafana-latest.linux-x64.tar.gz" "${_docker_repo}:${_grafana_version}"
62+
docker_build "alpine:3.10" "grafana-latest.linux-x64-musl.tar.gz" "${_docker_repo}:${_grafana_version}"
6363
if [ $BUILD_FAST = "0" ]; then
64-
docker_build "arm32v7/alpine:3.10" "grafana-latest.linux-armv7.tar.gz" "${_docker_repo}-arm32v7-linux:${_grafana_version}"
65-
docker_build "arm64v8/alpine:3.10" "grafana-latest.linux-arm64.tar.gz" "${_docker_repo}-arm64v8-linux:${_grafana_version}"
64+
docker_build "arm32v7/alpine:3.10" "grafana-latest.linux-armv7-musl.tar.gz" "${_docker_repo}-arm32v7-linux:${_grafana_version}"
65+
docker_build "arm64v8/alpine:3.10" "grafana-latest.linux-arm64-musl.tar.gz" "${_docker_repo}-arm64v8-linux:${_grafana_version}"
6666
fi
6767
# Tag as 'latest' for official release; otherwise tag as grafana/grafana:master
6868
if echo "$_grafana_tag" | grep -q "^v"; then

scripts/build/build-all.sh

+11-1
Original file line numberDiff line numberDiff line change
@@ -13,10 +13,13 @@ EXTRA_OPTS="$@"
1313

1414
CCARMV6=/opt/rpi-tools/arm-bcm2708/arm-linux-gnueabihf/bin/arm-linux-gnueabihf-gcc
1515
CCARMV7=arm-linux-gnueabihf-gcc
16+
CCARMV7_MUSL=/tmp/arm-linux-musleabihf-cross/bin/arm-linux-musleabihf-gcc
1617
CCARM64=aarch64-linux-gnu-gcc
18+
CCARM64_MUSL=/tmp/aarch64-linux-musl-cross/bin/aarch64-linux-musl-gcc
1719
CCOSX64=/tmp/osxcross/target/bin/o64-clang
1820
CCWIN64=x86_64-w64-mingw32-gcc
1921
CCX64=/tmp/x86_64-centos6-linux-gnu/bin/x86_64-centos6-linux-gnu-gcc
22+
CCX64_MUSL=/tmp/x86_64-linux-musl-cross/bin/x86_64-linux-musl-gcc
2023

2124
cd /go/src/github.com/grafana/grafana
2225
echo "current dir: $(pwd)"
@@ -37,13 +40,17 @@ if echo "$EXTRA_OPTS" | grep -vq enterprise ; then
3740
go run build.go -goarch armv6 -cc ${CCARMV6} ${OPT} build
3841
go run build.go -goarch armv7 -cc ${CCARMV7} ${OPT} build
3942
go run build.go -goarch arm64 -cc ${CCARM64} ${OPT} build
43+
go run build.go -goarch armv7 -libc musl -cc ${CCARMV7_MUSL} ${OPT} build
44+
go run build.go -goarch arm64 -libc musl -cc ${CCARM64_MUSL} ${OPT} build
4045
go run build.go -goos darwin -cc ${CCOSX64} ${OPT} build
4146
fi
4247

4348
go run build.go -goos windows -cc ${CCWIN64} ${OPT} build
4449

4550
# Do not remove CC from the linux build, its there for compatibility with Centos6
46-
CC=${CCX64} go run build.go ${OPT} build
51+
go run build.go -cc ${CCX64} ${OPT} build
52+
53+
go run build.go -cc ${CCX64_MUSL} -libc musl ${OPT} build
4754

4855
yarn install --pure-lockfile --no-progress
4956

@@ -68,6 +75,7 @@ source /etc/profile.d/rvm.sh
6875

6976
echo "Packaging"
7077
go run build.go -goos linux -pkg-arch amd64 ${OPT} package-only
78+
go run build.go -goos linux -pkg-arch amd64 -libc musl ${OPT} -skipRpm -skipDeb package-only
7179
#removing amd64 phantomjs bin for armv7/arm64 packages
7280
rm tools/phantomjs/phantomjs
7381

@@ -76,6 +84,8 @@ if echo "$EXTRA_OPTS" | grep -vq enterprise ; then
7684
go run build.go -goos linux -pkg-arch armv6 ${OPT} -skipRpm package-only
7785
go run build.go -goos linux -pkg-arch armv7 ${OPT} package-only
7886
go run build.go -goos linux -pkg-arch arm64 ${OPT} package-only
87+
go run build.go -goos linux -pkg-arch armv7 -libc musl ${OPT} -skipRpm -skipDeb package-only
88+
go run build.go -goos linux -pkg-arch arm64 -libc musl ${OPT} -skipRpm -skipDeb package-only
7989

8090
if [ -d '/tmp/phantomjs/darwin' ]; then
8191
cp /tmp/phantomjs/darwin/phantomjs tools/phantomjs/phantomjs

scripts/build/build.sh

+22-10
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,7 @@
11
#!/bin/bash
2+
3+
# shellcheck disable=SC2086
4+
25
#
36
# This script is executed from within the container.
47
#
@@ -7,8 +10,11 @@ set -e
710
##########
811
CCARMV6=/opt/rpi-tools/arm-bcm2708/arm-linux-gnueabihf/bin/arm-linux-gnueabihf-gcc
912
CCARMV7=arm-linux-gnueabihf-gcc
13+
CCARMV7_MUSL=/tmp/arm-linux-musleabihf-cross/bin/arm-linux-musleabihf-gcc
1014
CCARM64=aarch64-linux-gnu-gcc
15+
CCARM64_MUSL=/tmp/aarch64-linux-musl-cross/bin/aarch64-linux-musl-gcc
1116
CCX64=/tmp/x86_64-centos6-linux-gnu/bin/x86_64-centos6-linux-gnu-gcc
17+
CCX64_MUSL=/tmp/x86_64-linux-musl-cross/bin/x86_64-linux-musl-gcc
1218

1319
BUILD_FAST=0
1420
BUILD_BACKEND=1
@@ -69,17 +75,20 @@ function build_backend_linux_amd64() {
6975
if [ ! -d "dist" ]; then
7076
mkdir dist
7177
fi
72-
CC=${CCX64} go run build.go "${OPT}" build
78+
CC=${CCX64} go run build.go ${OPT} build
79+
CC=${CCX64_MUSL} go run build.go -libc musl ${OPT} build
7380
}
7481

7582
function build_backend() {
7683
if [ ! -d "dist" ]; then
7784
mkdir dist
7885
fi
7986

80-
go run build.go -goarch armv6 -cc ${CCARMV6} "${OPT}" build
81-
go run build.go -goarch armv7 -cc ${CCARMV7} "${OPT}" build
82-
go run build.go -goarch arm64 -cc ${CCARM64} "${OPT}" build
87+
go run build.go -goarch armv6 -cc ${CCARMV6} ${OPT} build
88+
go run build.go -goarch armv7 -cc ${CCARMV7} ${OPT} build
89+
go run build.go -goarch arm64 -cc ${CCARM64} ${OPT} build
90+
go run build.go -goarch armv7 -libc musl -cc ${CCARMV7_MUSL} ${OPT} build
91+
go run build.go -goarch arm64 -libc musl -cc ${CCARM64_MUSL} ${OPT} build
8392
build_backend_linux_amd64
8493
}
8594

@@ -89,22 +98,25 @@ function build_frontend() {
8998
fi
9099
yarn install --pure-lockfile --no-progress
91100
echo "Building frontend"
92-
go run build.go "${OPT}" build-frontend
101+
go run build.go ${OPT} build-frontend
93102
echo "FRONTEND: finished"
94103
}
95104

96105
function package_linux_amd64() {
97106
echo "Packaging Linux AMD64"
98-
go run build.go -goos linux -pkg-arch amd64 "${OPT}" package-only
107+
go run build.go -goos linux -pkg-arch amd64 ${OPT} package-only
108+
go run build.go -goos linux -pkg-arch amd64 ${OPT} -libc musl -skipRpm -skipDeb package-only
99109
go run build.go latest
100110
echo "PACKAGE LINUX AMD64: finished"
101111
}
102112

103113
function package_all() {
104114
echo "Packaging ALL"
105-
go run build.go -goos linux -pkg-arch armv6 "${OPT}" -skipRpm package-only
106-
go run build.go -goos linux -pkg-arch armv7 "${OPT}" package-only
107-
go run build.go -goos linux -pkg-arch arm64 "${OPT}" package-only
115+
go run build.go -goos linux -pkg-arch armv6 ${OPT} -skipRpm package-only
116+
go run build.go -goos linux -pkg-arch armv7 ${OPT} package-only
117+
go run build.go -goos linux -pkg-arch arm64 ${OPT} package-only
118+
go run build.go -goos linux -pkg-arch armv7 -libc musl -skipRpm -skipDeb ${OPT} package-only
119+
go run build.go -goos linux -pkg-arch arm64 -libc musl -skipRpm -skipDeb ${OPT} package-only
108120
package_linux_amd64
109121
echo "PACKAGE ALL: finished"
110122
}
@@ -115,7 +127,7 @@ function package_setup() {
115127
rm -rf dist
116128
fi
117129
mkdir dist
118-
go run build.go -gen-version "${OPT}" > dist/grafana.version
130+
go run build.go -gen-version ${OPT} > dist/grafana.version
119131
# Load ruby, needed for packing with fpm
120132
# shellcheck disable=SC1091
121133
source /etc/profile.d/rvm.sh

scripts/build/ci-build/Dockerfile

+19-1
Original file line numberDiff line numberDiff line change
@@ -74,7 +74,10 @@ FROM ubuntu:14.04
7474
ENV GOVERSION=1.13.1 \
7575
PATH=/usr/local/go/bin:$PATH \
7676
GOPATH=/go \
77-
NODEVERSION=10.14.2
77+
NODEVERSION=10.14.2 \
78+
CHKSUM_ARMV7_MUSL=3043bd16a0287b02f3f1045ec5f7fa42bd47bd8329183fc52b179dbb80a56d5f \
79+
CHKSUM_ARMV8_MUSL=1a935561ebe3155bfd02b1b25cd7533c81906dd47b1de1fa43b74209130b941c \
80+
CHKSUM_AMD64_MUSL=0fada482750176671d24c31b944fae7c3bf0a826011b3decc12d47b21c3cb5c2
7881

7982
COPY --from=toolchain /tmp/x86_64-centos6-linux-gnu.tar.xz /tmp/
8083
COPY --from=toolchain /tmp/osxcross.tar.xz /tmp/
@@ -108,6 +111,21 @@ RUN apt-get update && \
108111
| tar -xz -C /usr/local && \
109112
git clone https://github.com/raspberrypi/tools.git /opt/rpi-tools --depth=1
110113

114+
# Install musl cross compilers
115+
RUN cd /tmp && \
116+
curl -O https://musl.cc/arm-linux-musleabihf-cross.tgz && \
117+
[ "$(sha256sum arm-linux-musleabihf-cross.tgz|cut -f1 -d ' ')" = "$CHKSUM_ARMV7_MUSL" ] && \
118+
tar xf arm-linux-musleabihf-cross.tgz && \
119+
rm arm-linux-musleabihf-cross.tgz && \
120+
curl -O https://musl.cc/aarch64-linux-musl-cross.tgz && \
121+
[ "$(sha256sum aarch64-linux-musl-cross.tgz|cut -f1 -d ' ')" = "$CHKSUM_ARMV8_MUSL" ] && \
122+
tar xf aarch64-linux-musl-cross.tgz && \
123+
rm aarch64-linux-musl-cross.tgz && \
124+
curl -O https://musl.cc/x86_64-linux-musl-cross.tgz && \
125+
[ "$(sha256sum x86_64-linux-musl-cross.tgz|cut -f1 -d ' ')" = "$CHKSUM_AMD64_MUSL" ] && \
126+
tar xf x86_64-linux-musl-cross.tgz && \
127+
rm x86_64-linux-musl-cross.tgz
128+
111129
RUN apt-get install -y \
112130
gcc libc-dev make && \
113131
gpg2 --keyserver hkp://keys.gnupg.net --recv-keys 409B6B1796C275462A1703113804BB82D39DC0E3 7D2BAF1CF37B13E2069D6956105BD0E739499BDB && \

scripts/build/ci-build/build-deploy.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
#!/bin/bash
22
set -eo pipefail
33

4-
_version="1.2.10"
4+
_version="1.2.11"
55
_tag="grafana/build-container:${_version}"
66

77
_dpath=$(dirname "${BASH_SOURCE[0]}")

0 commit comments

Comments
 (0)