mysql

  1. FROM debian:stretch-slim
  2. # add our user and group first to make sure their IDs get assigned consistently, regardless of whatever dependencies get added
  3. RUN groupadd -r mysql && useradd -r -g mysql mysql
  4. RUN apt-get update && apt-get install -y --no-install-recommends gnupg dirmngr && rm -rf /var/lib/apt/lists/*
  5. # add gosu for easy step-down from root
  6. ENV GOSU_VERSION 1.7
  7. RUN set -x \
  8. && apt-get update && apt-get install -y --no-install-recommends ca-certificates wget && rm -rf /var/lib/apt/lists/* \
  9. && wget -O /usr/local/bin/gosu "https://github.com/tianon/gosu/releases/download/$GOSU_VERSION/gosu-$(dpkg --print-architecture)" \
  10. && wget -O /usr/local/bin/gosu.asc "https://github.com/tianon/gosu/releases/download/$GOSU_VERSION/gosu-$(dpkg --print-architecture).asc" \
  11. && export GNUPGHOME="$(mktemp -d)" \
  12. && gpg --batch --keyserver ha.pool.sks-keyservers.net --recv-keys B42F6819007F00F88E364FD4036A9C25BF357DD4 \
  13. && gpg --batch --verify /usr/local/bin/gosu.asc /usr/local/bin/gosu \
  14. && gpgconf --kill all \
  15. && rm -rf "$GNUPGHOME" /usr/local/bin/gosu.asc \
  16. && chmod +x /usr/local/bin/gosu \
  17. && gosu nobody true \
  18. && apt-get purge -y --auto-remove ca-certificates wget
  19. RUN mkdir /docker-entrypoint-initdb.d
  20. RUN apt-get update && apt-get install -y --no-install-recommends \
  21. # for MYSQL_RANDOM_ROOT_PASSWORD
  22. pwgen \
  23. # for mysql_ssl_rsa_setup
  24. openssl \
  25. # FATAL ERROR: please install the following Perl modules before executing /usr/local/mysql/scripts/mysql_install_db:
  26. # File::Basename
  27. # File::Copy
  28. # Sys::Hostname
  29. # Data::Dumper
  30. perl \
  31. && rm -rf /var/lib/apt/lists/*
  32. RUN set -ex; \
  33. # gpg: key 5072E1F5: public key "MySQL Release Engineering <mysql-build@oss.oracle.com>" imported
  34. key='A4A9406876FCBD3C456770C88C718D3B5072E1F5'; \
  35. export GNUPGHOME="$(mktemp -d)"; \
  36. gpg --batch --keyserver ha.pool.sks-keyservers.net --recv-keys "$key"; \
  37. gpg --batch --export "$key" > /etc/apt/trusted.gpg.d/mysql.gpg; \
  38. gpgconf --kill all; \
  39. rm -rf "$GNUPGHOME"; \
  40. apt-key list > /dev/null
  41. ENV MYSQL_MAJOR 5.7
  42. ENV MYSQL_VERSION 5.7.25-1debian9
  43. RUN echo "deb http://repo.mysql.com/apt/debian/ stretch mysql-${MYSQL_MAJOR}" > /etc/apt/sources.list.d/mysql.list
  44. # the "/var/lib/mysql" stuff here is because the mysql-server postinst doesn't have an explicit way to disable the mysql_install_db codepath besides having a database already "configured" (ie, stuff in /var/lib/mysql/mysql)
  45. # also, we set debconf keys to make APT a little quieter
  46. RUN { \
  47. echo mysql-community-server mysql-community-server/data-dir select ''; \
  48. echo mysql-community-server mysql-community-server/root-pass password ''; \
  49. echo mysql-community-server mysql-community-server/re-root-pass password ''; \
  50. echo mysql-community-server mysql-community-server/remove-test-db select false; \
  51. } | debconf-set-selections \
  52. && apt-get update && apt-get install -y mysql-server="${MYSQL_VERSION}" && rm -rf /var/lib/apt/lists/* \
  53. && rm -rf /var/lib/mysql && mkdir -p /var/lib/mysql /var/run/mysqld \
  54. && chown -R mysql:mysql /var/lib/mysql /var/run/mysqld \
  55. # ensure that /var/run/mysqld (used for socket and lock files) is writable regardless of the UID our mysqld instance ends up having at runtime
  56. && chmod 777 /var/run/mysqld \
  57. # comment out a few problematic configuration values
  58. && find /etc/mysql/ -name '*.cnf' -print0 \
  59. | xargs -0 grep -lZE '^(bind-address|log)' \
  60. | xargs -rt -0 sed -Ei 's/^(bind-address|log)/#&/' \
  61. # don't reverse lookup hostnames, they are usually another container
  62. && echo '[mysqld]\nskip-host-cache\nskip-name-resolve' > /etc/mysql/conf.d/docker.cnf
  63. VOLUME /var/lib/mysql
  64. COPY docker-entrypoint.sh /usr/local/bin/
  65. RUN ln -s usr/local/bin/docker-entrypoint.sh /entrypoint.sh # backwards compat
  66. ENTRYPOINT ["docker-entrypoint.sh"]
  67. EXPOSE 3306 33060
  68. CMD ["mysqld"]
  1. capsule image runc mysql "/usr/local/bin/docker-entrypoint.sh mysqld --user=root" -id=mysql -v /root/mysql/logs:/logs -v /root/mysql/data:/var/lib/mysql -p 3306:3306 -d
  2. capsule exec mysql bash
  3. > show databases;
  4. > create database demo;
  5. > use mysql;
  6. > update user set host='%' where user='root';
  7. > flush privileges;
  8. > exit

error: exec: “/usr/local/bin/docker-entrypoint.sh”: stat /usr/local/bin/docker-entrypoint.sh: permission denied

这里要把—user这种选项的处理逻辑删掉, 因为目前不支持chown。

  1. #!/bin/bash
  2. set -eo pipefail
  3. shopt -s nullglob
  4. # if command starts with an option, prepend mysqld
  5. if [ "${1:0:1}" = '-' ]; then
  6. set -- mysqld "$@"
  7. fi
  8. # skip setup if they want an option that stops mysqld
  9. wantHelp=
  10. for arg; do
  11. case "$arg" in
  12. -'?'|--help|--print-defaults|-V|--version)
  13. wantHelp=1
  14. break
  15. ;;
  16. esac
  17. done
  18. # usage: file_env VAR [DEFAULT]
  19. # ie: file_env 'XYZ_DB_PASSWORD' 'example'
  20. # (will allow for "$XYZ_DB_PASSWORD_FILE" to fill in the value of
  21. # "$XYZ_DB_PASSWORD" from a file, especially for Docker's secrets feature)
  22. file_env() {
  23. local var="$1"
  24. local fileVar="${var}_FILE"
  25. local def="${2:-}"
  26. if [ "${!var:-}" ] && [ "${!fileVar:-}" ]; then
  27. echo >&2 "error: both $var and $fileVar are set (but are exclusive)"
  28. exit 1
  29. fi
  30. local val="$def"
  31. if [ "${!var:-}" ]; then
  32. val="${!var}"
  33. elif [ "${!fileVar:-}" ]; then
  34. val="$(< "${!fileVar}")"
  35. fi
  36. export "$var"="$val"
  37. unset "$fileVar"
  38. }
  39. # usage: process_init_file FILENAME MYSQLCOMMAND...
  40. # ie: process_init_file foo.sh mysql -uroot
  41. # (process a single initializer file, based on its extension. we define this
  42. # function here, so that initializer scripts (*.sh) can use the same logic,
  43. # potentially recursively, or override the logic used in subsequent calls)
  44. process_init_file() {
  45. local f="$1"; shift
  46. local mysql=( "$@" )
  47. case "$f" in
  48. *.sh) echo "$0: running $f"; . "$f" ;;
  49. *.sql) echo "$0: running $f"; "${mysql[@]}" < "$f"; echo ;;
  50. *.sql.gz) echo "$0: running $f"; gunzip -c "$f" | "${mysql[@]}"; echo ;;
  51. *) echo "$0: ignoring $f" ;;
  52. esac
  53. echo
  54. }
  55. _check_config() {
  56. toRun=( "$@" --verbose --help )
  57. if ! errors="$("${toRun[@]}" 2>&1 >/dev/null)"; then
  58. cat >&2 <<-EOM
  59. ERROR: mysqld failed while attempting to check config
  60. command was: "${toRun[*]}"
  61. $errors
  62. EOM
  63. exit 1
  64. fi
  65. }
  66. # Fetch value from server config
  67. # We use mysqld --verbose --help instead of my_print_defaults because the
  68. # latter only show values present in config files, and not server defaults
  69. _get_config() {
  70. local conf="$1"; shift
  71. "$@" --verbose --help --log-bin-index="$(mktemp -u)" 2>/dev/null \
  72. | awk '$1 == "'"$conf"'" && /^[^ \t]/ { sub(/^[^ \t]+[ \t]+/, ""); print; exit }'
  73. # match "datadir /some/path with/spaces in/it here" but not "--xyz=abc\n datadir (xyz)"
  74. }
  75. if [ "$1" = 'mysqld' -a -z "$wantHelp" ]; then
  76. # still need to check config, container may have started with --user
  77. _check_config "$@"
  78. # Get config
  79. DATADIR="$(_get_config 'datadir' "$@")"
  80. if [ ! -d "$DATADIR/mysql" ]; then
  81. file_env 'MYSQL_ROOT_PASSWORD'
  82. if [ -z "$MYSQL_ROOT_PASSWORD" -a -z "$MYSQL_ALLOW_EMPTY_PASSWORD" -a -z "$MYSQL_RANDOM_ROOT_PASSWORD" ]; then
  83. echo >&2 'error: database is uninitialized and password option is not specified '
  84. echo >&2 ' You need to specify one of MYSQL_ROOT_PASSWORD, MYSQL_ALLOW_EMPTY_PASSWORD and MYSQL_RANDOM_ROOT_PASSWORD'
  85. exit 1
  86. fi
  87. mkdir -p "$DATADIR"
  88. echo 'Initializing database'
  89. "$@" --initialize-insecure
  90. echo 'Database initialized'
  91. if command -v mysql_ssl_rsa_setup > /dev/null && [ ! -e "$DATADIR/server-key.pem" ]; then
  92. # https://github.com/mysql/mysql-server/blob/23032807537d8dd8ee4ec1c4d40f0633cd4e12f9/packaging/deb-in/extra/mysql-systemd-start#L81-L84
  93. echo 'Initializing certificates'
  94. mysql_ssl_rsa_setup --datadir="$DATADIR"
  95. echo 'Certificates initialized'
  96. fi
  97. SOCKET="$(_get_config 'socket' "$@")"
  98. "$@" --skip-networking --socket="${SOCKET}" &
  99. pid="$!"
  100. mysql=( mysql --protocol=socket -uroot -hlocalhost --socket="${SOCKET}" )
  101. for i in {30..0}; do
  102. if echo 'SELECT 1' | "${mysql[@]}" &> /dev/null; then
  103. break
  104. fi
  105. echo 'MySQL init process in progress...'
  106. sleep 1
  107. done
  108. if [ "$i" = 0 ]; then
  109. echo >&2 'MySQL init process failed.'
  110. exit 1
  111. fi
  112. if [ -z "$MYSQL_INITDB_SKIP_TZINFO" ]; then
  113. # sed is for https://bugs.mysql.com/bug.php?id=20545
  114. mysql_tzinfo_to_sql /usr/share/zoneinfo | sed 's/Local time zone must be set--see zic manual page/FCTY/' | "${mysql[@]}" mysql
  115. fi
  116. if [ ! -z "$MYSQL_RANDOM_ROOT_PASSWORD" ]; then
  117. export MYSQL_ROOT_PASSWORD="$(pwgen -1 32)"
  118. echo "GENERATED ROOT PASSWORD: $MYSQL_ROOT_PASSWORD"
  119. fi
  120. rootCreate=
  121. # default root to listen for connections from anywhere
  122. file_env 'MYSQL_ROOT_HOST' '%'
  123. if [ ! -z "$MYSQL_ROOT_HOST" -a "$MYSQL_ROOT_HOST" != 'localhost' ]; then
  124. # no, we don't care if read finds a terminating character in this heredoc
  125. # https://unix.stackexchange.com/questions/265149/why-is-set-o-errexit-breaking-this-read-heredoc-expression/265151#265151
  126. read -r -d '' rootCreate <<-EOSQL || true
  127. CREATE USER 'root'@'${MYSQL_ROOT_HOST}' IDENTIFIED BY '${MYSQL_ROOT_PASSWORD}' ;
  128. GRANT ALL ON *.* TO 'root'@'${MYSQL_ROOT_HOST}' WITH GRANT OPTION ;
  129. EOSQL
  130. fi
  131. "${mysql[@]}" <<-EOSQL
  132. -- What's done in this file shouldn't be replicated
  133. -- or products like mysql-fabric won't work
  134. SET @@SESSION.SQL_LOG_BIN=0;
  135. ALTER USER 'root'@'localhost' IDENTIFIED BY '${MYSQL_ROOT_PASSWORD}' ;
  136. GRANT ALL ON *.* TO 'root'@'localhost' WITH GRANT OPTION ;
  137. ${rootCreate}
  138. DROP DATABASE IF EXISTS test ;
  139. FLUSH PRIVILEGES ;
  140. EOSQL
  141. if [ ! -z "$MYSQL_ROOT_PASSWORD" ]; then
  142. mysql+=( -p"${MYSQL_ROOT_PASSWORD}" )
  143. fi
  144. file_env 'MYSQL_DATABASE'
  145. if [ "$MYSQL_DATABASE" ]; then
  146. echo "CREATE DATABASE IF NOT EXISTS \`$MYSQL_DATABASE\` ;" | "${mysql[@]}"
  147. mysql+=( "$MYSQL_DATABASE" )
  148. fi
  149. file_env 'MYSQL_USER'
  150. file_env 'MYSQL_PASSWORD'
  151. if [ "$MYSQL_USER" -a "$MYSQL_PASSWORD" ]; then
  152. echo "CREATE USER '$MYSQL_USER'@'%' IDENTIFIED BY '$MYSQL_PASSWORD' ;" | "${mysql[@]}"
  153. if [ "$MYSQL_DATABASE" ]; then
  154. echo "GRANT ALL ON \`$MYSQL_DATABASE\`.* TO '$MYSQL_USER'@'%' ;" | "${mysql[@]}"
  155. fi
  156. echo 'FLUSH PRIVILEGES ;' | "${mysql[@]}"
  157. fi
  158. echo
  159. ls /docker-entrypoint-initdb.d/ > /dev/null
  160. for f in /docker-entrypoint-initdb.d/*; do
  161. process_init_file "$f" "${mysql[@]}"
  162. done
  163. if [ ! -z "$MYSQL_ONETIME_PASSWORD" ]; then
  164. "${mysql[@]}" <<-EOSQL
  165. ALTER USER 'root'@'%' PASSWORD EXPIRE;
  166. EOSQL
  167. fi
  168. if ! kill -s TERM "$pid" || ! wait "$pid"; then
  169. echo >&2 'MySQL init process failed.'
  170. exit 1
  171. fi
  172. echo
  173. echo 'MySQL init process done. Ready for start up.'
  174. echo
  175. fi
  176. fi
  177. exec "$@"

redis

  1. FROM debian:stretch-slim
  2. # add our user and group first to make sure their IDs get assigned consistently, regardless of whatever dependencies get added
  3. RUN groupadd -r redis && useradd -r -g redis redis
  4. # grab gosu for easy step-down from root
  5. # https://github.com/tianon/gosu/releases
  6. ENV GOSU_VERSION 1.10
  7. RUN set -ex; \
  8. \
  9. fetchDeps=" \
  10. ca-certificates \
  11. dirmngr \
  12. gnupg \
  13. wget \
  14. "; \
  15. apt-get update; \
  16. apt-get install -y --no-install-recommends $fetchDeps; \
  17. rm -rf /var/lib/apt/lists/*; \
  18. \
  19. dpkgArch="$(dpkg --print-architecture | awk -F- '{ print $NF }')"; \
  20. wget -O /usr/local/bin/gosu "https://github.com/tianon/gosu/releases/download/$GOSU_VERSION/gosu-$dpkgArch"; \
  21. wget -O /usr/local/bin/gosu.asc "https://github.com/tianon/gosu/releases/download/$GOSU_VERSION/gosu-$dpkgArch.asc"; \
  22. export GNUPGHOME="$(mktemp -d)"; \
  23. gpg --batch --keyserver ha.pool.sks-keyservers.net --recv-keys B42F6819007F00F88E364FD4036A9C25BF357DD4; \
  24. gpg --batch --verify /usr/local/bin/gosu.asc /usr/local/bin/gosu; \
  25. gpgconf --kill all; \
  26. rm -r "$GNUPGHOME" /usr/local/bin/gosu.asc; \
  27. chmod +x /usr/local/bin/gosu; \
  28. gosu nobody true; \
  29. \
  30. apt-get purge -y --auto-remove $fetchDeps
  31. ENV REDIS_VERSION 5.0.4
  32. ENV REDIS_DOWNLOAD_URL http://download.redis.io/releases/redis-5.0.4.tar.gz
  33. ENV REDIS_DOWNLOAD_SHA 3ce9ceff5a23f60913e1573f6dfcd4aa53b42d4a2789e28fa53ec2bd28c987dd
  34. # for redis-sentinel see: http://redis.io/topics/sentinel
  35. RUN set -ex; \
  36. \
  37. buildDeps=' \
  38. ca-certificates \
  39. wget \
  40. \
  41. gcc \
  42. libc6-dev \
  43. make \
  44. '; \
  45. apt-get update; \
  46. apt-get install -y $buildDeps --no-install-recommends; \
  47. rm -rf /var/lib/apt/lists/*; \
  48. \
  49. wget -O redis.tar.gz "$REDIS_DOWNLOAD_URL"; \
  50. echo "$REDIS_DOWNLOAD_SHA *redis.tar.gz" | sha256sum -c -; \
  51. mkdir -p /usr/src/redis; \
  52. tar -xzf redis.tar.gz -C /usr/src/redis --strip-components=1; \
  53. rm redis.tar.gz; \
  54. \
  55. # disable Redis protected mode [1] as it is unnecessary in context of Docker
  56. # (ports are not automatically exposed when running inside Docker, but rather explicitly by specifying -p / -P)
  57. # [1]: https://github.com/antirez/redis/commit/edd4d555df57dc84265fdfb4ef59a4678832f6da
  58. grep -q '^#define CONFIG_DEFAULT_PROTECTED_MODE 1$' /usr/src/redis/src/server.h; \
  59. sed -ri 's!^(#define CONFIG_DEFAULT_PROTECTED_MODE) 1$!\1 0!' /usr/src/redis/src/server.h; \
  60. grep -q '^#define CONFIG_DEFAULT_PROTECTED_MODE 0$' /usr/src/redis/src/server.h; \
  61. # for future reference, we modify this directly in the source instead of just supplying a default configuration flag because apparently "if you specify any argument to redis-server, [it assumes] you are going to specify everything"
  62. # see also https://github.com/docker-library/redis/issues/4#issuecomment-50780840
  63. # (more exactly, this makes sure the default behavior of "save on SIGTERM" stays functional by default)
  64. \
  65. make -C /usr/src/redis -j "$(nproc)"; \
  66. make -C /usr/src/redis install; \
  67. \
  68. rm -r /usr/src/redis; \
  69. \
  70. apt-get purge -y --auto-remove $buildDeps
  71. RUN mkdir /data && chown redis:redis /data
  72. VOLUME /data
  73. WORKDIR /data
  74. COPY docker-entrypoint.sh /usr/local/bin/
  75. ENTRYPOINT ["docker-entrypoint.sh"]
  76. EXPOSE 6379
  77. CMD ["redis-server"]
  1. capsule image runc redis /usr/local/bin/docker-entrypoint.sh redis-server --id=redis -p 6379:6379 -d

将/usr/local/bin/docker-entrypoint.sh中chown的部分删掉,因为capsule不支持多用户,目前只能用root。
修改后的是这样的:

  1. #!/bin/sh
  2. set -e
  3. # first arg is `-f` or `--some-option`
  4. # or first arg is `something.conf`
  5. if [ "${1#-}" != "$1" ] || [ "${1%.conf}" != "$1" ]; then
  6. set -- redis-server "$@"
  7. fi
  8. exec "$@"

capsule-demo-app

  1. FROM java:8
  2. VOLUME /tmp
  3. ADD capsule-demo-app.jar app.jar
  4. ENTRYPOINT [ "sh", "-c", "java -jar /app.jar"]
  1. capsule image runc capsule-demo-app "java -jar /app.jar" -id capsule-demo-container -e "SPRING_PROFILES_ACTIVE=prod" -p 8080:8080 -d -link mysql:mysql-container -link redis:redis-container