SimulacionController.cs 24KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Threading.Tasks;
  5. using Microsoft.AspNetCore.Http;
  6. using Microsoft.AspNetCore.Mvc;
  7. using Microsoft.EntityFrameworkCore;
  8. using WebApplication3.Models;
  9. using WebApplication3.Clases;
  10. using System.IO;
  11. using System.IO.Compression;
  12. using System.Threading;
  13. using Newtonsoft.Json;
  14. namespace WebApplication3.Controllers
  15. {
  16. [Route("api/[controller]")]
  17. [ApiController]
  18. public class SimulacionController : ControllerBase
  19. {
  20. private readonly AquiferContext _context;
  21. private readonly PozoController pozoController;
  22. private readonly RecargasController recargasController;
  23. private readonly AcuiferoController acuiferoController;
  24. private readonly ServicioController serviceController;
  25. private readonly BalanceHidricoController balanceHidricoController;
  26. private readonly PiezometriaController piezometriaController;
  27. private readonly ServiceQueueController serviceQueueController;
  28. public SimulacionController(AquiferContext context)
  29. {
  30. _context = context;
  31. pozoController = new PozoController(context);
  32. recargasController = new RecargasController(context);
  33. acuiferoController = new AcuiferoController(context);
  34. serviceController = new ServicioController(context);
  35. balanceHidricoController = new BalanceHidricoController(context);
  36. piezometriaController= new PiezometriaController(context);
  37. serviceQueueController = new ServiceQueueController(context);
  38. }
  39. // GET: api/Simulacion
  40. [HttpGet]
  41. public async Task<ActionResult<IEnumerable<Simulacion>>> GetSimulaciones()
  42. {
  43. return await _context.Simulaciones.ToListAsync();
  44. }
  45. // GET: api/Simulacion/5
  46. [HttpGet("{id}")]
  47. [ProducesResponseType(200)]
  48. [ProducesResponseType(404)]
  49. public async Task<ActionResult<Simulacion>> GetSimulacion(uint id)
  50. {
  51. var simulacion = await _context.Simulaciones.FindAsync(id);
  52. if (simulacion == null)
  53. {
  54. return NotFound();
  55. }
  56. return simulacion;
  57. }
  58. // POST: api/Simulacion
  59. // To protect from overposting attacks, see https://go.microsoft.com/fwlink/?linkid=2123754
  60. [HttpPost]
  61. [ProducesResponseType(201)]
  62. public async Task<ActionResult<Simulacion>> PostSimulacion(Simulacion simulacion)
  63. {
  64. _context.Simulaciones.Add(simulacion);
  65. await _context.SaveChangesAsync();
  66. //prepacion de los valores por defecto para la simulación
  67. pozoController.SetPozosSimulacion(simulacion.IdAcuifero,simulacion.Id);
  68. recargasController.SetRecargasSimulacion(simulacion.IdAcuifero, simulacion.Id);
  69. return CreatedAtAction("GetSimulacion", new { id = simulacion.Id }, simulacion);
  70. }
  71. [HttpPost("Simula/{simulacionId}")]
  72. [ProducesResponseType(200)]
  73. public async Task<bool> QueueSim(uint simulacionId)
  74. {
  75. Simulacion simulacion = _context.Simulaciones.Find(simulacionId);
  76. ServiceQueue serviceQueue = new ();
  77. serviceQueue.IdSimulacion = simulacionId;
  78. serviceQueue.IdAcuifero = simulacion.IdAcuifero;
  79. await serviceQueueController.PostServiceQueue(serviceQueue);
  80. return true;
  81. }
  82. [HttpPut("{simulationId}/UpdateSimWellData")]
  83. [ProducesResponseType(200)]
  84. public async Task<ActionResult<bool>> UpdateSimWellsFromCsv(uint simulationId, [FromBody] DatosPostSimulacion data)
  85. {
  86. bool result = true;
  87. Parser parser = new();
  88. Simulacion sim = _context.Simulaciones.Find(simulationId);
  89. string file = Constants.PathExtract + data.Fichero.Nombre;
  90. //1. Se eliminan los pozos con sus respectivas time series de la bbdd ya que se va a sobreescribir la informacion
  91. await pozoController.DeletePozoFromSim(simulationId);
  92. //2. Guardamos el fichero que nos viene en el body en formato base64
  93. try
  94. {
  95. byte[] tempBytes = Convert.FromBase64String(data.Fichero.Base64);
  96. System.IO.File.WriteAllBytes( file, tempBytes);
  97. }
  98. catch (Exception ex)
  99. {
  100. return Problem(ex.Message);
  101. }
  102. WellData[] wellDatas = parser.parseWellCSV(file);
  103. await pozoController.PostMultipleWells((uint)sim.IdAcuifero,wellDatas, simulationId);
  104. return result;
  105. }
  106. [HttpPut("{simulationId}/UpdateRechargeWellData")]
  107. [ProducesResponseType(200)]
  108. public async Task<ActionResult<bool>> UpdateSimRechargesFromCsv(uint simulationId, [FromBody] DatosPostSimulacion data)
  109. {
  110. bool result = true;
  111. Parser parser = new();
  112. Simulacion sim = _context.Simulaciones.Find(simulationId);
  113. string file = Constants.PathExtract + data.Fichero.Nombre;
  114. //1. Se eliminan los pozos con sus respectivas time series de la bbdd ya que se va a sobreescribir la informacion
  115. await pozoController.DeletePozoFromSim(simulationId);
  116. //2. Guardamos el fichero que nos viene en el body en formato base64
  117. try
  118. {
  119. byte[] tempBytes = Convert.FromBase64String(data.Fichero.Base64);
  120. System.IO.File.WriteAllBytes(file, tempBytes);
  121. }
  122. catch (Exception ex)
  123. {
  124. return Problem(ex.Message);
  125. }
  126. RechargeData[] rechargeDatas = parser.parseRechargeCSV(file);
  127. await recargasController.PostMultipleRecharges((uint)sim.IdAcuifero, rechargeDatas,sim.Id);
  128. return result;
  129. }
  130. // PUT: api/Simulacion/5
  131. [HttpPut("{id}")]
  132. [ProducesResponseType(201)]
  133. [ProducesResponseType(400)]
  134. [ProducesResponseType(404)]
  135. public async Task<IActionResult> PutSimulacion(uint id, [FromBody] DatosPostSimulacion datos)
  136. {
  137. Simulacion simulacion = _context.Simulaciones.Find(id);
  138. if (id != simulacion.Id)
  139. {
  140. return BadRequest();
  141. }
  142. simulacion.Nombre = (datos.Nombre != null && datos.Nombre != simulacion.Nombre) ? datos.Nombre : simulacion.Nombre;
  143. _context.Entry(simulacion).State = EntityState.Modified;
  144. try
  145. {
  146. await _context.SaveChangesAsync();
  147. }
  148. catch (DbUpdateConcurrencyException)
  149. {
  150. if (!SimulacionExists(id))
  151. {
  152. return NotFound();
  153. }
  154. else
  155. {
  156. throw;
  157. }
  158. }
  159. return CreatedAtAction("GetSimulacion", new { id = simulacion.Id }, simulacion);
  160. }
  161. // DELETE: api/Simulacion/5
  162. [HttpDelete("{id}")]
  163. [ProducesResponseType(209)]
  164. public async Task<IActionResult> DeleteSimulacion(uint id)
  165. {
  166. var simulacion = await _context.Simulaciones.FindAsync(id);
  167. if (simulacion == null)
  168. {
  169. return NotFound();
  170. }
  171. _context.Simulaciones.Remove(simulacion);
  172. await _context.SaveChangesAsync();
  173. return NoContent();
  174. }
  175. [NonAction]
  176. public async Task<bool> SimulaAsync(ServiceQueue serviceQueue)
  177. {
  178. await serviceQueueController.SetCorriendo(serviceQueue.Id);
  179. uint simulacionId = (uint)serviceQueue.IdSimulacion;
  180. Simulacion simulacion = _context.Simulaciones.Find(simulacionId);
  181. Acuifero aquifer = _context.Acuiferos.Find(simulacion.IdAcuifero);
  182. //Obtenemos el nombre de fichero original del acuifero
  183. string zip = acuiferoController.GetFicheroZip((uint)simulacion.IdAcuifero);
  184. Logger.WriteLog("estoy simulando");
  185. if (zip != string.Empty)
  186. {
  187. Parser parser = new ();
  188. string zip_name = "simulacion" + simulacion.Nombre;
  189. string zip_copy = zip_name + ".zip";
  190. System.IO.File.Copy(Constants.PathZip + zip, Constants.PathZip + zip_copy, true);
  191. string workingDirectory = Constants.PathExtract + zip_name + "\\";
  192. //Descomprimimos copia para escribir en fichero
  193. Directory.CreateDirectory(workingDirectory);
  194. ZipFile.ExtractToDirectory(Constants.PathZip + zip_copy, workingDirectory, true);
  195. workingDirectory += zip.Replace(".zip", "\\");
  196. //Obtenemos fichero de pozo y recargas
  197. List<string> files = Directory.GetFiles(workingDirectory).ToList<string>();
  198. string pozoFile = files.Find(x => x.Contains(".VMW"));
  199. string recargaFile = files.Find(x => x.Contains(".VMP"));
  200. string inFile = files.Find(x => x.Contains(".MODFLOW.IN") || x.Contains(".modflow.in"));
  201. string zoneFile = files.Find(x => x.Contains(".ZBI") || x.Contains(".zbi"));
  202. Logger.WriteLog("ficheros clave buscados");
  203. //3- Parseamos pozos
  204. if (System.IO.File.Exists(pozoFile))
  205. {
  206. Logger.WriteLog("vamos con los pozos");
  207. List<WellData> infoPozosFichero = parser.ParseWellData(pozoFile).ToList<WellData>();
  208. List<Pozo> pozos = pozoController.GetPozosSimulacion(simulacionId);
  209. List<string> lines = new();
  210. //Por ahora asumimos que el fichero tiene la misma cantidad de pozos que en bbdd para esta simulacion
  211. string[] maxDefault = infoPozosFichero[0].ZMax.Split(" ");
  212. lines.Add(pozos.Count.ToString());
  213. foreach (Pozo p in pozos)
  214. {
  215. List<TsPozo> timeSeriesPozo = pozoController.GetTsPozoFromPozo(p.Id);
  216. lines.Add("0"); //Activo o inactivo
  217. lines.Add(p.Nombre);
  218. lines.Add(p.X.ToString());
  219. lines.Add(p.Y.ToString());
  220. lines.Add(p.Minimo.ToString());
  221. maxDefault[0] = p.Maximo.ToString();
  222. string aux = String.Join(' ', maxDefault);
  223. lines.Add(aux); //Maximo
  224. lines.Add(infoPozosFichero[0].ScreenIntervals.ToString()); //screen intervals
  225. lines.Add(infoPozosFichero[0].Bot);
  226. lines.Add(infoPozosFichero[0].Top);
  227. lines.Add(timeSeriesPozo.Count.ToString());
  228. for (int i = 0; i < timeSeriesPozo.Count; i++)
  229. {
  230. lines.Add(timeSeriesPozo[i].MarcaTiempo);
  231. lines.Add(timeSeriesPozo[i].MarcaTiempoEnd);
  232. lines.Add(timeSeriesPozo[i].Valor + " " + timeSeriesPozo[i].InfoComplementaria);
  233. }
  234. }
  235. //System.IO.File.WriteAllLines(@"C:\Users\Admin\Desktop\Test.VMW", lines);
  236. System.IO.File.WriteAllLines(pozoFile, lines);
  237. Logger.WriteLog("pozos terminados");
  238. }
  239. //4- Parseamos recargas
  240. if (System.IO.File.Exists(recargaFile))
  241. {
  242. Logger.WriteLog("vamos con recargas");
  243. List<Recarga> recargas = recargasController.GetRecargasSimulacion(simulacionId);
  244. string text = System.IO.File.ReadAllText(recargaFile);
  245. string[] data = text.Split("\r\n\r\n");
  246. int layers = int.Parse(data[0][0].ToString());
  247. int blankLinePerBlock = layers + 1;
  248. int rechargeBlockInex = blankLinePerBlock * 2;
  249. List<string> lines = new();
  250. //Hasta el bloque de recargas vamos poniendo el texto que le precede
  251. for (int i = 0; i < rechargeBlockInex; i++)
  252. {
  253. lines.AddRange(data[i].Split("\r\n").ToList());
  254. lines.Add("");
  255. }
  256. //Bloque recargas
  257. lines.Add(recargas.Count.ToString());
  258. int index = 1;
  259. foreach (Recarga r in recargas)
  260. {
  261. List<TsRecarga> tsRecarga = recargasController.GetTsRecargaFromRecaga(r.Id);
  262. lines.Add(index + " 1 Index 1");
  263. lines.Add(tsRecarga.Count + " rchr entries");
  264. tsRecarga.ForEach(x =>
  265. lines.Add(index + " " + x.Valor + " " + x.MarcaTiempo + " " + x.MarcaTiempoEnd + " " + x.InfoComplementaria)
  266. );
  267. //lines.Add(index + " " + tsRecarga[0].Valor + " " + tsRecarga[0].MarcaTiempo + " " + tsRecarga[1].MarcaTiempo + " " + tsRecarga[0].InfoComplementaria);
  268. }
  269. lines.Add("");
  270. //Hasta el final
  271. for (int i = rechargeBlockInex + 1; i < data.Length; i++)
  272. {
  273. lines.AddRange(data[i].Split("\r\n").ToList());
  274. //controlamos que en la ultima iteracion no introduzca un \r\n
  275. if (i < data.Length - 1)
  276. {
  277. lines.Add("");
  278. }
  279. }
  280. //System.IO.File.WriteAllLines(@"C:\Users\Admin\Desktop\TestRec.VMP", lines);
  281. System.IO.File.WriteAllLines(recargaFile, lines);
  282. }
  283. //5.Ejecutamos el modelo
  284. //Primeramente, para ejecutar el modelo, hay que modificar el contenido del fichero con extension .in para que no de error al ejecutarse el modelo
  285. if (System.IO.File.Exists(inFile))
  286. {
  287. Logger.WriteLog("tratamos el fichero .IN");
  288. List<string> inFileLines = parser.ParseInFile(inFile);
  289. System.IO.File.WriteAllLines(inFile, inFileLines);
  290. }
  291. string inFileName = inFile.Split("\\").ToList().Last();
  292. Logger.WriteLog("ejecuto modflow");
  293. await serviceController.SimulaModflow(workingDirectory, inFileName);
  294. Thread.Sleep(5000);
  295. Logger.WriteLog("fin modflow");
  296. string modelResultsFile = files.Find(x => x.Contains(".LST"));
  297. if (parser.ModflowError(modelResultsFile))
  298. {
  299. Logger.WriteLog("El fichero LST para la simulacion con id interno "+simulacion.Id+" tiene error");
  300. await serviceQueueController.SetError(serviceQueue.Id);
  301. return true;
  302. }
  303. //6.Ejecucion Zonebudget
  304. //Seguidamente miramos el fichero de resultado de zonas (si existe)
  305. if (zoneFile != null && System.IO.File.Exists(zoneFile))
  306. {
  307. files = Directory.GetFiles(workingDirectory).ToList<string>();
  308. string budgetFile = files.Find(x => x.Contains(".BGT"));
  309. Logger.WriteLog("budgetFile" + budgetFile);
  310. await serviceController.SimulaZoneBudget(workingDirectory, budgetFile, zoneFile);
  311. Thread.Sleep(5000);
  312. Logger.WriteLog("fin zonebudget");
  313. }
  314. //7. Parseo de salidas
  315. files = Directory.GetFiles(workingDirectory).ToList<string>();
  316. string zoneResultsFile = files.Find(x => x.Contains(".zblst"));
  317. string piezometryFile = files.Find(x => x.Contains(".HDS"));
  318. //Parseo del fichero de balance hidrico por zonas
  319. if (System.IO.File.Exists(zoneResultsFile))
  320. {
  321. Logger.WriteLog("inicio balance zonas");
  322. ZoneBalance[] zoneBalance = parser.ParseZonesBalance(zoneResultsFile);
  323. //Se inserta el balance hidrico en bbdd
  324. foreach (ZoneBalance zone in zoneBalance)
  325. {
  326. //Obtenemos la respuestas de la creacion con el objeto
  327. var resultBalance = await balanceHidricoController.PostBalance(zone.ToBalanceHidrico(simulacionId));
  328. uint balanceId = resultBalance.Value;
  329. var dataIn = await balanceHidricoController.PostBalanceIn(zone.In.ToDatosBalanceIn(balanceId));
  330. uint balanceDataInId = dataIn.Value;
  331. List<ConexionZonasIn> conexionZonasIns = new();
  332. zone.In.ZoneConnections.ForEach(x => conexionZonasIns.Add(x.ToConexionIn(balanceDataInId)));
  333. await balanceHidricoController.PostMultipleConnectionIn(conexionZonasIns.ToArray());
  334. var dataOut = await balanceHidricoController.PostBalanceOut(zone.Out.ToDatosBalanceOut(balanceId));
  335. uint balanceDataOutId = dataOut.Value;
  336. List<ConexionZonasOut> conexionZonasOuts = new();
  337. zone.Out.ZoneConnections.ForEach(x => conexionZonasOuts.Add(x.ToConexionOut(balanceDataOutId)));
  338. await balanceHidricoController.PostMultipleConnectionOut(conexionZonasOuts.ToArray());
  339. }
  340. Logger.WriteLog("fin balance hidrico zonas");
  341. }
  342. //Parseo del fichero de resultados sobre el modelo entero
  343. if (System.IO.File.Exists(modelResultsFile))
  344. {
  345. Logger.WriteLog("inicio balance modelo entero");
  346. ZoneBalance[] modelBalance = parser.ParseWholeModelBalance(modelResultsFile);
  347. Logger.WriteLog("este es el lenght del balance entero " + modelBalance.Length);
  348. //insertamos en bbdd
  349. try
  350. {
  351. foreach (ZoneBalance zone in modelBalance)
  352. {
  353. var result = await balanceHidricoController.PostBalance(zone.ToBalanceHidrico(simulacionId));
  354. uint balanceId = result.Value;
  355. await balanceHidricoController.PostBalanceIn(zone.In.ToDatosBalanceIn(balanceId));
  356. await balanceHidricoController.PostBalanceOut(zone.Out.ToDatosBalanceOut(balanceId));
  357. }
  358. }
  359. catch (Exception ex)
  360. {
  361. Logger.WriteLog(ex.Message);
  362. Logger.WriteLog(ex.InnerException.Message);
  363. throw;
  364. }
  365. Logger.WriteLog("fin balance modelo entero");
  366. }
  367. //Parseo del fichero HDS para obtener el mapa de piezometria
  368. if (System.IO.File.Exists(piezometryFile))
  369. {
  370. Logger.WriteLog("inicio piezometria");
  371. List<Piezometria> piezometrias = parser.ParseBinaryFile(piezometryFile, simulacionId);
  372. foreach (Piezometria p in piezometrias)
  373. {
  374. await piezometriaController.PostPiezometria(p);
  375. }
  376. Logger.WriteLog("fin piezometria");
  377. }
  378. piezometriaController.GetDimensions(simulacionId, out int rows, out int cols);
  379. await serviceController.GetIsolinesWithPython(workingDirectory, piezometryFile, rows, cols);
  380. Thread.Sleep(5000);
  381. Logger.WriteLog("ya tengo la imagen de isolineas");
  382. Logger.WriteLog(workingDirectory + Constants.SVGFileName);
  383. Byte[] bytes = System.IO.File.ReadAllBytes(workingDirectory + Constants.SVGFileName);
  384. String base64Image = Convert.ToBase64String(bytes);
  385. Logger.WriteLog("guardo imagen de isolineas");
  386. Logger.WriteLog(base64Image);
  387. await SetIsolinesImage(simulacionId, base64Image);
  388. Logger.WriteLog("ya esta guardada");
  389. /*
  390. * se transforma la imagen vectorial obtenida anteriormente para transformarla en un geojson
  391. */
  392. string svgFileOriginal = workingDirectory + Constants.SVGFileName;
  393. string svgModifiedFile = workingDirectory + Constants.SVGGeoJson;
  394. string geoJsonFile = workingDirectory + Constants.GeoJsonFileName;
  395. if (System.IO.File.Exists(svgFileOriginal))
  396. {
  397. if (Utilities.ModifySVG(svgFileOriginal, aquifer.WorldOriginX, aquifer.WorldOriginY, aquifer.Length, aquifer.Height, svgModifiedFile))
  398. {
  399. await serviceController.GetGeoJsonFromSVG(workingDirectory, svgModifiedFile);
  400. if (System.IO.File.Exists(geoJsonFile))
  401. {
  402. await SetGeoJson(simulacionId, geoJsonFile);
  403. }
  404. }
  405. }
  406. await serviceQueueController.SetTerminado(serviceQueue.Id);
  407. Logger.WriteLog("terminado");
  408. try
  409. {
  410. Directory.Delete(workingDirectory);
  411. }catch(Exception ex)
  412. {
  413. Logger.WriteLog(ex.Message);
  414. Logger.WriteLog(ex.InnerException.Message);
  415. }
  416. }
  417. return true;
  418. }
  419. [HttpGet("Geojson/{id}")]
  420. [ProducesResponseType(200)]
  421. public async Task<GeoJson> GetGeoJson(uint id)
  422. {
  423. Simulacion s = await _context.Simulaciones.FindAsync(id);
  424. GeoJson jsonObject = JsonConvert.DeserializeObject<GeoJson>(s.Geojson);
  425. return jsonObject;
  426. }
  427. private bool SimulacionExists(uint id)
  428. {
  429. return _context.Simulaciones.Any(e => e.Id == id);
  430. }
  431. private async Task<IActionResult> SetGeoJson(uint simulationId, string geojsonFile)
  432. {
  433. string allText = System.IO.File.ReadAllText(geojsonFile);
  434. allText = allText.Replace("[[[","[[");
  435. allText = allText.Replace("]]]", "]]");
  436. var simulation = _context.Simulaciones.Find(simulationId);
  437. if (simulationId!= simulation.Id)
  438. {
  439. return BadRequest();
  440. }
  441. simulation.Geojson = allText;
  442. _context.Entry(simulation).State = EntityState.Modified;
  443. try
  444. {
  445. await _context.SaveChangesAsync();
  446. }
  447. catch (DbUpdateConcurrencyException)
  448. {
  449. if (!SimulacionExists(simulationId))
  450. {
  451. return NotFound();
  452. }
  453. else
  454. {
  455. throw;
  456. }
  457. }
  458. return CreatedAtAction("GetSimulacion", new { id = simulation.Id }, simulation);
  459. }
  460. private async Task<IActionResult> SetIsolinesImage(uint simulacionId, string imageBase64)
  461. {
  462. Simulacion simulacion = _context.Simulaciones.Find(simulacionId);
  463. Logger.WriteLog("sim "+simulacion.Id);
  464. if (simulacionId != simulacion.Id)
  465. {
  466. return BadRequest();
  467. }
  468. simulacion.IsolineImage = imageBase64;
  469. Logger.WriteLog(imageBase64);
  470. _context.Entry(simulacion).State = EntityState.Modified;
  471. try
  472. {
  473. await _context.SaveChangesAsync();
  474. }
  475. catch (Exception ex)
  476. {
  477. if (!SimulacionExists(simulacionId))
  478. {
  479. return NotFound();
  480. }
  481. else
  482. {
  483. Logger.WriteLog(ex.Message);
  484. Logger.WriteLog(ex.InnerException.Message);
  485. throw;
  486. }
  487. }
  488. Logger.WriteLog("OKIDOKI ISO SAVED");
  489. return CreatedAtAction("GetSimulacion", new { id = simulacion.Id }, simulacion);
  490. }
  491. }
  492. }