2018-02-10 00:00:55 +00:00
use std ::path ::{ Path , PathBuf } ;
2021-11-07 17:53:39 +00:00
use rocket ::serde ::json ::Json ;
use rocket ::{ fs ::NamedFile , http ::ContentType , Route } ;
2018-10-10 18:40:39 +00:00
use serde_json ::Value ;
2018-02-10 00:00:55 +00:00
2021-07-15 19:52:17 +00:00
use crate ::{
2022-04-24 06:47:49 +00:00
api ::core ::now ,
2021-07-15 19:52:17 +00:00
error ::Error ,
util ::{ Cached , SafeString } ,
CONFIG ,
} ;
2018-02-10 00:00:55 +00:00
pub fn routes ( ) -> Vec < Route > {
2019-12-27 17:37:14 +00:00
// If addding more routes here, consider also adding them to
2019-12-06 21:19:07 +00:00
// crate::utils::LOGGED_ROUTES to make sure they appear in the log
2019-01-25 17:23:51 +00:00
if CONFIG . web_vault_enabled ( ) {
2021-06-25 18:33:51 +00:00
routes! [ web_index , app_id , web_files , attachments , alive , static_files ]
2018-06-12 19:09:42 +00:00
} else {
2019-08-31 15:25:31 +00:00
routes! [ attachments , alive , static_files ]
2018-06-12 19:09:42 +00:00
}
2018-02-10 00:00:55 +00:00
}
#[ get( " / " ) ]
2021-11-07 17:53:39 +00:00
async fn web_index ( ) -> Cached < Option < NamedFile > > {
Cached ::short ( NamedFile ::open ( Path ::new ( & CONFIG . web_vault_folder ( ) ) . join ( " index.html " ) ) . await . ok ( ) , false )
2018-02-10 00:00:55 +00:00
}
2018-07-12 19:46:50 +00:00
#[ get( " /app-id.json " ) ]
2021-11-07 17:53:39 +00:00
fn app_id ( ) -> Cached < ( ContentType , Json < Value > ) > {
2018-07-13 13:05:00 +00:00
let content_type = ContentType ::new ( " application " , " fido.trusted-apps+json " ) ;
2021-12-28 16:24:42 +00:00
Cached ::long (
2021-11-07 17:53:39 +00:00
(
2021-12-28 16:24:42 +00:00
content_type ,
Json ( json! ( {
" trustedFacets " : [
{
" version " : { " major " : 1 , " minor " : 0 } ,
" ids " : [
// Per <https://fidoalliance.org/specs/fido-v2.0-id-20180227/fido-appid-and-facets-v2.0-id-20180227.html#determining-the-facetid-of-a-calling-application>:
//
// "In the Web case, the FacetID MUST be the Web Origin [RFC6454]
// of the web page triggering the FIDO operation, written as
// a URI with an empty path. Default ports are omitted and any
// path component is ignored."
//
// This leaves it unclear as to whether the path must be empty,
// or whether it can be non-empty and will be ignored. To be on
// the safe side, use a proper web origin (with empty path).
& CONFIG . domain_origin ( ) ,
" ios:bundle-id:com.8bit.bitwarden " ,
" android:apk-key-hash:dUGFzUzf3lmHSLBDBIv+WaFyZMI " ]
} ]
} ) ) ,
) ,
true ,
)
2018-07-12 19:46:50 +00:00
}
2019-01-19 20:36:34 +00:00
#[ get( " /<p..> " , rank = 10) ] // Only match this if the other routes don't match
2021-11-07 17:53:39 +00:00
async fn web_files ( p : PathBuf ) -> Cached < Option < NamedFile > > {
Cached ::long ( NamedFile ::open ( Path ::new ( & CONFIG . web_vault_folder ( ) ) . join ( p ) ) . await . ok ( ) , true )
2018-02-10 00:00:55 +00:00
}
2021-05-16 05:46:57 +00:00
#[ get( " /attachments/<uuid>/<file_id> " ) ]
2021-11-07 17:53:39 +00:00
async fn attachments ( uuid : SafeString , file_id : SafeString ) -> Option < NamedFile > {
NamedFile ::open ( Path ::new ( & CONFIG . attachments_folder ( ) ) . join ( uuid ) . join ( file_id ) ) . await . ok ( )
2018-02-10 00:00:55 +00:00
}
2021-10-09 12:16:27 +00:00
// We use DbConn here to let the alive healthcheck also verify the database connection.
use crate ::db ::DbConn ;
2018-02-10 00:00:55 +00:00
#[ get( " /alive " ) ]
2021-10-09 12:16:27 +00:00
fn alive ( _conn : DbConn ) -> Json < String > {
2022-04-24 06:47:49 +00:00
now ( )
2018-02-10 00:00:55 +00:00
}
2019-02-16 02:44:30 +00:00
2022-01-23 22:40:59 +00:00
#[ get( " /vw_static/<filename> " ) ]
2021-11-07 17:53:39 +00:00
fn static_files ( filename : String ) -> Result < ( ContentType , & 'static [ u8 ] ) , Error > {
2019-02-16 02:44:30 +00:00
match filename . as_ref ( ) {
2021-11-07 17:53:39 +00:00
" mail-github.png " = > Ok ( ( ContentType ::PNG , include_bytes! ( " ../static/images/mail-github.png " ) ) ) ,
" logo-gray.png " = > Ok ( ( ContentType ::PNG , include_bytes! ( " ../static/images/logo-gray.png " ) ) ) ,
" error-x.svg " = > Ok ( ( ContentType ::SVG , include_bytes! ( " ../static/images/error-x.svg " ) ) ) ,
" hibp.png " = > Ok ( ( ContentType ::PNG , include_bytes! ( " ../static/images/hibp.png " ) ) ) ,
" vaultwarden-icon.png " = > Ok ( ( ContentType ::PNG , include_bytes! ( " ../static/images/vaultwarden-icon.png " ) ) ) ,
" bootstrap.css " = > Ok ( ( ContentType ::CSS , include_bytes! ( " ../static/scripts/bootstrap.css " ) ) ) ,
" bootstrap-native.js " = > Ok ( ( ContentType ::JavaScript , include_bytes! ( " ../static/scripts/bootstrap-native.js " ) ) ) ,
" identicon.js " = > Ok ( ( ContentType ::JavaScript , include_bytes! ( " ../static/scripts/identicon.js " ) ) ) ,
" datatables.js " = > Ok ( ( ContentType ::JavaScript , include_bytes! ( " ../static/scripts/datatables.js " ) ) ) ,
" datatables.css " = > Ok ( ( ContentType ::CSS , include_bytes! ( " ../static/scripts/datatables.css " ) ) ) ,
2021-06-19 17:22:19 +00:00
" jquery-3.6.0.slim.js " = > {
2021-11-07 17:53:39 +00:00
Ok ( ( ContentType ::JavaScript , include_bytes! ( " ../static/scripts/jquery-3.6.0.slim.js " ) ) )
2021-04-06 20:54:42 +00:00
}
2020-02-19 05:27:00 +00:00
_ = > err! ( format! ( " Static file not found: {} " , filename ) ) ,
2019-02-16 02:44:30 +00:00
}
2019-12-27 17:37:14 +00:00
}