-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathfunciones.sql
More file actions
589 lines (531 loc) · 19.5 KB
/
funciones.sql
File metadata and controls
589 lines (531 loc) · 19.5 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
----------------------------------
-- Definición de Funciones -
----------------------------------
create or replace function obtener_edad(persona datos_personales) returns smallint as
$$
declare
intervalo_fechas interval;
fecha_fin timestamp;
begin
if persona.fecha_fallecimiento is not null then
fecha_fin := persona.fecha_fallecimiento::timestamp;
else
fecha_fin := now();
end if;
intervalo_fechas = age(fecha_fin, persona.fecha_nacimiento::timestamp);
return ceil(date_part('year', intervalo_fechas))::smallint;
end;
$$ language plpgsql;
create or replace function obtener_identificacion(persona datos_personales) returns text as
$$
begin
return persona.nombre || ' ' || persona.apellido;
end;
$$ language plpgsql;
create or replace function porcentaje_aleatorio_entre(l_inf real, l_sup real) returns real as
$$
begin
return floor((random() * (l_sup - l_inf + 1) + l_inf)) / 100;
end;
$$ language plpgsql;
create or replace function velocidad_maxima(velocidad_tope real) returns real as
$$
declare
porcentaje real;
begin
porcentaje := porcentaje_aleatorio_entre(85, 99);
return velocidad_tope * porcentaje;
end;
$$ language plpgsql;
create or replace function velocidad_minima(velocidad_base real) returns real as
$$
declare
porcentaje real;
begin
porcentaje := porcentaje_aleatorio_entre(50, 69);
return velocidad_base * porcentaje;
end;
$$ language plpgsql;
create or replace function velocidad_media(velocidad_base real) returns real as
$$
declare
porcentaje real;
begin
porcentaje := porcentaje_aleatorio_entre(70, 84);
return velocidad_base * porcentaje;
end;
$$ language plpgsql;
create or replace function distancia_recorrida(velocidad_media real) returns real as
$$
declare
porcentaje real;
begin
porcentaje := porcentaje_aleatorio_entre(95, 105);
return velocidad_media * porcentaje;
end;
$$ language plpgsql;
create or replace function cantidad_vueltas(distancia_recorrida real, longitud_circuito real) returns real as
$$
begin
return distancia_recorrida / longitud_circuito;
end;
$$ language plpgsql;
create or replace function generar_estadistica(longitud_circuito real, velocidad_maxima_vehiculo real) returns estadistica as
$$
declare
velocidad_maxima real;
velocidad_minima real;
velocidad_media real;
cantidad_vueltas real;
distancia_recorrida real;
begin
velocidad_maxima := velocidad_maxima(velocidad_maxima_vehiculo);
velocidad_minima := velocidad_minima(velocidad_maxima_vehiculo);
velocidad_media := velocidad_media(velocidad_maxima_vehiculo);
distancia_recorrida := distancia_recorrida(velocidad_media);
cantidad_vueltas := cantidad_vueltas(distancia_recorrida, longitud_circuito);
return row (velocidad_maxima,
velocidad_minima,
velocidad_media,
mejor_tiempo_vuelta(cantidad_vueltas),
peor_tiempo_vuelta(cantidad_vueltas),
-1::smallint, -- Usamos un numero negativo para saber que tenemos que recalcular la posicion una vez hayamos generado todas las estadisticas de lahora
distancia_recorrida);
end;
$$ language plpgsql;
create or replace function mejor_tiempo_vuelta(cantidad_vueltas real) returns interval as
$$
declare
tiempo_mejor interval; -- tiempo medio (minutos)
prom interval; -- suma de tiempos parciales para promediar
begin
tiempo_mejor := interval '60 minutes' / cantidad_vueltas; -- Vemos cuál es la media de tiempo por vuelta
select avg(tiempo_mejor * porcentaje_aleatorio_entre(90, 99)) into prom; -- Calculamos el promedio y retornamos
return prom;
end;
$$ language plpgsql;
-- Estima el peor tiempo de vuelta basado en una cantidad de vueltas
create or replace function peor_tiempo_vuelta(cantidad_vueltas real) returns interval as
$$
declare
tiempo_peor interval; -- tiempo medio (en minutos)
prom interval; -- Promedio de peor tiempo
begin
tiempo_peor := interval '60 minutes' / cantidad_vueltas; -- Vemos cuál es la media de tiempo por vuelta
select avg(tiempo_peor * porcentaje_aleatorio_entre(101, 110)) into prom; -- Calculamos promedio y retornamos
return prom;
end;
$$ language plpgsql;
create or replace function aplicar_penalizacion(falla falla_tecnica, intervalo interval) returns interval as
$$
begin
-- Si la falla es grave, entonces vamos a simplemente arrijar una excepción con un código personalizado
if falla.gravedad = 3::smallint then
raise exception using
errcode = 'GRAVE';
end if;
return intervalo + falla.promedio_tiempo;
end;
$$ language plpgsql;
----------------------------------
-- Funciones de Equipo Tecnico -
----------------------------------
CREATE OR REPLACE FUNCTION generador_equipo_tecnico() RETURNS DATOS_PERSONALES
AS
$$
DECLARE
nombre varchar(20);
apellido varchar(20);
genero sexo := 'M';
fecha_nac DATE;
fecha_muerte DATE;
random smallint;
random_dia int;
random_mes int;
random_ano int;
BEGIN
SELECT INTO random floor(random() * (11)); --Se crearan 10 nombres y apellidos, por lo cual el random tiene que estar entre 0 y 10
SELECT INTO random_dia floor(random() * (28) + 1); -- Se crea un dia de 0 a 30 para las fechas
SELECT INTO random_mes floor(random() * (12) + 1); -- Se genera un numero de 1 a 12 para los meses
SELECT INTO random_ano floor(random() * (20 - 10) + 10); -- Para la generacion del año, tiene que tener al menos + de 20 años, por lo cualgeneramos un numero random y le sumamo 1890, para crear el año
if (random < 6) then
nombre := nombre_hombre(random);
genero := 'M';
else
nombre := nombre_mujer(random);
genero := 'F';
end if;
apellido := generar_apellido(random);
fecha_nac := fecha_nacimiento(random_dia, random_mes, random_ano);
fecha_muerte := fecha_muerte(random_dia, random_mes, random_ano);
RETURN row (nombre,apellido,fecha_nac,genero,fecha_muerte);
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION nombre_hombre(random smallint) RETURNS varchar(20)
AS
$$
DECLARE
nombre varchar(20);
BEGIN
-- 5 nombres aleatorios de hombre, dependiendo del numero random que salga se asignara uno
if random = 1 then
nombre := 'John';
elsif random = 2 then
nombre := 'Frank';
elsif random = 3 then
nombre := 'Luis';
elsif random = 4 then
nombre := 'Marcos';
else
nombre := 'Alejandro';
end if;
RETURN nombre;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION nombre_mujer(random smallint) RETURNS varchar(20)
AS
$$
DECLARE
nombre varchar(20);
BEGIN
-- 5 nombres aleatorios de mujer, dependiendo del numero random que salga se asignara uno
-- Como los numeros randon que entren aqui seran mayores a 5, se les resta 5 para que cumplan las condiciones
random := random - 5;
if random = 1 then
nombre := 'Andrea';
elsif random = 2 then
nombre := 'Valentina';
elsif random = 3 then
nombre := 'Luisa';
elsif random = 4 then
nombre := 'Cindy';
else
nombre := 'Antonieta';
end if;
RETURN nombre;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION generar_apellido(random smallint) RETURNS varchar(20)
AS
$$
DECLARE
apellido varchar(20);
BEGIN
-- Apellidos random
if random = 1 then
apellido := 'Montilla';
elsif random = 2 then
apellido := 'Chineti';
elsif random = 3 then
apellido := 'Grande';
elsif random = 4 then
apellido := 'Canovas';
elsif random = 5 then
apellido := 'Quintero';
elsif random = 6 then
apellido := 'Valencia';
elsif random = 7 then
apellido := 'Serra';
elsif random = 8 then
apellido := 'Salas';
elsif random = 9 then
apellido := 'Mendez';
else
apellido := 'Marin';
end if;
RETURN apellido;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION fecha_nacimiento(dia int, mes int, ano int) RETURNS DATE
AS
$$
DECLARE
fecha DATE;
BEGIN
-- Generacion de la fecha de nacimiento, se pasan los randoms como ints y se usa la funcion make_date para crear un tipo date
fecha := make_date(ano + 1890, mes, dia);
RETURN fecha;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION fecha_muerte(dia int, mes int, ano int) RETURNS DATE
AS
$$
DECLARE
fecha DATE;
BEGIN
-- Generacion de la fecha de muerte, se pasan los randoms como ints y se usa la funcion make_date para crear un tipo date
fecha := make_date(ano + 1940, mes, dia);
RETURN fecha;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION equipo_tecnico_completo() RETURNS DATOS_PERSONALES[][]
AS
$$
DECLARE
equipo_tecnico DATOS_PERSONALES[][] := '{{NULL,NULL,NULL,NULL},{NULL,NULL,NULL,NULL}}' ;
i smallint;
BEGIN
-- Procedimiento usado para crear la matriz de Equipo tecnico, donde la primera fila corresponde a los mecanicos y la segunda al director de los pits.
FOR i IN 1..4
LOOP
equipo_tecnico[1][i] := generador_equipo_tecnico();
END LOOP;
equipo_tecnico[2][1] := generador_equipo_tecnico();
RETURN equipo_tecnico;
END;
$$ LANGUAGE plpgsql;
-----------------------------------------
-- Fin de Funciones de Equipo Tecnico -
-----------------------------------------
--Funcion utilizada para devolver la velocidad media en funcion a la cantidad de vueltas
CREATE OR REPLACE FUNCTION consulta_velocidad_media(cantidad_vueltas smallint, longitud_cir real, hora int) RETURNS REAL
AS
$$
BEGIN
--Para los reportes se buscara la velocidad media en la carrera, par esto se necesita la cantidad de vueltas
--finales que se registraron y se dividira entre 24, se multiplicara por la distancia del circuito para
--tener una velocidad media aproximada, ya que eso dira cuanto recorrio aproximadamente cada hora
RETURN (cantidad_vueltas * longitud_cir) / hora;
END;
$$ LANGUAGE plpgsql;
--Funcion utilizada para buscar el mejor tiempo de vuelta
CREATE OR REPLACE FUNCTION consulta_mejor_tiempo(estadisticas estadistica[], hora int) RETURNS interval
AS
$$
DECLARE
mejor interval;
BEGIN
select min((e).mejor_tiempo_vuelta) from unnest(estadisticas[1:hora]) e into mejor;
return mejor;
END;
$$ LANGUAGE plpgsql;
--Funcion usada para calcular la distancia en vueltas con el puesto anterior
--Los parametros serian: la distancia recorrida hasta la hora del participante anterior, la distancia recorrida del corredor y la longitud del circuito
CREATE OR REPLACE FUNCTION consultar_vueltas_anterior(distancia_anterior real, distancia real, longitud real) RETURNS smallint
AS
$$
DECLARE
BEGIN
--Para saber la distancia en vueltas se va a restar las distancias recorridas en la hora
--y se dividiran entre la longitud del circuito par sacar la cantidad de vueltas
--luego se les hara floor para devolver una cantidad entera
if ((cantidad_vueltas(distancia - distancia_anterior, longitud) > 0) AND (cantidad_vueltas(distancia - distancia_anterior, longitud) < 1)) then
return ceiling(cantidad_vueltas(distancia - distancia_anterior, longitud));
end if;
RETURN FLOOR(cantidad_vueltas(distancia - distancia_anterior, longitud));
END;
$$ LANGUAGE plpgsql;
--Funcion utilizada para calcular la distancia que ha recorrido un participante en total
CREATE OR REPLACE FUNCTION calcular_distancia_recorrida(estadisticas estadistica[], hora int) RETURNS real
AS
$$
DECLARE
cont int = array_length(estadisticas, 1);
total real = 0;
BEGIN
-- Utilizamos la función de suma para ayudarnos a hacer la suma rápidamente
select sum((e).distancia_recorrida) from unnest(estadisticas) e into total;
return total;
END;
$$ LANGUAGE plpgsql;
-- funcion creada para calcualr el tiempo con respecto al piloto anterior
--los parametros serian: la distancia recorrida del piloto anterior, la velocidad media a la que iba en esa hora y la distancia recorrida por el piloto
CREATE OR REPLACE FUNCTION calcular_tiempo_anterior(distancia_recorrida_anterior real, velocidad_anterior real,
distancia real) RETURNS time
AS
$$
DECLARE
tiempo interval = '00:00:00';
distancia_faltante real = 0;
tiempo_faltante real;
minutos int;
BEGIN
--Para calcular el tiempo se usara una regla de 3, primero se calculara la distancia que debe recorrer
-- para alcanzar al piloto, siendo una resta de distancias
distancia_faltante = distancia - distancia_recorrida_anterior;
--Luego se multiplicara por 60 y se dividira entra la velocidad a la que iba el piloto para saber en cuanto
--tiempo lo alcanzara usando la velocidad media como referencia
tiempo_faltante = ((distancia_faltante * 60) / velocidad_anterior);
if (tiempo_faltante - FLOOR(tiempo_faltante) > 0.60) then
tiempo_faltante = tiempo_faltante + 1 - 0.60;
end if;
--Luego se creara un tipo date armado con el tiempo
minutos = FLOOR(tiempo_faltante::int);
RETURN make_time(0, minutos, (tiempo_faltante - FLOOR(tiempo_faltante)) * 100);
END;
$$ LANGUAGE plpgsql;
-- funcion creada para calcualr el tiempo con respecto al piloto anterior
--los parametros serian: la distancia recorrida del piloto anterior, la velocidad media a la que iba en esa hora y la distancia recorrida por el piloto
CREATE OR REPLACE FUNCTION calcular_vueltas_hora(estadisticas estadistica[], hora int, longitud real) RETURNS real
AS
$$
DECLARE
distancia_recorrida real;
BEGIN
distancia_recorrida = calcular_distancia_recorrida(estadisticas, hora);
RETURN FLOOR(cantidad_vueltas(distancia_recorrida, longitud));
END;
$$ LANGUAGE plpgsql;
-- Generador aleatorio de fallas técnicas, que pueden ser aplicadas para las estadísticas por hora
--funcion usada para convertir reals en tiempo
CREATE OR REPLACE FUNCTION convertir_tiempo(tiempo real) RETURNS interval
AS $$
DECLARE
minutos int;
segundos real;
BEGIN
minutos = FLOOR(tiempo)::int;
segundos = (tiempo - FLOOR(tiempo)) * 100;
RETURN make_interval(mins => minutos,secs => segundos);
END;
$$ LANGUAGE plpgsql;
-- DANIEL
-- funcion que devuelve la cantidad de horas que un participante estuvo en participacion
CREATE OR REPLACE FUNCTION consulta_horas_en_carrera(estadisticas estadistica[]) RETURNS integer
AS $$
DECLARE
horas int = array_length(estadisticas,1);
BEGIN
RETURN horas;
END;
$$ LANGUAGE plpgsql;
--Funcion que crea la hora final
CREATE OR REPLACE FUNCTION convertir_tiempo_final(tiempo real, horas integer) RETURNS interval
AS $$
DECLARE
minutos int;
segundos real;
BEGIN
minutos = FLOOR(tiempo)::int;
segundos = (tiempo - FLOOR(tiempo)) * 100;
-- Como es la hora final, al real que le paso hay que agregarle las 24 horas, ya que es lo que determianara el excedente
RETURN make_interval(hours => horas, mins => minutos,secs => segundos);
END;
$$ LANGUAGE plpgsql;
create or replace function generar_falla_tecnica() returns falla_tecnica
language plpgsql
as
$$
declare
random integer; -- número aleatorio para selección de falla técnica
falla falla_tecnica; -- Falla Técnica a devolver. Esta puede ser: NULL, Cambio de cauchos, cambio de frenos y Problema de motor
begin
random := (porcentaje_aleatorio_entre(1, 20) * 100)::integer; -- Generamos un número aleatorio entre
-- Dependiendo de la falla técnica generada, hay penalizaciones de tiempo posible. Estas son manejadas como intervalos
case
when random between 1 and 14 then
falla := null;
when random between 15 and 17 then
falla := row ('Cambio de cauchos', 1, make_interval(secs := porcentaje_aleatorio_entre(5, 12) * 100));
when random between 18 and 19 then
falla := row ('Cambio de frenos', 2, make_interval(secs := porcentaje_aleatorio_entre(15, 30) * 100));
when random = 20 then
falla := row ('Problema de motor', 3, null); -- este intervalo no importa, puesto que de todas formas el carro debería ser retirado
end case;
return falla;
end;
$$;
-- Calcula el tiempo que transcurre desde que se cumple el final de la carrera, y vuelve a cruzar la meta
-- Parámetros
-- cantidad_vueltas: la cantidad de vueltas totales que dio
-- velocidad_media: La velocidad que se tuvo en la carrera
-- longitud_circuito: La longitud total del circuito
create or replace function calcular_tiempo_sobrante(cantidad_vueltas real, velocidad_media real, longitud_circuito real) returns interval
language plpgsql
as
$$
declare
kms_restantes real; -- Kilómetros restantes para llegar a la meta
recorrido_vuelta real; -- Parte real de vuelta completado
distancia_restante real; -- distancia para completar la vuelta
begin
-- Obtenemos la parte decimal de la cantidad de vueltas
recorrido_vuelta := cantidad_vueltas - floor(cantidad_vueltas);
-- Calculamos cuanta fracción de vuelta falta para completarla
distancia_restante := 1.0 - recorrido_vuelta;
-- Traducimos lo que valta en la distancia del circuito
kms_restantes := distancia_restante * longitud_circuito;
-- Regla de 3: si recorre velocidad_media kilómetros en 60 minutos, recorre kms_restantes en x minutos
return kms_restantes * make_interval(mins := 60) / velocidad_media;
end;
$$;
-- https://stackoverflow.com/questions/44209460/how-to-transpose-two-dimension-arrays-in-postgresql
CREATE OR REPLACE FUNCTION transpose_2d(anyarray)
RETURNS anyarray AS $$
SELECT array_agg(v ORDER BY j) matrix FROM (
SELECT j, array_agg(v ORDER BY i) AS v FROM (
SELECT i, j, $1[i][j] AS v FROM (
SELECT generate_subscripts($1, 2) j, q.* FROM (
SELECT generate_subscripts($1, 1) AS i, $1
) q
) r
) s
GROUP BY j
) t
$$ LANGUAGE sql IMMUTABLE;
CREATE OR REPLACE FUNCTION consultar_anterior_tiempo(id_actual int ,distancia_recorrida real, id_evento int)
RETURNS interval
language plpgsql
as $$
DECLARE
id_anterior real;
distancia_anterior real;
id_evento_anterior int;
velocidad_anterior real;
dif_recorrido real;
tiempo_tardado real;
dif_tiempo interval;
minutos int;
segundos real;
BEGIN
SELECT siguiente_participacion(id_actual) INTO id_anterior;
SELECT calcular_distancia_recorrida(estadisticas_hora,array_length(estadisticas_hora,1)) INTO distancia_anterior FROM participacion WHERE id = id_anterior;
SELECT evento_id INTO id_evento_anterior FROM participacion WHERE id = id_anterior;
SELECT (estadisticas_hora)[array_length(estadisticas_hora,1)].vel_media INTO velocidad_anterior FROM participacion where id = id_anterior;
if ((id_evento_anterior = -1) OR (id_evento_anterior != id_evento)) then
return 0;
else
dif_recorrido = distancia_recorrida - distancia_anterior;
tiempo_tardado = (dif_recorrido * 60)/velocidad_anterior;
minutos = floor(tiempo_tardado);
segundos = (tiempo_tardado - floor(tiempo_tardado))* 100;
return make_interval(mins => minutos, secs =>segundos);
end if;
return 0;
end;
$$;
CREATE OR REPLACE FUNCTION siguiente_participacion(id_actual int)
RETURNS int
language plpgsql
as
$$
DECLARE
id_sig int;
BEGIN
id_sig = (select id from participacion where id= (id_actual + 2));
if( id_sig > 0) then
return id_sig;
else
return -1;
end if;
return -1;
end;
$$;
CREATE OR REPLACE FUNCTION consulta_velocidad_maxima(id_actual int)
RETURNS int
language plpgsql
as
$$
DECLARE
estadisticas estadistica[];
horas int;
maximo real;
i int;
BEGIN
select estadisticas_hora INTO estadisticas from participacion where id = id_actual;
maximo = 0;
horas = array_length(estadisticas,1);
maximo = estadisticas[horas].vel_maxima;
return maximo;
END;
$$