use crate::burnrl::environment::TrictracEnvironment; use crate::burnrl::utils::{soft_update_linear, Config}; use burn::backend::{ndarray::NdArrayDevice, NdArray}; use burn::module::Module; use burn::nn::{Linear, LinearConfig}; use burn::optim::AdamWConfig; use burn::record::{CompactRecorder, Recorder}; use burn::tensor::activation::{relu, softmax}; use burn::tensor::backend::{AutodiffBackend, Backend}; use burn::tensor::Tensor; use burn_rl::agent::{SACActor, SACCritic, SACNets, SACOptimizer, SACTrainingConfig, SAC}; use burn_rl::base::{Action, Agent, ElemType, Environment, Memory, Model, State}; use std::time::SystemTime; #[derive(Module, Debug)] pub struct Actor { linear_0: Linear, linear_1: Linear, linear_2: Linear, } impl Actor { pub fn new(input_size: usize, dense_size: usize, output_size: usize) -> Self { Self { linear_0: LinearConfig::new(input_size, dense_size).init(&Default::default()), linear_1: LinearConfig::new(dense_size, dense_size).init(&Default::default()), linear_2: LinearConfig::new(dense_size, output_size).init(&Default::default()), } } } impl Model, Tensor> for Actor { fn forward(&self, input: Tensor) -> Tensor { let layer_0_output = relu(self.linear_0.forward(input)); let layer_1_output = relu(self.linear_1.forward(layer_0_output)); softmax(self.linear_2.forward(layer_1_output), 1) } fn infer(&self, input: Tensor) -> Tensor { self.forward(input) } } impl SACActor for Actor {} #[derive(Module, Debug)] pub struct Critic { linear_0: Linear, linear_1: Linear, linear_2: Linear, } impl Critic { pub fn new(input_size: usize, dense_size: usize, output_size: usize) -> Self { Self { linear_0: LinearConfig::new(input_size, dense_size).init(&Default::default()), linear_1: LinearConfig::new(dense_size, dense_size).init(&Default::default()), linear_2: LinearConfig::new(dense_size, output_size).init(&Default::default()), } } fn consume(self) -> (Linear, Linear, Linear) { (self.linear_0, self.linear_1, self.linear_2) } } impl Model, Tensor> for Critic { fn forward(&self, input: Tensor) -> Tensor { let layer_0_output = relu(self.linear_0.forward(input)); let layer_1_output = relu(self.linear_1.forward(layer_0_output)); self.linear_2.forward(layer_1_output) } fn infer(&self, input: Tensor) -> Tensor { self.forward(input) } } impl SACCritic for Critic { fn soft_update(this: Self, that: &Self, tau: ElemType) -> Self { let (linear_0, linear_1, linear_2) = this.consume(); Self { linear_0: soft_update_linear(linear_0, &that.linear_0, tau), linear_1: soft_update_linear(linear_1, &that.linear_1, tau), linear_2: soft_update_linear(linear_2, &that.linear_2, tau), } } } #[allow(unused)] const MEMORY_SIZE: usize = 4096; type MyAgent = SAC>; #[allow(unused)] pub fn run< E: Environment + AsMut, B: AutodiffBackend, >( conf: &Config, visualized: bool, ) -> impl Agent { let mut env = E::new(visualized); env.as_mut().max_steps = conf.max_steps; let state_dim = <::StateType as State>::size(); let action_dim = <::ActionType as Action>::size(); let actor = Actor::::new(state_dim, conf.dense_size, action_dim); let critic_1 = Critic::::new(state_dim, conf.dense_size, action_dim); let critic_2 = Critic::::new(state_dim, conf.dense_size, action_dim); let mut nets = SACNets::, Critic>::new(actor, critic_1, critic_2); let mut agent = MyAgent::default(); let config = SACTrainingConfig { gamma: conf.gamma, tau: conf.tau, learning_rate: conf.learning_rate, min_probability: conf.min_probability, batch_size: conf.batch_size, clip_grad: Some(burn::grad_clipping::GradientClippingConfig::Value( conf.clip_grad, )), }; let mut memory = Memory::::default(); let optimizer_config = AdamWConfig::new().with_grad_clipping(config.clip_grad.clone()); let mut optimizer = SACOptimizer::new( optimizer_config.clone().init(), optimizer_config.clone().init(), optimizer_config.clone().init(), optimizer_config.init(), ); let mut step = 0_usize; for episode in 0..conf.num_episodes { let mut episode_done = false; let mut episode_reward = 0.0; let mut episode_duration = 0_usize; let mut state = env.state(); let mut now = SystemTime::now(); while !episode_done { if let Some(action) = MyAgent::::react_with_model(&state, &nets.actor) { let snapshot = env.step(action); episode_reward += <::RewardType as Into>::into( snapshot.reward().clone(), ); memory.push( state, *snapshot.state(), action, snapshot.reward().clone(), snapshot.done(), ); if config.batch_size < memory.len() { nets = agent.train::(nets, &memory, &mut optimizer, &config); } step += 1; episode_duration += 1; if snapshot.done() || episode_duration >= conf.max_steps { env.reset(); episode_done = true; println!( "{{\"episode\": {episode}, \"reward\": {episode_reward:.4}, \"steps count\": {episode_duration}, \"duration\": {}}}", now.elapsed().unwrap().as_secs() ); now = SystemTime::now(); } else { state = *snapshot.state(); } } } } let valid_agent = agent.valid(nets.actor); if let Some(path) = &conf.save_path { if let Some(model) = valid_agent.model() { save_model(model, path); } } valid_agent } pub fn save_model(model: &Actor>, path: &String) { let recorder = CompactRecorder::new(); let model_path = format!("{path}.mpk"); println!("info: Modèle de validation sauvegardé : {model_path}"); recorder .record(model.clone().into_record(), model_path.into()) .unwrap(); } pub fn load_model(dense_size: usize, path: &String) -> Option>> { let model_path = format!("{path}.mpk"); // println!("Chargement du modèle depuis : {model_path}"); CompactRecorder::new() .load(model_path.into(), &NdArrayDevice::default()) .map(|record| { Actor::new( ::StateType::size(), dense_size, ::ActionType::size(), ) .load_record(record) }) .ok() }