SHT30 온습도 모니터링 시스템 전체 소스(서버 PHP, STM32 펌웨어, SQL, 테스트). 전체 코드리뷰에서 도출된 보안 하드닝 10건 반영: - 요청 서명 HMAC-SHA256 전환(펌웨어 sig.c/서버 config.php/호스트 패리티 동시) - 재전송 방어 + 기본 API_KEY fail-closed + 디바이스 문자열 정제(api/sensor_data.php) - 오프라인 SMS 중복 발송 경합 제거(cron_heartbeat.php, 원자적 선점) - CSV 수식 주입 방지(monthly_report.php), 감사로그 회전 락(retention_cleanup.php) - 브루트포스 카운터 원자화(login.php), 예시 TOTP 비밀키 무효화, 마이그레이션 멱등화 _backup/(하드코딩 실 비밀값 포함)·config.local.php·런타임 상태는 .gitignore 제외.
146 lines
6.4 KiB
Bash
146 lines
6.4 KiB
Bash
#!/usr/bin/env bash
|
|
# =============================================================================
|
|
# vendor.sh - 제3자 의존성 벤더링 (NETWORKED 빌드 머신 전용)
|
|
#
|
|
# 폐쇄망(air-gapped) 타깃으로 전달하기 위해, 빌드/스테이징 머신에서 핀 고정된
|
|
# 제3자 의존성을 내려받아 SHA-256 으로 무결성을 검증한 뒤 firmware/third_party
|
|
# 아래에 펼친다. **네트워크가 허용되는 유일한 단계**다. 이후 build.ps1/CMake 는
|
|
# 네트워크 없이 빌드한다.
|
|
#
|
|
# 사용:
|
|
# firmware/scripts/vendor.sh # 표준 벤더링
|
|
# FORCE=1 firmware/scripts/vendor.sh # 기존 트리 덮어쓰기(재벤더링)
|
|
# KEEP_ARCHIVES=1 firmware/scripts/vendor.sh # 압축본 보존(_archives)
|
|
#
|
|
# 요구 도구: curl(또는 wget), sha256sum(또는 shasum), unzip, tar.
|
|
#
|
|
# 중요: 아래 DEPS 의 URL/SHA256/TAG 는 운영에서 쓸 정확한 핀 버전으로 교체할 것.
|
|
# SHA256 이 'TODO' 인 항목은 검증을 생략하고 계산된 해시를 출력만 한다
|
|
# (최초 1회 기록용). 기록 후 반드시 채워 넣어 재현성을 고정한다.
|
|
# TODO(vendor): 핀 버전 확정.
|
|
# =============================================================================
|
|
set -euo pipefail
|
|
|
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
FW_ROOT="$(cd "${SCRIPT_DIR}/.." && pwd)"
|
|
THIRD_PARTY="${FW_ROOT}/third_party"
|
|
ARCHIVE_DIR="${THIRD_PARTY}/_archives"
|
|
|
|
FORCE="${FORCE:-0}"
|
|
KEEP_ARCHIVES="${KEEP_ARCHIVES:-0}"
|
|
|
|
mkdir -p "${THIRD_PARTY}"
|
|
[ "${KEEP_ARCHIVES}" = "1" ] && mkdir -p "${ARCHIVE_DIR}"
|
|
|
|
# ── 핀 고정 의존성 (name|tag|url|sha256) ─────────────────────────────────────
|
|
# name : third_party 하위 디렉터리명 (CMakeLists 경로와 일치)
|
|
# tag : 릴리스/태그(추적용)
|
|
# url : 다운로드 URL
|
|
# sha256 : 압축 파일 SHA-256(소문자). 'TODO' 면 검증 생략(해시 출력만).
|
|
# TODO(vendor): 아래 값들을 정확한 핀 버전으로 교체.
|
|
DEPS=(
|
|
"STM32CubeF4|v1.28.0|https://github.com/STMicroelectronics/STM32CubeF4/archive/refs/tags/v1.28.0.zip|TODO"
|
|
"FreeRTOS-Kernel|V11.1.0|https://github.com/FreeRTOS/FreeRTOS-Kernel/archive/refs/tags/V11.1.0.zip|TODO"
|
|
"lwip|STABLE-2_2_0_RELEASE|https://github.com/lwip-tcpip/lwip/archive/refs/tags/STABLE-2_2_0_RELEASE.zip|TODO"
|
|
"mbedtls|v3.6.2|https://github.com/Mbed-TLS/mbedtls/releases/download/mbedtls-3.6.2/mbedtls-3.6.2.tar.bz2|TODO"
|
|
)
|
|
|
|
# ── 헬퍼: 다운로드 ───────────────────────────────────────────────────────────
|
|
download() { # url out
|
|
local url="$1" out="$2"
|
|
if command -v curl >/dev/null 2>&1; then
|
|
curl -fL --proto '=https' --tlsv1.2 -o "${out}" "${url}"
|
|
elif command -v wget >/dev/null 2>&1; then
|
|
wget --https-only -O "${out}" "${url}"
|
|
else
|
|
echo "ERROR: curl 또는 wget 이 필요합니다." >&2; exit 1
|
|
fi
|
|
}
|
|
|
|
# ── 헬퍼: SHA-256 ────────────────────────────────────────────────────────────
|
|
sha256_of() { # path -> stdout(소문자 hex)
|
|
local p="$1"
|
|
if command -v sha256sum >/dev/null 2>&1; then
|
|
sha256sum "${p}" | awk '{print tolower($1)}'
|
|
elif command -v shasum >/dev/null 2>&1; then
|
|
shasum -a 256 "${p}" | awk '{print tolower($1)}'
|
|
else
|
|
echo "ERROR: sha256sum 또는 shasum 이 필요합니다." >&2; exit 1
|
|
fi
|
|
}
|
|
|
|
# ── 헬퍼: 압축 해제(zip / tar.*) ─────────────────────────────────────────────
|
|
extract() { # archive dest
|
|
local archive="$1" dest="$2"
|
|
mkdir -p "${dest}"
|
|
case "${archive}" in
|
|
*.zip)
|
|
command -v unzip >/dev/null 2>&1 || { echo "ERROR: unzip 필요" >&2; exit 1; }
|
|
unzip -q "${archive}" -d "${dest}" ;;
|
|
*.tar.bz2|*.tbz2|*.tar.gz|*.tgz|*.tar.xz)
|
|
tar -xf "${archive}" -C "${dest}" ;;
|
|
*)
|
|
echo "ERROR: 지원하지 않는 압축 형식: ${archive}" >&2; exit 1 ;;
|
|
esac
|
|
}
|
|
|
|
for entry in "${DEPS[@]}"; do
|
|
IFS='|' read -r name tag url sha256 <<< "${entry}"
|
|
target="${THIRD_PARTY}/${name}"
|
|
|
|
echo "──────────────────────────────────────────────"
|
|
echo " 벤더링: ${name} (${tag})"
|
|
|
|
if [ -d "${target}" ] && [ "${FORCE}" != "1" ]; then
|
|
echo " 이미 존재 — 건너뜀(재벤더링은 FORCE=1). ${target}"
|
|
continue
|
|
fi
|
|
rm -rf "${target}"
|
|
|
|
tmp="$(mktemp -d)"
|
|
trap 'rm -rf "${tmp}"' EXIT
|
|
fname="$(basename "${url%%\?*}")"
|
|
archive="${tmp}/${fname}"
|
|
|
|
echo " 다운로드: ${url}"
|
|
download "${url}" "${archive}"
|
|
|
|
# ── 무결성 검증 ────────────────────────────────────────────────────────────
|
|
actual="$(sha256_of "${archive}")"
|
|
if [ "${sha256}" = "TODO" ] || [ -z "${sha256}" ]; then
|
|
echo " [경고] SHA-256 미설정 — 검증 생략. 계산값을 기록하세요:"
|
|
echo " ${name} : ${actual}"
|
|
elif [ "${actual}" != "$(echo "${sha256}" | tr 'A-Z' 'a-z')" ]; then
|
|
echo " ERROR: SHA-256 불일치(${name})" >&2
|
|
echo " 기대: ${sha256}" >&2
|
|
echo " 실제: ${actual}" >&2
|
|
exit 1
|
|
else
|
|
echo " SHA-256 확인됨: ${actual}"
|
|
fi
|
|
|
|
# ── 해제 + 최상위 단일 폴더 벗기기 ────────────────────────────────────────
|
|
ex="${tmp}/x"
|
|
extract "${archive}" "${ex}"
|
|
# 최상위가 단일 디렉터리면 그 내용을 target 으로, 아니면 ex 자체를 target 으로.
|
|
shopt -s nullglob
|
|
tops=("${ex}"/*)
|
|
shopt -u nullglob
|
|
if [ "${#tops[@]}" -eq 1 ] && [ -d "${tops[0]}" ]; then
|
|
mv "${tops[0]}" "${target}"
|
|
else
|
|
mv "${ex}" "${target}"
|
|
fi
|
|
|
|
if [ "${KEEP_ARCHIVES}" = "1" ]; then
|
|
cp -f "${archive}" "${ARCHIVE_DIR}/${fname}"
|
|
fi
|
|
|
|
rm -rf "${tmp}"
|
|
trap - EXIT
|
|
echo " 완료 → ${target}"
|
|
done
|
|
|
|
echo "──────────────────────────────────────────────"
|
|
echo " 벤더링 완료. 이제 폐쇄망 빌드 가능."
|
|
echo " TODO(vendor): 위 출력된 SHA-256 값을 DEPS 에 채워 재현성을 고정하세요."
|