Skip to content

Commit 2bad673

Browse files
authored
Merge pull request nodejs#2149 from jnoordsij/add-node-23
feat: Node.js 23.0.0
2 parents 7b8f973 + 1ef28c2 commit 2bad673

13 files changed

+709
-0
lines changed

23/alpine3.19/Dockerfile

+112
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,112 @@
1+
FROM alpine:3.19
2+
3+
ENV NODE_VERSION 23.0.0
4+
5+
RUN addgroup -g 1000 node \
6+
&& adduser -u 1000 -G node -s /bin/sh -D node \
7+
&& apk add --no-cache \
8+
libstdc++ \
9+
&& apk add --no-cache --virtual .build-deps \
10+
curl \
11+
&& ARCH= OPENSSL_ARCH='linux*' && alpineArch="$(apk --print-arch)" \
12+
&& case "${alpineArch##*-}" in \
13+
x86_64) ARCH='x64' CHECKSUM="738b28f80f9aedaae3a7e91db8294cb96e2f87078fe7ff8826d612405ee2d8cb" OPENSSL_ARCH=linux-x86_64;; \
14+
x86) OPENSSL_ARCH=linux-elf;; \
15+
aarch64) OPENSSL_ARCH=linux-aarch64;; \
16+
arm*) OPENSSL_ARCH=linux-armv4;; \
17+
ppc64le) OPENSSL_ARCH=linux-ppc64le;; \
18+
s390x) OPENSSL_ARCH=linux-s390x;; \
19+
*) ;; \
20+
esac \
21+
&& if [ -n "${CHECKSUM}" ]; then \
22+
set -eu; \
23+
curl -fsSLO --compressed "https://unofficial-builds.nodejs.org/download/release/v$NODE_VERSION/node-v$NODE_VERSION-linux-$ARCH-musl.tar.xz"; \
24+
echo "$CHECKSUM node-v$NODE_VERSION-linux-$ARCH-musl.tar.xz" | sha256sum -c - \
25+
&& tar -xJf "node-v$NODE_VERSION-linux-$ARCH-musl.tar.xz" -C /usr/local --strip-components=1 --no-same-owner \
26+
&& ln -s /usr/local/bin/node /usr/local/bin/nodejs; \
27+
else \
28+
echo "Building from source" \
29+
# backup build
30+
&& apk add --no-cache --virtual .build-deps-full \
31+
binutils-gold \
32+
g++ \
33+
gcc \
34+
gnupg \
35+
libgcc \
36+
linux-headers \
37+
make \
38+
python3 \
39+
py-setuptools \
40+
# use pre-existing gpg directory, see https://github.com/nodejs/docker-node/pull/1895#issuecomment-1550389150
41+
&& export GNUPGHOME="$(mktemp -d)" \
42+
# gpg keys listed at https://github.com/nodejs/node#release-keys
43+
&& for key in \
44+
C0D6248439F1D5604AAFFB4021D900FFDB233756 \
45+
4ED778F539E3634C779C87C6D7062848A1AB005C \
46+
141F07595B7B3FFE74309A937405533BE57C7D57 \
47+
74F12602B6F1C4E913FAA37AD3A89613643B6201 \
48+
DD792F5973C6DE52C432CBDAC77ABFA00DDBF2B7 \
49+
CC68F5A3106FF448322E48ED27F5E38D5B0A215F \
50+
8FCCA13FEF1D0C2E91008E09770F7A9A5AE15600 \
51+
890C08DB8579162FEE0DF9DB8BEAB4DFCF555EF4 \
52+
C82FA3AE1CBEDC6BE46B9360C43CEC45C17AB93C \
53+
108F52B48DB57BB0CC439B2997B01419BD92F80A \
54+
A363A499291CBBC940DD62E41F10027AF002F8B0 \
55+
; do \
56+
gpg --batch --keyserver hkps://keys.openpgp.org --recv-keys "$key" || \
57+
gpg --batch --keyserver keyserver.ubuntu.com --recv-keys "$key" ; \
58+
done \
59+
&& curl -fsSLO --compressed "https://nodejs.org/dist/v$NODE_VERSION/node-v$NODE_VERSION.tar.xz" \
60+
&& curl -fsSLO --compressed "https://nodejs.org/dist/v$NODE_VERSION/SHASUMS256.txt.asc" \
61+
&& gpg --batch --decrypt --output SHASUMS256.txt SHASUMS256.txt.asc \
62+
&& gpgconf --kill all \
63+
&& rm -rf "$GNUPGHOME" \
64+
&& grep " node-v$NODE_VERSION.tar.xz\$" SHASUMS256.txt | sha256sum -c - \
65+
&& tar -xf "node-v$NODE_VERSION.tar.xz" \
66+
&& cd "node-v$NODE_VERSION" \
67+
&& ./configure \
68+
&& make -j$(getconf _NPROCESSORS_ONLN) V= \
69+
&& make install \
70+
&& apk del .build-deps-full \
71+
&& cd .. \
72+
&& rm -Rf "node-v$NODE_VERSION" \
73+
&& rm "node-v$NODE_VERSION.tar.xz" SHASUMS256.txt.asc SHASUMS256.txt; \
74+
fi \
75+
&& rm -f "node-v$NODE_VERSION-linux-$ARCH-musl.tar.xz" \
76+
# Remove unused OpenSSL headers to save ~34MB. See this NodeJS issue: https://github.com/nodejs/node/issues/46451
77+
&& find /usr/local/include/node/openssl/archs -mindepth 1 -maxdepth 1 ! -name "$OPENSSL_ARCH" -exec rm -rf {} \; \
78+
&& apk del .build-deps \
79+
# smoke tests
80+
&& node --version \
81+
&& npm --version
82+
83+
ENV YARN_VERSION 1.22.22
84+
85+
RUN apk add --no-cache --virtual .build-deps-yarn curl gnupg tar \
86+
# use pre-existing gpg directory, see https://github.com/nodejs/docker-node/pull/1895#issuecomment-1550389150
87+
&& export GNUPGHOME="$(mktemp -d)" \
88+
&& for key in \
89+
6A010C5166006599AA17F08146C2130DFD2497F5 \
90+
; do \
91+
gpg --batch --keyserver hkps://keys.openpgp.org --recv-keys "$key" || \
92+
gpg --batch --keyserver keyserver.ubuntu.com --recv-keys "$key" ; \
93+
done \
94+
&& curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz" \
95+
&& curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz.asc" \
96+
&& gpg --batch --verify yarn-v$YARN_VERSION.tar.gz.asc yarn-v$YARN_VERSION.tar.gz \
97+
&& gpgconf --kill all \
98+
&& rm -rf "$GNUPGHOME" \
99+
&& mkdir -p /opt \
100+
&& tar -xzf yarn-v$YARN_VERSION.tar.gz -C /opt/ \
101+
&& ln -s /opt/yarn-v$YARN_VERSION/bin/yarn /usr/local/bin/yarn \
102+
&& ln -s /opt/yarn-v$YARN_VERSION/bin/yarnpkg /usr/local/bin/yarnpkg \
103+
&& rm yarn-v$YARN_VERSION.tar.gz.asc yarn-v$YARN_VERSION.tar.gz \
104+
&& apk del .build-deps-yarn \
105+
# smoke test
106+
&& yarn --version \
107+
&& rm -rf /tmp/*
108+
109+
COPY docker-entrypoint.sh /usr/local/bin/
110+
ENTRYPOINT ["docker-entrypoint.sh"]
111+
112+
CMD [ "node" ]

23/alpine3.19/docker-entrypoint.sh

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
#!/bin/sh
2+
set -e
3+
4+
# Run command with node if the first argument contains a "-" or is not a system command. The last
5+
# part inside the "{}" is a workaround for the following bug in ash/dash:
6+
# https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=874264
7+
if [ "${1#-}" != "${1}" ] || [ -z "$(command -v "${1}")" ] || { [ -f "${1}" ] && ! [ -x "${1}" ]; }; then
8+
set -- node "$@"
9+
fi
10+
11+
exec "$@"

23/alpine3.20/Dockerfile

+112
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,112 @@
1+
FROM alpine:3.20
2+
3+
ENV NODE_VERSION 23.0.0
4+
5+
RUN addgroup -g 1000 node \
6+
&& adduser -u 1000 -G node -s /bin/sh -D node \
7+
&& apk add --no-cache \
8+
libstdc++ \
9+
&& apk add --no-cache --virtual .build-deps \
10+
curl \
11+
&& ARCH= OPENSSL_ARCH='linux*' && alpineArch="$(apk --print-arch)" \
12+
&& case "${alpineArch##*-}" in \
13+
x86_64) ARCH='x64' CHECKSUM="738b28f80f9aedaae3a7e91db8294cb96e2f87078fe7ff8826d612405ee2d8cb" OPENSSL_ARCH=linux-x86_64;; \
14+
x86) OPENSSL_ARCH=linux-elf;; \
15+
aarch64) OPENSSL_ARCH=linux-aarch64;; \
16+
arm*) OPENSSL_ARCH=linux-armv4;; \
17+
ppc64le) OPENSSL_ARCH=linux-ppc64le;; \
18+
s390x) OPENSSL_ARCH=linux-s390x;; \
19+
*) ;; \
20+
esac \
21+
&& if [ -n "${CHECKSUM}" ]; then \
22+
set -eu; \
23+
curl -fsSLO --compressed "https://unofficial-builds.nodejs.org/download/release/v$NODE_VERSION/node-v$NODE_VERSION-linux-$ARCH-musl.tar.xz"; \
24+
echo "$CHECKSUM node-v$NODE_VERSION-linux-$ARCH-musl.tar.xz" | sha256sum -c - \
25+
&& tar -xJf "node-v$NODE_VERSION-linux-$ARCH-musl.tar.xz" -C /usr/local --strip-components=1 --no-same-owner \
26+
&& ln -s /usr/local/bin/node /usr/local/bin/nodejs; \
27+
else \
28+
echo "Building from source" \
29+
# backup build
30+
&& apk add --no-cache --virtual .build-deps-full \
31+
binutils-gold \
32+
g++ \
33+
gcc \
34+
gnupg \
35+
libgcc \
36+
linux-headers \
37+
make \
38+
python3 \
39+
py-setuptools \
40+
# use pre-existing gpg directory, see https://github.com/nodejs/docker-node/pull/1895#issuecomment-1550389150
41+
&& export GNUPGHOME="$(mktemp -d)" \
42+
# gpg keys listed at https://github.com/nodejs/node#release-keys
43+
&& for key in \
44+
C0D6248439F1D5604AAFFB4021D900FFDB233756 \
45+
4ED778F539E3634C779C87C6D7062848A1AB005C \
46+
141F07595B7B3FFE74309A937405533BE57C7D57 \
47+
74F12602B6F1C4E913FAA37AD3A89613643B6201 \
48+
DD792F5973C6DE52C432CBDAC77ABFA00DDBF2B7 \
49+
CC68F5A3106FF448322E48ED27F5E38D5B0A215F \
50+
8FCCA13FEF1D0C2E91008E09770F7A9A5AE15600 \
51+
890C08DB8579162FEE0DF9DB8BEAB4DFCF555EF4 \
52+
C82FA3AE1CBEDC6BE46B9360C43CEC45C17AB93C \
53+
108F52B48DB57BB0CC439B2997B01419BD92F80A \
54+
A363A499291CBBC940DD62E41F10027AF002F8B0 \
55+
; do \
56+
gpg --batch --keyserver hkps://keys.openpgp.org --recv-keys "$key" || \
57+
gpg --batch --keyserver keyserver.ubuntu.com --recv-keys "$key" ; \
58+
done \
59+
&& curl -fsSLO --compressed "https://nodejs.org/dist/v$NODE_VERSION/node-v$NODE_VERSION.tar.xz" \
60+
&& curl -fsSLO --compressed "https://nodejs.org/dist/v$NODE_VERSION/SHASUMS256.txt.asc" \
61+
&& gpg --batch --decrypt --output SHASUMS256.txt SHASUMS256.txt.asc \
62+
&& gpgconf --kill all \
63+
&& rm -rf "$GNUPGHOME" \
64+
&& grep " node-v$NODE_VERSION.tar.xz\$" SHASUMS256.txt | sha256sum -c - \
65+
&& tar -xf "node-v$NODE_VERSION.tar.xz" \
66+
&& cd "node-v$NODE_VERSION" \
67+
&& ./configure \
68+
&& make -j$(getconf _NPROCESSORS_ONLN) V= \
69+
&& make install \
70+
&& apk del .build-deps-full \
71+
&& cd .. \
72+
&& rm -Rf "node-v$NODE_VERSION" \
73+
&& rm "node-v$NODE_VERSION.tar.xz" SHASUMS256.txt.asc SHASUMS256.txt; \
74+
fi \
75+
&& rm -f "node-v$NODE_VERSION-linux-$ARCH-musl.tar.xz" \
76+
# Remove unused OpenSSL headers to save ~34MB. See this NodeJS issue: https://github.com/nodejs/node/issues/46451
77+
&& find /usr/local/include/node/openssl/archs -mindepth 1 -maxdepth 1 ! -name "$OPENSSL_ARCH" -exec rm -rf {} \; \
78+
&& apk del .build-deps \
79+
# smoke tests
80+
&& node --version \
81+
&& npm --version
82+
83+
ENV YARN_VERSION 1.22.22
84+
85+
RUN apk add --no-cache --virtual .build-deps-yarn curl gnupg tar \
86+
# use pre-existing gpg directory, see https://github.com/nodejs/docker-node/pull/1895#issuecomment-1550389150
87+
&& export GNUPGHOME="$(mktemp -d)" \
88+
&& for key in \
89+
6A010C5166006599AA17F08146C2130DFD2497F5 \
90+
; do \
91+
gpg --batch --keyserver hkps://keys.openpgp.org --recv-keys "$key" || \
92+
gpg --batch --keyserver keyserver.ubuntu.com --recv-keys "$key" ; \
93+
done \
94+
&& curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz" \
95+
&& curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz.asc" \
96+
&& gpg --batch --verify yarn-v$YARN_VERSION.tar.gz.asc yarn-v$YARN_VERSION.tar.gz \
97+
&& gpgconf --kill all \
98+
&& rm -rf "$GNUPGHOME" \
99+
&& mkdir -p /opt \
100+
&& tar -xzf yarn-v$YARN_VERSION.tar.gz -C /opt/ \
101+
&& ln -s /opt/yarn-v$YARN_VERSION/bin/yarn /usr/local/bin/yarn \
102+
&& ln -s /opt/yarn-v$YARN_VERSION/bin/yarnpkg /usr/local/bin/yarnpkg \
103+
&& rm yarn-v$YARN_VERSION.tar.gz.asc yarn-v$YARN_VERSION.tar.gz \
104+
&& apk del .build-deps-yarn \
105+
# smoke test
106+
&& yarn --version \
107+
&& rm -rf /tmp/*
108+
109+
COPY docker-entrypoint.sh /usr/local/bin/
110+
ENTRYPOINT ["docker-entrypoint.sh"]
111+
112+
CMD [ "node" ]

23/alpine3.20/docker-entrypoint.sh

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
#!/bin/sh
2+
set -e
3+
4+
# Run command with node if the first argument contains a "-" or is not a system command. The last
5+
# part inside the "{}" is a workaround for the following bug in ash/dash:
6+
# https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=874264
7+
if [ "${1#-}" != "${1}" ] || [ -z "$(command -v "${1}")" ] || { [ -f "${1}" ] && ! [ -x "${1}" ]; }; then
8+
set -- node "$@"
9+
fi
10+
11+
exec "$@"

23/bookworm-slim/Dockerfile

+106
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,106 @@
1+
FROM debian:bookworm-slim
2+
3+
RUN groupadd --gid 1000 node \
4+
&& useradd --uid 1000 --gid node --shell /bin/bash --create-home node
5+
6+
ENV NODE_VERSION 23.0.0
7+
8+
RUN ARCH= OPENSSL_ARCH= && dpkgArch="$(dpkg --print-architecture)" \
9+
&& case "${dpkgArch##*-}" in \
10+
amd64) ARCH='x64' OPENSSL_ARCH='linux-x86_64';; \
11+
ppc64el) ARCH='ppc64le' OPENSSL_ARCH='linux-ppc64le';; \
12+
s390x) ARCH='s390x' OPENSSL_ARCH='linux*-s390x';; \
13+
arm64) ARCH='arm64' OPENSSL_ARCH='linux-aarch64';; \
14+
armhf) ARCH='armv7l' OPENSSL_ARCH='linux-armv4';; \
15+
i386) ARCH='x86' OPENSSL_ARCH='linux-elf';; \
16+
*) echo "unsupported architecture"; exit 1 ;; \
17+
esac \
18+
&& set -ex \
19+
# libatomic1 for arm
20+
&& apt-get update && apt-get install -y ca-certificates curl wget gnupg dirmngr xz-utils libatomic1 --no-install-recommends \
21+
&& rm -rf /var/lib/apt/lists/* \
22+
# use pre-existing gpg directory, see https://github.com/nodejs/docker-node/pull/1895#issuecomment-1550389150
23+
&& export GNUPGHOME="$(mktemp -d)" \
24+
# gpg keys listed at https://github.com/nodejs/node#release-keys
25+
&& for key in \
26+
C0D6248439F1D5604AAFFB4021D900FFDB233756 \
27+
4ED778F539E3634C779C87C6D7062848A1AB005C \
28+
141F07595B7B3FFE74309A937405533BE57C7D57 \
29+
74F12602B6F1C4E913FAA37AD3A89613643B6201 \
30+
DD792F5973C6DE52C432CBDAC77ABFA00DDBF2B7 \
31+
CC68F5A3106FF448322E48ED27F5E38D5B0A215F \
32+
8FCCA13FEF1D0C2E91008E09770F7A9A5AE15600 \
33+
890C08DB8579162FEE0DF9DB8BEAB4DFCF555EF4 \
34+
C82FA3AE1CBEDC6BE46B9360C43CEC45C17AB93C \
35+
108F52B48DB57BB0CC439B2997B01419BD92F80A \
36+
A363A499291CBBC940DD62E41F10027AF002F8B0 \
37+
; do \
38+
gpg --batch --keyserver hkps://keys.openpgp.org --recv-keys "$key" || \
39+
gpg --batch --keyserver keyserver.ubuntu.com --recv-keys "$key" ; \
40+
done \
41+
&& curl -fsSLO --compressed "https://nodejs.org/dist/v$NODE_VERSION/node-v$NODE_VERSION-linux-$ARCH.tar.xz" \
42+
&& curl -fsSLO --compressed "https://nodejs.org/dist/v$NODE_VERSION/SHASUMS256.txt.asc" \
43+
&& gpg --batch --decrypt --output SHASUMS256.txt SHASUMS256.txt.asc \
44+
&& gpgconf --kill all \
45+
&& rm -rf "$GNUPGHOME" \
46+
&& grep " node-v$NODE_VERSION-linux-$ARCH.tar.xz\$" SHASUMS256.txt | sha256sum -c - \
47+
&& tar -xJf "node-v$NODE_VERSION-linux-$ARCH.tar.xz" -C /usr/local --strip-components=1 --no-same-owner \
48+
&& rm "node-v$NODE_VERSION-linux-$ARCH.tar.xz" SHASUMS256.txt.asc SHASUMS256.txt \
49+
# Remove unused OpenSSL headers to save ~34MB. See this NodeJS issue: https://github.com/nodejs/node/issues/46451
50+
&& find /usr/local/include/node/openssl/archs -mindepth 1 -maxdepth 1 ! -name "$OPENSSL_ARCH" -exec rm -rf {} \; \
51+
&& apt-mark auto '.*' > /dev/null \
52+
&& find /usr/local -type f -executable -exec ldd '{}' ';' \
53+
| awk '/=>/ { so = $(NF-1); if (index(so, "/usr/local/") == 1) { next }; gsub("^/(usr/)?", "", so); print so }' \
54+
| sort -u \
55+
| xargs -r dpkg-query --search \
56+
| cut -d: -f1 \
57+
| sort -u \
58+
| xargs -r apt-mark manual \
59+
&& apt-get purge -y --auto-remove -o APT::AutoRemove::RecommendsImportant=false \
60+
&& ln -s /usr/local/bin/node /usr/local/bin/nodejs \
61+
# smoke tests
62+
&& node --version \
63+
&& npm --version
64+
65+
ENV YARN_VERSION 1.22.22
66+
67+
RUN set -ex \
68+
&& savedAptMark="$(apt-mark showmanual)" \
69+
&& apt-get update && apt-get install -y ca-certificates curl wget gnupg dirmngr --no-install-recommends \
70+
&& rm -rf /var/lib/apt/lists/* \
71+
# use pre-existing gpg directory, see https://github.com/nodejs/docker-node/pull/1895#issuecomment-1550389150
72+
&& export GNUPGHOME="$(mktemp -d)" \
73+
&& for key in \
74+
6A010C5166006599AA17F08146C2130DFD2497F5 \
75+
; do \
76+
gpg --batch --keyserver hkps://keys.openpgp.org --recv-keys "$key" || \
77+
gpg --batch --keyserver keyserver.ubuntu.com --recv-keys "$key" ; \
78+
done \
79+
&& curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz" \
80+
&& curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz.asc" \
81+
&& gpg --batch --verify yarn-v$YARN_VERSION.tar.gz.asc yarn-v$YARN_VERSION.tar.gz \
82+
&& gpgconf --kill all \
83+
&& rm -rf "$GNUPGHOME" \
84+
&& mkdir -p /opt \
85+
&& tar -xzf yarn-v$YARN_VERSION.tar.gz -C /opt/ \
86+
&& ln -s /opt/yarn-v$YARN_VERSION/bin/yarn /usr/local/bin/yarn \
87+
&& ln -s /opt/yarn-v$YARN_VERSION/bin/yarnpkg /usr/local/bin/yarnpkg \
88+
&& rm yarn-v$YARN_VERSION.tar.gz.asc yarn-v$YARN_VERSION.tar.gz \
89+
&& apt-mark auto '.*' > /dev/null \
90+
&& { [ -z "$savedAptMark" ] || apt-mark manual $savedAptMark > /dev/null; } \
91+
&& find /usr/local -type f -executable -exec ldd '{}' ';' \
92+
| awk '/=>/ { so = $(NF-1); if (index(so, "/usr/local/") == 1) { next }; gsub("^/(usr/)?", "", so); print so }' \
93+
| sort -u \
94+
| xargs -r dpkg-query --search \
95+
| cut -d: -f1 \
96+
| sort -u \
97+
| xargs -r apt-mark manual \
98+
&& apt-get purge -y --auto-remove -o APT::AutoRemove::RecommendsImportant=false \
99+
# smoke test
100+
&& yarn --version \
101+
&& rm -rf /tmp/*
102+
103+
COPY docker-entrypoint.sh /usr/local/bin/
104+
ENTRYPOINT ["docker-entrypoint.sh"]
105+
106+
CMD [ "node" ]

23/bookworm-slim/docker-entrypoint.sh

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
#!/bin/sh
2+
set -e
3+
4+
# Run command with node if the first argument contains a "-" or is not a system command. The last
5+
# part inside the "{}" is a workaround for the following bug in ash/dash:
6+
# https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=874264
7+
if [ "${1#-}" != "${1}" ] || [ -z "$(command -v "${1}")" ] || { [ -f "${1}" ] && ! [ -x "${1}" ]; }; then
8+
set -- node "$@"
9+
fi
10+
11+
exec "$@"

0 commit comments

Comments
 (0)