SimulacionController.cs 21KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534
  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 PozoController pozoController;
  22. private RecargasController recargasController;
  23. private AcuiferoController acuiferoController;
  24. private ServicioController serviceController;
  25. private BalanceHidricoController balanceHidricoController;
  26. private PiezometriaController piezometriaController;
  27. private 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. public async Task<ActionResult<Simulacion>> GetSimulacion(uint id)
  48. {
  49. var simulacion = await _context.Simulaciones.FindAsync(id);
  50. if (simulacion == null)
  51. {
  52. return NotFound();
  53. }
  54. return simulacion;
  55. }
  56. // POST: api/Simulacion
  57. // To protect from overposting attacks, see https://go.microsoft.com/fwlink/?linkid=2123754
  58. [HttpPost]
  59. public async Task<ActionResult<Simulacion>> PostSimulacion(Simulacion simulacion)
  60. {
  61. _context.Simulaciones.Add(simulacion);
  62. await _context.SaveChangesAsync();
  63. //prepacion de los valores por defecto para la simulación
  64. pozoController.SetPozosSimulacion(simulacion.IdAcuifero,simulacion.Id);
  65. recargasController.SetRecargasSimulacion(simulacion.IdAcuifero, simulacion.Id);
  66. return CreatedAtAction("GetSimulacion", new { id = simulacion.Id }, simulacion);
  67. }
  68. [HttpPost("Simula/{simulacionId}")]
  69. public async Task<bool> QueueSim(uint simulacionId)
  70. {
  71. Simulacion simulacion = _context.Simulaciones.Find(simulacionId);
  72. ServiceQueue serviceQueue = new ServiceQueue();
  73. serviceQueue.IdSimulacion = simulacionId;
  74. serviceQueue.IdAcuifero = simulacion.IdAcuifero;
  75. await serviceQueueController.PostServiceQueue(serviceQueue);
  76. return true;
  77. }
  78. [HttpPut("UpdateSimWellData")]
  79. public async Task<ActionResult<bool>> UpdateSimWellsFromCsv(uint simulationId, [FromBody] DatosPostSimulacion data)
  80. {
  81. bool result = true;
  82. Parser parser = new();
  83. Simulacion sim = _context.Simulaciones.Find(simulationId);
  84. string file = Constants.PathExtract + data.Fichero.Nombre;
  85. //1. Se eliminan los pozos con sus respectivas time series de la bbdd ya que se va a sobreescribir la informacion
  86. await pozoController.DeletePozoFromSim(simulationId);
  87. //2. Guardamos el fichero que nos viene en el body en formato base64
  88. try
  89. {
  90. byte[] tempBytes = Convert.FromBase64String(data.Fichero.Base64);
  91. System.IO.File.WriteAllBytes( file, tempBytes);
  92. }
  93. catch (Exception ex)
  94. {
  95. return Problem(ex.Message);
  96. }
  97. WellData[] wellDatas = parser.parseWellCSV(file);
  98. await pozoController.PostMultipleWells((uint)sim.IdAcuifero,wellDatas);
  99. return result;
  100. }
  101. [HttpPut]
  102. public async Task<ActionResult<bool>> UpdateSimRechargesFromCsv(uint simulationId, [FromBody] DatosPostSimulacion data)
  103. {
  104. bool result = true;
  105. Parser parser = new();
  106. Simulacion sim = _context.Simulaciones.Find(simulationId);
  107. string file = Constants.PathExtract + data.Fichero.Nombre;
  108. //1. Se eliminan los pozos con sus respectivas time series de la bbdd ya que se va a sobreescribir la informacion
  109. await pozoController.DeletePozoFromSim(simulationId);
  110. //2. Guardamos el fichero que nos viene en el body en formato base64
  111. try
  112. {
  113. byte[] tempBytes = Convert.FromBase64String(data.Fichero.Base64);
  114. System.IO.File.WriteAllBytes(file, tempBytes);
  115. }
  116. catch (Exception ex)
  117. {
  118. return Problem(ex.Message);
  119. }
  120. RechargeData[] rechargeDatas = parser.parseRechargeCSV(file);
  121. await recargasController.PostMultipleRecharges((uint)sim.IdAcuifero, rechargeDatas);
  122. return result;
  123. }
  124. // PUT: api/Simulacion/5
  125. // To protect from overposting attacks, see https://go.microsoft.com/fwlink/?linkid=2123754s
  126. [HttpPut("{id}")]
  127. public async Task<IActionResult> PutSimulacion(uint id, [FromBody] DatosPostSimulacion datos)
  128. {
  129. Simulacion simulacion = _context.Simulaciones.Find(id);
  130. if (id != simulacion.Id)
  131. {
  132. return BadRequest();
  133. }
  134. simulacion.Nombre = (datos.Nombre != null && datos.Nombre != simulacion.Nombre) ? datos.Nombre : simulacion.Nombre;
  135. _context.Entry(simulacion).State = EntityState.Modified;
  136. try
  137. {
  138. await _context.SaveChangesAsync();
  139. }
  140. catch (DbUpdateConcurrencyException)
  141. {
  142. if (!SimulacionExists(id))
  143. {
  144. return NotFound();
  145. }
  146. else
  147. {
  148. throw;
  149. }
  150. }
  151. return CreatedAtAction("GetSimulacion", new { id = simulacion.Id }, simulacion);
  152. }
  153. // DELETE: api/Simulacion/5
  154. [HttpDelete("{id}")]
  155. public async Task<IActionResult> DeleteSimulacion(uint id)
  156. {
  157. var simulacion = await _context.Simulaciones.FindAsync(id);
  158. if (simulacion == null)
  159. {
  160. return NotFound();
  161. }
  162. _context.Simulaciones.Remove(simulacion);
  163. await _context.SaveChangesAsync();
  164. return NoContent();
  165. }
  166. [NonAction]
  167. public async Task<bool> SimulaAsync(ServiceQueue serviceQueue)
  168. {
  169. Logger logger = Logger.GetInstance();
  170. await serviceQueueController.SetCorriendo(serviceQueue.Id);
  171. uint simulacionId = (uint)serviceQueue.IdSimulacion;
  172. Simulacion simulacion = _context.Simulaciones.Find(simulacionId);
  173. //Obtenemos el nombre de fichero original del acuifero
  174. string zip = acuiferoController.GetFicheroZip((uint)simulacion.IdAcuifero);
  175. Logger.WriteLog("estoy simulando");
  176. if (zip != string.Empty)
  177. {
  178. Parser parser = new Parser();
  179. string zip_name = "simulaicion" + simulacion.Nombre;
  180. string zip_copy = zip_name + ".zip";
  181. System.IO.File.Copy(Constants.PathZip + zip, Constants.PathZip + zip_copy, true);
  182. string workingDirectory = Constants.PathExtract + zip_name + "\\";
  183. //Descomprimimos copia para escribir en fichero
  184. Directory.CreateDirectory(workingDirectory);
  185. ZipFile.ExtractToDirectory(Constants.PathZip + zip_copy, workingDirectory, true);
  186. //Obtenemos fichero de pozo y recargas
  187. List<string> files = Directory.GetFiles(workingDirectory).ToList<string>();
  188. string pozoFile = files.Find(x => x.Contains(".VMW"));
  189. string recargaFile = files.Find(x => x.Contains(".VMP"));
  190. string inFile = files.Find(x => x.Contains(".MODFLOW.IN") || x.Contains(".modflow.in"));
  191. string zoneFile = files.Find(x => x.Contains(".ZBI") || x.Contains(".zbi"));
  192. //3- Parseamos pozos
  193. if (System.IO.File.Exists(pozoFile))
  194. {
  195. List<WellData> infoPozosFichero = parser.ParseWellData(pozoFile).ToList<WellData>();
  196. List<Pozo> pozos = pozoController.GetPozosSimulacion(simulacionId);
  197. List<string> lines = new List<string>();
  198. //Por ahora asumimos que el fichero tiene la misma cantidad de pozos que en bbdd para esta simulacion
  199. string[] maxDefault = infoPozosFichero[0].ZMax.Split(" ");
  200. lines.Add(pozos.Count.ToString());
  201. foreach (Pozo p in pozos)
  202. {
  203. List<TsPozo> timeSeriesPozo = pozoController.GetTsPozoFromPozo(p.Id);
  204. lines.Add("0"); //Activo o inactivo
  205. lines.Add(p.Nombre);
  206. lines.Add(p.Latitud.ToString());
  207. lines.Add(p.Longitud.ToString());
  208. lines.Add(p.Minimo.ToString());
  209. maxDefault[0] = p.Maximo.ToString();
  210. string aux = String.Join(' ', maxDefault);
  211. lines.Add(aux); //Maximo
  212. lines.Add(infoPozosFichero[0].ScreenIntervals.ToString()); //screen intervals
  213. lines.Add(infoPozosFichero[0].Top);
  214. lines.Add(infoPozosFichero[0].Bot);
  215. lines.Add(timeSeriesPozo.Count.ToString());
  216. for (int i = 0; i < timeSeriesPozo.Count - 1; i++)
  217. {
  218. lines.Add(timeSeriesPozo[i].MarcaTiempo);
  219. lines.Add(timeSeriesPozo[i + 1].MarcaTiempo);
  220. lines.Add(timeSeriesPozo[i].Valor.ToString() + " " + timeSeriesPozo[i].InfoComplementaria);
  221. }
  222. }
  223. //System.IO.File.WriteAllLines(@"C:\Users\Admin\Desktop\Test.VMW", lines);
  224. System.IO.File.WriteAllLines(pozoFile, lines);
  225. }
  226. //4- Parseamos recargas
  227. if (System.IO.File.Exists(recargaFile))
  228. {
  229. List<Recarga> recargas = recargasController.GetRecargasSimulacion(simulacionId);
  230. string text = System.IO.File.ReadAllText(recargaFile);
  231. string[] data = text.Split("\r\n\r\n");
  232. int layers = int.Parse(data[0][0].ToString());
  233. int blankLinePerBlock = layers + 1;
  234. int rechargeBlockInex = blankLinePerBlock * 2;
  235. List<string> lines = new List<string>();
  236. //Hasta el bloque de recargas vamos poniendo el texto que le precede
  237. for (int i = 0; i < rechargeBlockInex; i++)
  238. {
  239. lines.AddRange(data[i].Split("\r\n").ToList());
  240. lines.Add("");
  241. }
  242. //Bloque recargas
  243. lines.Add(recargas.Count.ToString());
  244. int index = 1;
  245. foreach (Recarga r in recargas)
  246. {
  247. List<TsRecarga> tsRecarga = recargasController.GetTsRecargaFromRecaga(r.Id);
  248. lines.Add(index + " 1 Index 1");
  249. lines.Add("1 rchr entries");
  250. lines.Add(index + " " + tsRecarga[0].Valor + " " + tsRecarga[0].MarcaTiempo + " " + tsRecarga[1].MarcaTiempo + " " + tsRecarga[0].InfoComplementaria);
  251. }
  252. lines.Add("");
  253. //Hasta el final
  254. for (int i = rechargeBlockInex + 1; i < data.Length; i++)
  255. {
  256. lines.AddRange(data[i].Split("\r\n").ToList());
  257. //controlamos que en la ultima iteracion no introduzca un \r\n
  258. if (i < data.Length - 1)
  259. {
  260. lines.Add("");
  261. }
  262. }
  263. //System.IO.File.WriteAllLines(@"C:\Users\Admin\Desktop\TestRec.VMP", lines);
  264. System.IO.File.WriteAllLines(recargaFile, lines);
  265. }
  266. //5.Ejecutamos el modelo
  267. //Primeramente, para ejecutar el modelo, hay que modificar el contenido del fichero con extension .in para que no de error al ejecutarse el modelo
  268. if (System.IO.File.Exists(inFile))
  269. {
  270. List<string> inFileLines = parser.ParseInFile(inFile);
  271. System.IO.File.WriteAllLines(inFile, inFileLines);
  272. }
  273. string inFileName = inFile.Split("\\").ToList().Last();
  274. Logger.WriteLog("ejecuto modflow");
  275. await serviceController.SimulaModflow(workingDirectory, inFileName);
  276. Thread.Sleep(5000);
  277. Logger.WriteLog("fin modflow");
  278. //6.Ejecucion Zonebudget
  279. //Seguidamente miramos el fichero de resultado de zonas (si existe)
  280. if (zoneFile != null && System.IO.File.Exists(zoneFile))
  281. {
  282. files = Directory.GetFiles(workingDirectory).ToList<string>();
  283. string budgetFile = files.Find(x => x.Contains(".BGT"));
  284. Logger.WriteLog("budgetFile" + budgetFile);
  285. await serviceController.SimulaZoneBudget(workingDirectory, budgetFile, zoneFile);
  286. Thread.Sleep(5000);
  287. Logger.WriteLog("fin zonebudget");
  288. }
  289. //7. Parseo de salidas
  290. files = Directory.GetFiles(workingDirectory).ToList<string>();
  291. string zoneResultsFile = files.Find(x => x.Contains(".zblst"));
  292. string modelResultsFile = files.Find(x => x.Contains(".LST"));
  293. string piezometryFile = files.Find(x => x.Contains(".HDS"));
  294. //Parseo del fichero de balance hidrico por zonas
  295. if (System.IO.File.Exists(zoneResultsFile))
  296. {
  297. Logger.WriteLog("inicio balance zonas");
  298. ZoneBalance[] zoneBalance = parser.ParseZonesBalance(zoneResultsFile);
  299. //Se inserta el balance hidrico en bbdd
  300. foreach (ZoneBalance zone in zoneBalance)
  301. {
  302. //Obtenemos la respuestas de la creacion con el objeto
  303. var resultBalance = await balanceHidricoController.PostBalance(zone.ToBalanceHidrico(simulacionId));
  304. uint balanceId = resultBalance.Value;
  305. var dataIn = await balanceHidricoController.PostBalanceIn(zone.In.ToDatosBalanceIn(balanceId));
  306. uint balanceDataInId = dataIn.Value;
  307. List<ConexionZonasIn> conexionZonasIns = new List<ConexionZonasIn>();
  308. zone.In.ZoneConnections.ForEach(x => conexionZonasIns.Add(x.ToConexionIn(balanceDataInId)));
  309. await balanceHidricoController.PostMultipleConnectionIn(conexionZonasIns.ToArray());
  310. var dataOut = await balanceHidricoController.PostBalanceOut(zone.Out.ToDatosBalanceOut(balanceId));
  311. uint balanceDataOutId = dataOut.Value;
  312. List<ConexionZonasOut> conexionZonasOuts = new List<ConexionZonasOut>();
  313. zone.Out.ZoneConnections.ForEach(x => conexionZonasOuts.Add(x.ToConexionOut(balanceDataOutId)));
  314. await balanceHidricoController.PostMultipleConnectionOut(conexionZonasOuts.ToArray());
  315. }
  316. Logger.WriteLog("fin balance hidrico zonas");
  317. }
  318. //Parseo del fichero de resultados sobre el modelo entero
  319. if (System.IO.File.Exists(modelResultsFile))
  320. {
  321. Logger.WriteLog("inicio balance modelo entero");
  322. ZoneBalance[] modelBalance = parser.ParseWholeModelBalance(modelResultsFile);
  323. //insertamos en bbdd
  324. foreach (ZoneBalance zone in modelBalance)
  325. {
  326. var result = await balanceHidricoController.PostBalance(zone.ToBalanceHidrico(simulacionId));
  327. uint balanceId = result.Value;
  328. await balanceHidricoController.PostBalanceIn(zone.In.ToDatosBalanceIn(balanceId));
  329. await balanceHidricoController.PostBalanceOut(zone.Out.ToDatosBalanceOut(balanceId));
  330. }
  331. Logger.WriteLog("fin balance modelo entero");
  332. }
  333. //Parseo del fichero HDS para obtener el mapa de piezometria
  334. if (System.IO.File.Exists(piezometryFile))
  335. {
  336. Logger.WriteLog("inicio piezometria");
  337. List<Piezometria> piezometrias = parser.ParseBinaryFile(piezometryFile, simulacionId);
  338. foreach (Piezometria p in piezometrias)
  339. {
  340. await piezometriaController.PostPiezometria(p);
  341. }
  342. Logger.WriteLog("fin piezometria");
  343. }
  344. piezometriaController.getDimensions(simulacionId, out int rows, out int cols);
  345. await serviceController.GetIsolinesWithPython(workingDirectory, piezometryFile, rows, cols);
  346. Thread.Sleep(5000);
  347. Byte[] bytes = System.IO.File.ReadAllBytes(workingDirectory + "test.png");
  348. String base64Image = Convert.ToBase64String(bytes);
  349. await SetIsolinesImage(simulacionId, base64Image);
  350. await serviceQueueController.SetTerminado(serviceQueue.Id);
  351. Logger.WriteLog("terminado");
  352. }
  353. return true;
  354. }
  355. [HttpGet("Geojson")]
  356. public GeoJson GetGeoJson()
  357. {
  358. string allText = System.IO.File.ReadAllText(@"C:\Users\Admin\Desktop\AAAA\test_aa.geojson");
  359. allText = allText.Replace("[[[","[[");
  360. allText = allText.Replace("]]]", "]]");
  361. GeoJson jsonObject = JsonConvert.DeserializeObject<GeoJson>(allText);
  362. return jsonObject;
  363. }
  364. private bool SimulacionExists(uint id)
  365. {
  366. return _context.Simulaciones.Any(e => e.Id == id);
  367. }
  368. private async Task<IActionResult> SetIsolinesImage(uint simulacionId, string imageBase64)
  369. {
  370. var simulacion = _context.Simulaciones.Find(simulacionId);
  371. if (simulacionId != simulacion.Id)
  372. {
  373. return BadRequest();
  374. }
  375. simulacion.IsolineImage = imageBase64;
  376. _context.Entry(simulacion).State = EntityState.Modified;
  377. try
  378. {
  379. await _context.SaveChangesAsync();
  380. }
  381. catch (DbUpdateConcurrencyException)
  382. {
  383. if (!SimulacionExists(simulacionId))
  384. {
  385. return NotFound();
  386. }
  387. else
  388. {
  389. throw;
  390. }
  391. }
  392. return CreatedAtAction("GetSimulacion", new { id = simulacion.Id }, simulacion);
  393. }
  394. /*
  395. * TEST ENDPOINTS
  396. */
  397. [HttpPost("Simula/Python/{simulacionId}")]
  398. public async Task<bool> SimIsolinesWithPython(uint simulacionId)
  399. {
  400. string workingDirectory = @"C:\Users\Admin\Desktop\TestExtract\simulaicionsimulacion_test\";
  401. string headFile = @"C:\Users\Admin\Desktop\TestExtract\simulaicionsimulacion_test\EJEMPLO_SOTIEL.HDS";
  402. int rows, cols;
  403. piezometriaController.getDimensions(simulacionId, out rows, out cols);
  404. await serviceController.GetIsolinesWithPython(workingDirectory, headFile, rows, cols);
  405. Thread.Sleep(3000);
  406. return true;
  407. }
  408. [HttpGet("Piezometria")]
  409. public async Task<ActionResult<IEnumerable<Simulacion>>> SimPiezometria()
  410. {
  411. Logger.WriteLog("inicio piezometria");
  412. Parser parser = new Parser();
  413. string piezometryFile = @"C:\Users\Admin\Desktop\TestExtract\simulaicionsimulacion_test\EJEMPLO_SOTIEL.HDS";
  414. List<Piezometria> piezometrias = parser.ParseBinaryFile(piezometryFile, 4);
  415. Logger.WriteLog("Fin parseo fichero binario");
  416. Logger.WriteLog("Hay " + piezometrias.Count + " elementos de piezometria");
  417. foreach (Piezometria p in piezometrias)
  418. {
  419. await piezometriaController.PostPiezometria(p);
  420. }
  421. Logger.WriteLog("fin piezometria");
  422. return await GetSimulaciones();
  423. }
  424. }
  425. }