ஒற்றைப் பக்க பயன்பாடு (SPA) என்பது நிலையான ஜாவாஸ்கிரிப்ட் மற்றும் HTML கோப்புகள் மற்றும் படங்கள் மற்றும் பிற ஆதாரங்களின் தொகுப்பாகும். அவை மாறும் வகையில் மாறாததால், அவற்றை ஆன்லைனில் வெளியிடுவது மிகவும் எளிதானது. எளிய கிட்ஹப் பக்கங்களில் தொடங்கி (மற்றும் சிலருக்கு narod.ru உடன் கூட) மற்றும் Amazon S3 போன்ற CDN உடன் முடிவடையும் ஏராளமான மலிவான மற்றும் இலவச சேவைகள் உள்ளன. இருப்பினும், எனக்கு வேறு ஏதாவது தேவைப்பட்டது.
குபெர்னெட்ஸ் கிளஸ்டரின் ஒரு பகுதியாக தயாரிப்பிலும், SPA என்றால் என்னவென்று தெரியாத பின்-இறுதி டெவலப்பரின் மெஷினிலும் எளிதாகத் தொடங்குவதற்கு, SPA உடன் ஒரு டோக்கர் படம் எனக்குத் தேவைப்பட்டது.
எனக்கான பின்வரும் படத் தேவைகளை நான் தீர்மானித்துள்ளேன்:
- பயன்பாட்டின் எளிமை (ஆனால் சட்டசபை அல்ல);
- வட்டு மற்றும் ரேம் அடிப்படையில் குறைந்தபட்ச அளவு;
- வெவ்வேறு சூழல்களில் படத்தைப் பயன்படுத்தக்கூடிய சூழல் மாறிகள் மூலம் உள்ளமைவு;
- கோப்புகளின் மிகவும் திறமையான விநியோகம்.
எப்படி என்பதை இன்று நான் உங்களுக்கு சொல்கிறேன்:
- குடல் nginx;
- ஆதாரங்களில் இருந்து brotli உருவாக்க;
- சூழல் மாறிகளைப் புரிந்துகொள்ள நிலையான கோப்புகளை கற்பித்தல்;
- மற்றும் நிச்சயமாக இவை அனைத்திலிருந்தும் டோக்கர் படத்தை எவ்வாறு இணைப்பது.
இக்கட்டுரையின் நோக்கம் எனது அனுபவத்தைப் பகிர்ந்துகொள்வதும் அனுபவமிக்க சமூக உறுப்பினர்களை ஆக்கபூர்வமான விமர்சனங்களுக்குத் தூண்டுவதும் ஆகும்.
சட்டசபைக்கு ஒரு படத்தை உருவாக்குதல்
இறுதி டோக்கர் படத்தை சிறியதாக மாற்ற, நீங்கள் இரண்டு விதிகளை கடைபிடிக்க வேண்டும்: குறைந்தபட்ச அடுக்குகள் மற்றும் குறைந்தபட்ச அடிப்படை படம். மிகச்சிறிய அடிப்படைப் படங்களில் ஒன்று ஆல்பைன் லினக்ஸ் படம், அதனால் நான் அதைத் தேர்வு செய்கிறேன். ஆல்பைன் உற்பத்திக்கு ஏற்றது அல்ல என்று சிலர் வாதிடலாம், மேலும் அவை சரியாக இருக்கலாம். ஆனால் தனிப்பட்ட முறையில், அவருடன் எனக்கு எந்த பிரச்சனையும் இல்லை, அவருக்கு எதிராக எந்த வாதங்களும் இல்லை.
குறைவான அடுக்குகள் இருக்க, படத்தை 2 நிலைகளில் அசெம்பிள் செய்வேன். முதலாவது வரைவு; அனைத்து துணை பயன்பாடுகளும் தற்காலிக கோப்புகளும் அதில் இருக்கும். இறுதி கட்டத்தில் நான் விண்ணப்பத்தின் இறுதி பதிப்பை மட்டுமே எழுதுவேன்.
துணைப் படத்துடன் ஆரம்பிக்கலாம்.
ஒரு SPA பயன்பாட்டை தொகுக்க, உங்களுக்கு பொதுவாக node.js தேவைப்படும். npm மற்றும் நூல் தொகுப்பு மேலாளர்களுடன் வரும் அதிகாரப்பூர்வ படத்தை நான் எடுக்கிறேன். என் சார்பாக, நான் சில npm தொகுப்புகளை உருவாக்க தேவையான node-gyp ஐயும், Google வழங்கும் Brotli கம்ப்ரஸரையும் சேர்ப்பேன், இது எங்களுக்கு பின்னர் பயனுள்ளதாக இருக்கும்.
கருத்துகளுடன் Dockerfile.
# Базовый образ
FROM node:12-alpine
LABEL maintainer="Aleksey Maydokin <[email protected]>"
ENV BROTLI_VERSION 1.0.7
# Пакеты, которые нужны, чтобы собрать из исходников Brotli
RUN apk add --no-cache --virtual .build-deps
bash
gcc
libc-dev
make
linux-headers
cmake
curl
&& mkdir -p /usr/src
# Исходники Brotli скачиваем из официального репозитория
&& curl -LSs https://github.com/google/brotli/archive/v$BROTLI_VERSION.tar.gz | tar xzf - -C /usr/src
&& cd /usr/src/brotli-$BROTLI_VERSION
# Компилируем Brotli
&& ./configure-cmake --disable-debug && make -j$(getconf _NPROCESSORS_ONLN) && make install
# Добавляем node-gyp
&& yarn global add node-gyp
# Убираем за собой мусор
&& apk del .build-deps && yarn cache clean && rm -rf /usr/src
ஏற்கனவே இங்கே நான் மினிமலிசத்திற்காக போராடுகிறேன், எனவே படம் ஒரு பெரிய குழுவால் ஒன்றாக இணைக்கப்பட்டுள்ளது.
முடிக்கப்பட்ட படத்தை இங்கே காணலாம்:
Nginx
நிலையான உள்ளடக்கத்தை விநியோகிக்க நீங்கள் எந்த இணைய சேவையகத்தையும் பயன்படுத்தலாம். நான் nginx உடன் வேலை செய்யப் பழகிவிட்டேன், எனவே நான் இப்போது அதைப் பயன்படுத்துகிறேன்.
Nginx ஒரு அதிகாரப்பூர்வ டோக்கர் படத்தைக் கொண்டுள்ளது, ஆனால் இது எளிமையான நிலையான விநியோகத்திற்கான பல தொகுதிகளைக் கொண்டுள்ளது. டெலிவரியில் எவை சேர்க்கப்பட்டுள்ளன என்பதை ஒரு சிறப்பு குழு அல்லது அதிகாரப்பூர்வ Dockerfile இல் பார்க்கலாம்.
$ docker run --rm nginx:1-alpine nginx -V
nginx version: nginx/1.17.9
built by gcc 8.3.0 (Alpine 8.3.0)
built with OpenSSL 1.1.1d 10 Sep 2019
TLS SNI support enabled
configure arguments: --prefix=/etc/nginx --sbin-path=/usr/sbin/nginx --modules-path=/usr/lib/nginx/modules --conf-path=/etc/nginx/nginx.conf --error-log-path=/var/log/nginx/error.log --http-log-path=/var/log/nginx/access.log --pid-path=/var/run/nginx.pid --lock-path=/var/run/nginx.lock --http-client-body-temp-path=/var/cache/nginx/client_temp --http-proxy-temp-path=/var/cache/nginx/proxy_temp --http-fastcgi-temp-path=/var/cache/nginx/fastcgi_temp --http-uwsgi-temp-path=/var/cache/nginx/uwsgi_temp --http-scgi-temp-path=/var/cache/nginx/scgi_temp --with-perl_modules_path=/usr/lib/perl5/vendor_perl --user=nginx --group=nginx --with-compat --with-file-aio --with-threads --with-http_addition_module --with-http_auth_request_module --with-http_dav_module --with-http_flv_module --with-http_gunzip_module --with-http_gzip_static_module --with-http_mp4_module --with-http_random_index_module --with-http_realip_module --with-http_secure_link_module --with-http_slice_module --with-http_ssl_module --with-http_stub_status_module --with-http_sub_module --with-http_v2_module --with-mail --with-mail_ssl_module --with-stream --with-stream_realip_module --with-stream_ssl_module --with-stream_ssl_preread_module --with-cc-opt='-Os -fomit-frame-pointer' --with-ld-opt=-Wl,--as-needed
நான் டாக்கர்ஃபைலை ஒரு அடிப்படையாகப் பயன்படுத்துவேன், ஆனால் நிலையான உள்ளடக்கத்தை விநியோகிக்கத் தேவையானதை மட்டும் அதில் விட்டுவிடுவேன். எனது பதிப்பு HTTPS மூலம் வேலை செய்ய முடியாது, அங்கீகாரத்தை ஆதரிக்காது, மேலும் பல. ஆனால் எனது பதிப்பு ப்ரோட்லி அல்காரிதம் மூலம் சுருக்கப்பட்ட கோப்புகளை விநியோகிக்க முடியும், இது gzip ஐ விட சற்று அதிக திறன் கொண்டது. நாங்கள் கோப்புகளை ஒரு முறை சுருக்குவோம்; இதை பறக்க வேண்டிய அவசியமில்லை.
நான் முடித்த Dockerfile இதுதான். ரஷ்ய மொழியில் உள்ள கருத்துகள் என்னுடையது, ஆங்கிலத்தில் - அசலில் இருந்து.
டோக்கர்ஃபைல்
# Базовый образ снова Alpine
FROM alpine:3.9
LABEL maintainer="Aleksey Maydokin <[email protected]>"
ENV NGINX_VERSION 1.16.0
ENV NGX_BROTLI_VERSION 0.1.2
ENV BROTLI_VERSION 1.0.7
RUN set -x
&& addgroup -S nginx
&& adduser -D -S -h /var/cache/nginx -s /sbin/nologin -G nginx nginx
# Устанавливаем пакеты, которые нужны чтобы собрать nginx и модуль ngx_brotli к нему
&& apk add --no-cache --virtual .build-deps
gcc
libc-dev
make
linux-headers
curl
&& mkdir -p /usr/src
# Скачиваем исходники
&& curl -LSs https://nginx.org/download/nginx-$NGINX_VERSION.tar.gz | tar xzf - -C /usr/src
&& curl -LSs https://github.com/eustas/ngx_brotli/archive/v$NGX_BROTLI_VERSION.tar.gz | tar xzf - -C /usr/src
&& curl -LSs https://github.com/google/brotli/archive/v$BROTLI_VERSION.tar.gz | tar xzf - -C /usr/src
&& rm -rf /usr/src/ngx_brotli-$NGX_BROTLI_VERSION/deps/brotli/
&& ln -s /usr/src/brotli-$BROTLI_VERSION /usr/src/ngx_brotli-$NGX_BROTLI_VERSION/deps/brotli
&& cd /usr/src/nginx-$NGINX_VERSION
&& CNF="
--prefix=/etc/nginx
--sbin-path=/usr/sbin/nginx
--modules-path=/usr/lib/nginx/modules
--conf-path=/etc/nginx/nginx.conf
--error-log-path=/var/log/nginx/error.log
--http-log-path=/var/log/nginx/access.log
--pid-path=/var/run/nginx.pid
--lock-path=/var/run/nginx.lock
--http-client-body-temp-path=/var/cache/nginx/client_temp
--http-proxy-temp-path=/var/cache/nginx/proxy_temp
--http-fastcgi-temp-path=/var/cache/nginx/fastcgi_temp
--http-uwsgi-temp-path=/var/cache/nginx/uwsgi_temp
--http-scgi-temp-path=/var/cache/nginx/scgi_temp
--user=nginx
--group=nginx
--without-http_ssi_module
--without-http_userid_module
--without-http_access_module
--without-http_auth_basic_module
--without-http_mirror_module
--without-http_autoindex_module
--without-http_geo_module
--without-http_split_clients_module
--without-http_referer_module
--without-http_rewrite_module
--without-http_proxy_module
--without-http_fastcgi_module
--without-http_uwsgi_module
--without-http_scgi_module
--without-http_grpc_module
--without-http_memcached_module
--without-http_limit_conn_module
--without-http_limit_req_module
--without-http_empty_gif_module
--without-http_browser_module
--without-http_upstream_hash_module
--without-http_upstream_ip_hash_module
--without-http_upstream_least_conn_module
--without-http_upstream_keepalive_module
--without-http_upstream_zone_module
--without-http_gzip_module
--with-http_gzip_static_module
--with-threads
--with-compat
--with-file-aio
--add-dynamic-module=/usr/src/ngx_brotli-$NGX_BROTLI_VERSION
"
# Собираем
&& ./configure $CNF
&& make -j$(getconf _NPROCESSORS_ONLN)
&& make install
&& rm -rf /usr/src/
# Удаляем динамический brotli модуль, оставляя только статический
&& rm /usr/lib/nginx/modules/ngx_http_brotli_filter_module.so
&& sed -i '$ d' /etc/apk/repositories
# Bring in gettext so we can get `envsubst`, then throw
# the rest away. To do this, we need to install `gettext`
# then move `envsubst` out of the way so `gettext` can
# be deleted completely, then move `envsubst` back.
&& apk add --no-cache --virtual .gettext gettext
&& mv /usr/bin/envsubst /tmp/
&& runDeps="$(
scanelf --needed --nobanner /usr/sbin/nginx /usr/lib/nginx/modules/*.so /tmp/envsubst
| awk '{ gsub(/,/, "nso:", $2); print "so:" $2 }'
| sort -u
| xargs -r apk info --installed
| sort -u
)"
&& apk add --no-cache $runDeps
&& apk del .build-deps
&& apk del .gettext
&& mv /tmp/envsubst /usr/local/bin/
# Bring in tzdata so users could set the timezones through the environment
# variables
&& apk add --no-cache tzdata
# forward request and error logs to docker log collector
&& ln -sf /dev/stdout /var/log/nginx/access.log
&& ln -sf /dev/stderr /var/log/nginx/error.log
COPY nginx.conf /etc/nginx/nginx.conf
EXPOSE 80
STOPSIGNAL SIGTERM
CMD ["nginx", "-g", "daemon off;"]
நான் உடனடியாக nginx.conf ஐ சரிசெய்வேன், அதனால் gzip மற்றும் brotli இயல்பாகவே இயக்கப்படும். நான் கேச்சிங் தலைப்புகளையும் சேர்ப்பேன், ஏனென்றால் நாங்கள் எப்போதும் நிலையானதை மாற்ற மாட்டோம். மேலும் 404 கோரிக்கைகளையும் index.html க்கு திருப்பிவிடுவதே இறுதித் தொடுதலாகும், SPA இல் வழிசெலுத்துவதற்கு இது அவசியம்.
nginx.conf
user nginx;
worker_processes 1;
error_log /var/log/nginx/error.log warn;
pid /var/run/nginx.pid;
load_module /usr/lib/nginx/modules/ngx_http_brotli_static_module.so;
events {
worker_connections 1024;
}
http {
include mime.types;
default_type application/octet-stream;
log_format main '$remote_addr - $remote_user [$time_local] "$request" '
'$status $body_bytes_sent "$http_referer" '
'"$http_user_agent" "$http_x_forwarded_for"';
access_log /var/log/nginx/access.log main;
sendfile on;
keepalive_timeout 65;
gzip_static on;
brotli_static on;
server {
listen 80;
server_name localhost;
charset utf-8;
location / {
root html;
try_files $uri /index.html;
etag on;
expires max;
add_header Cache-Control public;
location = /index.html {
expires 0;
add_header Cache-Control "no-cache, public, must-revalidate, proxy-revalidate";
}
}
}
}
முடிக்கப்பட்ட படத்தை இங்கே பதிவிறக்கம் செய்யலாம்:
சூழல் மாறிகளைப் புரிந்துகொள்ள புள்ளியியல் கற்பித்தல்
SPA இல் அமைப்புகள் ஏன் தேவைப்படலாம்? எடுத்துக்காட்டாக, எந்த RESTful API ஐப் பயன்படுத்த வேண்டும் என்பதைக் குறிப்பிடுவதற்காக. பொதுவாக, விரும்பிய சூழலுக்கான அமைப்புகள் உருவாக்க கட்டத்தில் SPA க்கு மாற்றப்படும். நீங்கள் ஏதாவது மாற்ற வேண்டும் என்றால், நீங்கள் பயன்பாட்டை மீண்டும் உருவாக்க வேண்டும். எனக்கு அது வேண்டாம். CI கட்டத்தில் பயன்பாடு ஒருமுறை கட்டமைக்கப்பட வேண்டும், மேலும் சூழல் மாறிகளைப் பயன்படுத்தி CD கட்டத்தில் தேவையான அளவு கட்டமைக்கப்பட வேண்டும் என்று நான் விரும்புகிறேன்.
நிச்சயமாக, நிலையான கோப்புகள் எந்த சூழல் மாறிகளையும் புரிந்து கொள்ளாது. எனவே, நீங்கள் ஒரு தந்திரத்தைப் பயன்படுத்த வேண்டும். இறுதிப் படத்தில், நான் nginx ஐத் தொடங்க மாட்டேன், ஆனால் சூழல் மாறிகளைப் படிக்கும் ஒரு சிறப்பு ஷெல் ஸ்கிரிப்ட், அவற்றை நிலையான கோப்புகளில் எழுதி, அவற்றை சுருக்கி, பின்னர் மட்டுமே கட்டுப்பாட்டை nginx க்கு மாற்றும்.
இந்த நோக்கத்திற்காக, Dockerfile ENTRYPOINT அளவுருவை வழங்குகிறது. பின்வரும் ஸ்கிரிப்டை அவருக்கு வழங்குவோம் (உதாரணமாக கோணத்தைப் பயன்படுத்தி):
docker-entrypoint.sh
#!/bin/sh set -e FLAG_FILE="/configured" TARGET_DIR="/etc/nginx/html" replace_vars () { ENV_VARS='$(awk 'BEGIN{for(v in ENVIRON) print "
quot;v}')'
# В Angular ищем плейсхолдеры в main-файлах
for f in "$TARGET_DIR"/main*.js; do
# envsubst заменяет в файлах плейсхолдеры на значения из переменных окружения
echo "$(envsubst "$ENV_VARS" < "$f")" > "$f"
done
}
compress () {
for i in $(find "$TARGET_DIR" | grep -E ".css$|.html$|.js$|.svg$|.txt$|.ttf
quot;); do
# Используем максимальную степень сжатия
gzip -9kf "$i" && brotli -fZ "$i"
done
}
if [ "$1" = 'nginx' ]; then
# Флаг нужен, чтобы выполнить скрипт только при самом первом запуске
if [ ! -e "$FLAG_FILE" ]; then
echo "Running init script"
echo "Replacing env vars"
replace_vars
echo "Compressing files"
compress
touch $FLAG_FILE
echo "Done"
fi
fi
exec "$@"
ஸ்கிரிப்ட் அதன் வேலையைச் செய்ய, அமைப்புகள் இந்த வடிவத்தில் js கோப்புகளில் எழுதப்பட வேண்டும்:${API_URL}
.பெரும்பாலான நவீன SPAக்கள் உருவாக்கும்போது அவற்றின் கோப்புகளில் ஹாஷ்களைச் சேர்க்கின்றன என்பது குறிப்பிடத்தக்கது. உலாவி நீண்ட நேரம் கோப்பை பாதுகாப்பாக தேக்ககப்படுத்த இது அவசியம். கோப்பு மாறினால், அதன் ஹாஷ் மாறும், இது உலாவி கோப்பை மீண்டும் பதிவிறக்கம் செய்ய கட்டாயப்படுத்தும்.
துரதிர்ஷ்டவசமாக, எனது முறையில், சூழல் மாறிகள் மூலம் உள்ளமைவை மாற்றுவது கோப்பு ஹாஷில் மாற்றத்திற்கு வழிவகுக்காது, அதாவது உலாவி தற்காலிக சேமிப்பு வேறு வழியில் செல்லாததாக இருக்க வேண்டும். வெவ்வேறு சூழல்களில் வெவ்வேறு கட்டமைப்புகள் பயன்படுத்தப்படுவதால் எனக்கு இந்தப் பிரச்சனை இல்லை.
இறுதி படத்தை ஒன்றாக இணைத்தல்
இறுதியாக.
டோக்கர்ஃபைல்
# Первый базовый образ для сборки FROM alexxxnf/spa-builder as builder # Чтобы эффктивнее использовать кэш Docker-а, сначала устанавливаем только зависимости COPY ./package.json ./package-lock.json /app/ RUN cd /app && npm ci --no-audit # Потом собираем само приложение COPY . /app RUN cd /app && npm run build -- --prod --configuration=docker # Второй базовый образ для раздачи FROM alexxxnf/nginx-spa # Забираем из первого образа сначала компрессор COPY --from=builder /usr/local/bin/brotli /usr/local/bin # Потом добавляем чудо-скрипт COPY ./docker/docker-entrypoint.sh /docker-entrypoint.sh # И в конце забираем само приложение COPY --from=builder /app/dist/app /etc/nginx/html/ ENTRYPOINT ["/docker-entrypoint.sh"] CMD ["nginx", "-g", "daemon off;"]
இப்போது விளைந்த படத்தை எங்கு வேண்டுமானாலும் அசெம்பிள் செய்து பயன்படுத்தலாம்.
ஆதாரம்: www.habr.com