2023-08-04 12:52:05 -05:00
// This file is part of Moonfire NVR, a security camera network video recorder.
// Copyright (C) 2023 The Moonfire NVR Authors; see AUTHORS and LICENSE.txt.
// SPDX-License-Identifier: GPL-v3.0-or-later WITH GPL-3.0-linking-exception
//! Build script to bundle UI files if `bundled-ui` Cargo feature is selected.
use std ::fmt ::Write ;
use std ::path ::{ Path , PathBuf } ;
2023-10-14 15:52:06 -07:00
use std ::process ::Command ;
2023-08-04 12:52:05 -05:00
2023-12-17 18:07:25 -08:00
const UI_DIST_DIR_ENV_VAR : & str = " UI_DIST_DIR " ;
const DEFAULT_UI_DIST_DIR : & str = " ../ui/dist " ;
2023-10-14 15:52:06 -07:00
type BoxError = Box < dyn std ::error ::Error + Send + Sync + 'static > ;
2023-08-04 12:52:05 -05:00
fn ensure_link ( original : & Path , link : & Path ) {
match std ::fs ::read_link ( link ) {
Ok ( dst ) if dst = = original = > return ,
2023-10-14 15:52:06 -07:00
Ok ( _ ) = > std ::fs ::remove_file ( link ) . expect ( " removing stale symlink should succeed " ) ,
2023-08-04 12:52:05 -05:00
Err ( e ) if e . kind ( ) ! = std ::io ::ErrorKind ::NotFound = > {
panic! ( " couldn't create link {link:?} to original path {original:?} : {e} " )
}
_ = > { }
}
std ::os ::unix ::fs ::symlink ( original , link ) . expect ( " symlink creation should succeed " ) ;
}
struct File {
/// Path with `ui_files/` prefix and the encoding suffix; suitable for
/// passing to `include_bytes!` in the expanded code.
///
/// E.g. `ui_files/index.html.gz`.
include_path : String ,
encoding : FileEncoding ,
etag : blake3 ::Hash ,
}
#[ derive(Copy, Clone) ]
enum FileEncoding {
Uncompressed ,
Gzipped ,
}
impl FileEncoding {
fn to_str ( self ) -> & 'static str {
match self {
Self ::Uncompressed = > " FileEncoding::Uncompressed " ,
Self ::Gzipped = > " FileEncoding::Gzipped " ,
}
}
}
/// Map of "bare path" to the best representation.
///
/// A "bare path" has no prefix for the root and no suffix for encoding, e.g.
2023-12-17 18:07:25 -08:00
/// `favicons/blah.ico` rather than `../../ui/dist/favicons/blah.ico.gz`.
2023-08-04 12:52:05 -05:00
///
/// The best representation is gzipped if available, uncompressed otherwise.
2024-01-06 10:43:20 -08:00
type FileMap = std ::collections ::HashMap < String , File , ahash ::RandomState > ;
2023-08-04 12:52:05 -05:00
fn stringify_files ( files : & FileMap ) -> Result < String , std ::fmt ::Error > {
let mut buf = String ::new ( ) ;
2024-02-12 17:32:37 -08:00
writeln! ( buf , " const FILES: [BuildFile; {}] = [ " , files . len ( ) ) ? ;
2023-08-04 12:52:05 -05:00
for ( bare_path , file ) in files {
let include_path = & file . include_path ;
let etag = file . etag . to_hex ( ) ;
let encoding = file . encoding . to_str ( ) ;
2024-02-12 17:32:37 -08:00
writeln! ( buf , " BuildFile {{ bare_path: {bare_path:?}, data: include_bytes!({include_path:?}), etag: {etag:?}, encoding: {encoding} }}, " ) ? ;
2023-08-04 12:52:05 -05:00
}
2024-02-12 17:32:37 -08:00
writeln! ( buf , " ]; " ) ? ;
2023-08-04 12:52:05 -05:00
Ok ( buf )
}
2023-10-14 15:52:06 -07:00
fn handle_bundled_ui ( ) -> Result < ( ) , BoxError > {
2023-08-04 12:52:05 -05:00
// Nothing to do if the feature is off. cargo will re-run if features change.
if ! cfg! ( feature = " bundled-ui " ) {
2023-10-14 15:52:06 -07:00
return Ok ( ( ) ) ;
2023-08-04 12:52:05 -05:00
}
2023-10-14 15:52:06 -07:00
let ui_dir =
2023-12-17 18:07:25 -08:00
std ::env ::var ( UI_DIST_DIR_ENV_VAR ) . unwrap_or_else ( | _ | DEFAULT_UI_DIST_DIR . to_owned ( ) ) ;
2023-10-14 15:52:06 -07:00
2023-08-04 12:52:05 -05:00
// If the feature is on, also re-run if the actual UI files change.
2023-12-17 18:07:25 -08:00
println! ( " cargo:rerun-if-env-changed= {UI_DIST_DIR_ENV_VAR} " ) ;
2023-10-14 15:52:06 -07:00
println! ( " cargo:rerun-if-changed= {ui_dir} " ) ;
2023-08-04 12:52:05 -05:00
let out_dir : PathBuf = std ::env ::var_os ( " OUT_DIR " )
. expect ( " cargo should set OUT_DIR " )
. into ( ) ;
2023-10-14 15:52:06 -07:00
let abs_ui_dir = std ::fs ::canonicalize ( & ui_dir ) . map_err ( | e | format! ( " ui dir {ui_dir:?} should be accessible. Did you run `npm run build` first? \n \n caused by: \n {e} " ) ) ? ;
2023-08-04 12:52:05 -05:00
let mut files = FileMap ::default ( ) ;
for entry in walkdir ::WalkDir ::new ( & abs_ui_dir ) {
2023-10-14 15:52:06 -07:00
let entry = entry . map_err ( | e | {
format! ( " walkdir failed. Did you run `npm run build` first? \n \n caused by: \n {e} " )
} ) ? ;
2023-08-04 12:52:05 -05:00
if ! entry . file_type ( ) . is_file ( ) {
continue ;
}
let path = entry
. path ( )
. strip_prefix ( & abs_ui_dir )
. expect ( " walkdir should return root-prefixed entries " ) ;
let path = path . to_str ( ) . expect ( " ui file paths should be valid UTF-8 " ) ;
let ( bare_path , encoding ) ;
match path . strip_suffix ( " .gz " ) {
Some ( p ) = > {
bare_path = p ;
encoding = FileEncoding ::Gzipped ;
}
None = > {
bare_path = path ;
encoding = FileEncoding ::Uncompressed ;
if files . get ( bare_path ) . is_some ( ) {
continue ; // don't replace with suboptimal encoding.
}
}
}
let contents = std ::fs ::read ( entry . path ( ) ) . expect ( " ui files should be readable " ) ;
let etag = blake3 ::hash ( & contents ) ;
let include_path = format! ( " ui_files/ {path} " ) ;
files . insert (
bare_path . to_owned ( ) ,
File {
include_path ,
encoding ,
etag ,
} ,
) ;
}
2023-10-14 15:52:06 -07:00
if ! files . contains_key ( " index.html " ) {
return Err ( format! (
" No `index.html` within {ui_dir:?}. Did you run `npm run build` first? "
)
. into ( ) ) ;
}
2023-08-04 12:52:05 -05:00
let files = stringify_files ( & files ) . expect ( " write to String should succeed " ) ;
let mut out_rs_path = std ::path ::PathBuf ::new ( ) ;
out_rs_path . push ( & out_dir ) ;
out_rs_path . push ( " ui_files.rs " ) ;
std ::fs ::write ( & out_rs_path , files ) . expect ( " writing ui_files.rs should succeed " ) ;
let mut out_link_path = std ::path ::PathBuf ::new ( ) ;
out_link_path . push ( & out_dir ) ;
out_link_path . push ( " ui_files " ) ;
ensure_link ( & abs_ui_dir , & out_link_path ) ;
2023-10-14 15:52:06 -07:00
Ok ( ( ) )
}
fn handle_version ( ) -> Result < ( ) , BoxError > {
println! ( " cargo:rerun-if-env-changed=VERSION " ) ;
if std ::env ::var ( " VERSION " ) . is_ok ( ) {
return Ok ( ( ) ) ;
}
// Get version from `git describe`. Inspired by the `git-version` crate.
// We don't use that directly because `cross`'s default docker image doesn't install `git`,
// and thus we need the environment variable pass-through above.
// Avoid reruns when the output doesn't meaningfully change. I don't think this is quite right:
// it won't recognize toggling between `-dirty` and not. But it'll do.
let dir = Command ::new ( " git " )
. arg ( " rev-parse " )
. arg ( " --git-dir " )
. output ( ) ?
. stdout ;
let dir = String ::from_utf8 ( dir ) . unwrap ( ) ;
let dir = dir . strip_suffix ( '\n' ) . unwrap ( ) ;
println! ( " cargo:rerun-if-changed= {dir} /logs/HEAD " ) ;
println! ( " cargo:rerun-if-changed= {dir} /index " ) ;
// Plumb the version through.
let version = Command ::new ( " git " )
. arg ( " describe " )
. arg ( " --always " )
. arg ( " --dirty " )
. output ( ) ?
. stdout ;
let version = String ::from_utf8 ( version ) . unwrap ( ) ;
let version = version . strip_suffix ( '\n' ) . unwrap ( ) ;
println! ( " cargo:rustc-env=VERSION= {version} " ) ;
Ok ( ( ) )
}
fn main ( ) -> Result < ( ) , BoxError > {
// Explicitly declare dependencies, so this doesn't re-run if other source files change.
println! ( " cargo:rerun-if-changed=build.rs " ) ;
handle_bundled_ui ( ) ? ;
handle_version ( ) ? ;
Ok ( ( ) )
2023-08-04 12:52:05 -05:00
}