2024-12-06 14:45:14 +03:00
#[ cfg(feature = " ssr " ) ]
2025-02-14 17:50:58 +03:00
use actix_web ::{ web , App , HttpServer , HttpResponse , Responder } ;
use std ::sync ::Arc ;
use tokio ::sync ::Mutex ;
use compareware ::db ::{ Database , DbItem } ;
use compareware ::api ::{ create_item , get_items , delete_item_by_url } ;
2025-02-13 23:07:27 +03:00
2024-12-06 14:45:14 +03:00
#[ actix_web::main ]
async fn main ( ) -> std ::io ::Result < ( ) > {
use actix_files ::Files ;
use actix_web ::* ;
use leptos ::* ;
use leptos_actix ::{ generate_route_list , LeptosRoutes } ;
use compareware ::app ::* ;
2025-01-22 20:16:43 +03:00
use compareware ::db ::{ Database , DbItem } ;
2025-02-14 17:50:58 +03:00
use compareware ::api ::{ get_items , create_item , delete_item , delete_property , delete_item_by_url , delete_property_by_url , create_item_by_url , get_items_by_url } ; // Import API handlers
2025-01-22 20:16:43 +03:00
use std ::sync ::Arc ;
use tokio ::sync ::Mutex ;
2024-12-06 14:45:14 +03:00
2025-01-22 14:14:18 +03:00
// Load configuration
2024-12-06 14:45:14 +03:00
let conf = get_configuration ( None ) . await . unwrap ( ) ;
let addr = conf . leptos_options . site_addr ;
2025-01-22 14:14:18 +03:00
2025-01-22 20:16:43 +03:00
// Initialize the database
let db = Database ::new ( " compareware.db " ) . unwrap ( ) ;
db . create_schema ( ) . await . unwrap ( ) ; // Ensure the schema is created
let db = Arc ::new ( Mutex ::new ( db ) ) ; // Wrap the database in an Arc<Mutex<T>> for shared state
2024-12-06 14:45:14 +03:00
// Generate the list of routes in your Leptos App
let routes = generate_route_list ( App ) ;
println! ( " listening on http:// {} " , & addr ) ;
2025-01-22 14:14:18 +03:00
// Start the Actix Web server
2024-12-06 14:45:14 +03:00
HttpServer ::new ( move | | {
let leptos_options = & conf . leptos_options ;
let site_root = & leptos_options . site_root ;
2025-01-22 20:16:43 +03:00
let db = db . clone ( ) ; // Clone the Arc for each worker
2024-12-06 14:45:14 +03:00
App ::new ( )
2025-02-14 17:50:58 +03:00
. app_data ( web ::Data ::new ( db . clone ( ) ) )
. route ( " /{url}/items " , web ::get ( ) . to ( get_items_handler ) )
. route ( " /{url}/items " , web ::post ( ) . to ( create_item_handler ) )
. route ( " /{url}/items/{item_id} " , web ::delete ( ) . to ( delete_item_handler ) )
2025-01-23 14:29:42 +03:00
// Register custom API routes BEFORE Leptos server functions
. service (
web ::scope ( " /api " )
2025-01-28 14:36:17 +03:00
. route ( " /items " , web ::get ( ) . to ( get_items ) ) // GET /api/items
. route ( " /items " , web ::post ( ) . to ( create_item ) ) // POST /api/items
. route ( " /items/{id} " , web ::delete ( ) . to ( delete_item ) ) // DELETE /api/items/{id}
. route ( " /properties/{property} " , web ::delete ( ) . to ( delete_property ) ) , // DELETE /api/properties/{property}
2025-01-23 14:29:42 +03:00
)
2025-01-22 14:14:18 +03:00
// Register server functions
. route ( " /api/{tail:.*} " , leptos_actix ::handle_server_fns ( ) )
// Serve JS/WASM/CSS from `pkg`
2024-12-06 14:45:14 +03:00
. service ( Files ::new ( " /pkg " , format! ( " {site_root} /pkg " ) ) )
2025-01-22 14:14:18 +03:00
// Serve other assets from the `assets` directory
2024-12-06 14:45:14 +03:00
. service ( Files ::new ( " /assets " , site_root ) )
2025-01-22 14:14:18 +03:00
// Serve the favicon from /favicon.ico
2024-12-06 14:45:14 +03:00
. service ( favicon )
2025-01-22 14:14:18 +03:00
// Register Leptos routes
2024-12-06 14:45:14 +03:00
. leptos_routes ( leptos_options . to_owned ( ) , routes . to_owned ( ) , App )
2025-01-22 14:14:18 +03:00
// Pass Leptos options to the app
2024-12-06 14:45:14 +03:00
. app_data ( web ::Data ::new ( leptos_options . to_owned ( ) ) )
2025-01-23 14:29:42 +03:00
//.wrap(middleware::Compress::default())
2025-01-22 20:16:43 +03:00
// Pass the database as shared state
. app_data ( web ::Data ::new ( db ) )
2025-02-13 23:07:27 +03:00
// Register URL routing
. service ( web ::resource ( " / " ) . route ( web ::get ( ) . to ( index ) ) )
. service ( web ::resource ( " /{url} " ) . route ( web ::get ( ) . to ( url_handler ) ) )
2024-12-06 14:45:14 +03:00
} )
. bind ( & addr ) ?
. run ( )
. await
}
2025-02-14 17:50:58 +03:00
// Handler to get items for a specific URL
async fn get_items_handler (
db : web ::Data < Arc < Mutex < Database > > > ,
url : web ::Path < String > ,
) -> impl Responder {
get_items ( db , web ::Query ( url . into_inner ( ) ) ) . await
}
// Handler to create an item for a specific URL
async fn create_item_handler (
db : web ::Data < Arc < Mutex < Database > > > ,
url : web ::Path < String > ,
item : web ::Json < DbItem > ,
) -> impl Responder {
create_item ( db , web ::Query ( url . into_inner ( ) ) , item ) . await
}
// Handler to delete an item for a specific URL
async fn delete_item_handler (
db : web ::Data < Arc < Mutex < Database > > > ,
path : web ::Path < ( String , String ) > ,
) -> impl Responder {
let ( url , item_id ) = path . into_inner ( ) ;
delete_item_by_url ( db , web ::Path ::from ( url ) , web ::Path ::from ( item_id ) ) . await
}
2025-02-13 23:07:27 +03:00
#[ cfg(feature = " ssr " ) ]
// Define the index handler
async fn index ( ) -> HttpResponse {
HttpResponse ::Ok ( ) . body ( " Welcome to CompareWare! " )
}
#[ cfg(feature = " ssr " ) ]
// Define the URL handler
async fn url_handler ( url : web ::Path < String > ) -> HttpResponse {
let url = url . into_inner ( ) ;
// TO DO: Implement URL-based content storage and editing functionality
HttpResponse ::Ok ( ) . body ( format! ( " You are viewing the content at {} " , url ) )
}
2024-12-06 14:45:14 +03:00
#[ cfg(feature = " ssr " ) ]
#[ actix_web::get( " favicon.ico " ) ]
async fn favicon (
leptos_options : actix_web ::web ::Data < leptos ::LeptosOptions > ,
) -> actix_web ::Result < actix_files ::NamedFile > {
let leptos_options = leptos_options . into_inner ( ) ;
let site_root = & leptos_options . site_root ;
Ok ( actix_files ::NamedFile ::open ( format! (
" {site_root}/favicon.ico "
) ) ? )
}
#[ cfg(not(any(feature = " ssr " , feature = " csr " ))) ]
pub fn main ( ) {
// no client-side main function
// unless we want this to work with e.g., Trunk for pure client-side testing
// see lib.rs for hydration function instead
// see optional feature `csr` instead
}
#[ cfg(all(not(feature = " ssr " ), feature = " csr " )) ]
pub fn main ( ) {
// a client-side main function is required for using `trunk serve`
// prefer using `cargo leptos serve` instead
// to run: `trunk serve --open --features csr`
use compareware ::app ::* ;
console_error_panic_hook ::set_once ( ) ;
leptos ::mount_to_body ( App ) ;
2025-01-22 14:14:18 +03:00
}