Emmanuel BENOîT
e50775ec76
* The main loader script has been updated to generate the list of files it needs to load automatically. As a consequence, files that contained manually-maintained lists of scripts have been removed, and definition directories have been renamed accordingly. * PostgreSQL extension loading and configuration has been moved to a separate script to be loaded automatically in the main transaction. * Data and function definition scripts that had the -data or -functions suffix have been renamed (the suffix is unnecessary). * Unit tests have been reorganised to follow the definition's structure. * Documentation has been improved
839 lines
20 KiB
PL/PgSQL
839 lines
20 KiB
PL/PgSQL
-- LegacyWorlds Beta 6
|
|
-- PostgreSQL database scripts
|
|
--
|
|
-- Battle functions and utility views
|
|
--
|
|
-- Copyright(C) 2004-2010, DeepClone Development
|
|
-- --------------------------------------------------------
|
|
|
|
|
|
--
|
|
-- Checks if a battle needs to start at a given location
|
|
--
|
|
-- Parameters:
|
|
-- l_id Location to check
|
|
--
|
|
-- Returns:
|
|
-- Whether or not a new battle is to be added
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.check_start( l_id INT )
|
|
RETURNS BOOLEAN
|
|
STRICT STABLE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
BEGIN
|
|
PERFORM p.name_id
|
|
FROM verse.planets p
|
|
INNER JOIN fleets.fleets f ON f.location_id = p.name_id AND f.attacking
|
|
LEFT OUTER JOIN fleets.movements m
|
|
ON m.fleet_id = f.id
|
|
LEFT OUTER JOIN battles.battles b
|
|
ON b.location_id = p.name_id AND last_tick IS NULL
|
|
WHERE p.name_id = l_id AND m.fleet_id IS NULL and b.id IS NULL;
|
|
RETURN FOUND;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
--
|
|
-- Creates or returns an empire's battle record
|
|
--
|
|
-- Parameters:
|
|
-- e_id Empire identifier
|
|
--
|
|
-- Returns:
|
|
-- the empire battle record's identifier
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.goc_empire( e_id INT )
|
|
RETURNS BIGINT
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
r_id BIGINT;
|
|
BEGIN
|
|
LOOP
|
|
SELECT INTO r_id id FROM battles.empires
|
|
WHERE empire_id = e_id;
|
|
EXIT WHEN FOUND;
|
|
|
|
INSERT INTO battles.empires ( name , empire_id )
|
|
SELECT name , id FROM naming.empire_names
|
|
WHERE id = e_id
|
|
RETURNING id INTO r_id;
|
|
|
|
PERFORM * FROM battles.empires
|
|
WHERE empire_id = e_id AND id <> r_id;
|
|
EXIT WHEN NOT FOUND;
|
|
|
|
DELETE FROM battles.empires WHERE id = r_id;
|
|
END LOOP;
|
|
RETURN r_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Creates or returns a protagonist's record
|
|
--
|
|
-- Parameters:
|
|
-- b_id Battle identifier
|
|
-- e_id Empire identifier
|
|
-- mode Protagonist mode
|
|
-- tick Current tick
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.goc_protagonist( b_id BIGINT , e_id INT , mode BOOLEAN , tick BIGINT )
|
|
RETURNS BIGINT
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
be_id BIGINT;
|
|
p_id BIGINT;
|
|
lmode BOOLEAN;
|
|
BEGIN
|
|
be_id := battles.goc_empire( e_id );
|
|
|
|
LOOP
|
|
SELECT INTO p_id id FROM battles.protagonists
|
|
WHERE battle_id = b_id AND empire_id = be_id;
|
|
EXIT WHEN FOUND;
|
|
|
|
BEGIN
|
|
INSERT INTO battles.protagonists ( battle_id , empire_id )
|
|
VALUES ( b_id , be_id )
|
|
RETURNING id INTO p_id;
|
|
EXIT;
|
|
EXCEPTION
|
|
WHEN unique_violation THEN
|
|
-- Do nothing
|
|
END;
|
|
END LOOP;
|
|
|
|
SELECT INTO lmode attacking
|
|
FROM battles.status_changes WHERE protagonist_id = p_id
|
|
ORDER BY tick_identifier DESC LIMIT 1;
|
|
IF NOT FOUND OR lmode <> mode
|
|
THEN
|
|
INSERT INTO battles.status_changes ( protagonist_id , tick_identifier , attacking )
|
|
VALUES ( p_id , tick , mode );
|
|
END IF;
|
|
|
|
RETURN p_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Initialises a battle record
|
|
--
|
|
-- Parameters:
|
|
-- l_id Location
|
|
-- tick The current tick's identifier
|
|
--
|
|
-- Returns:
|
|
-- the battle's identifier
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.initialise( l_id INT , tick BIGINT )
|
|
RETURNS BIGINT
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
-- Battle ID
|
|
b_id BIGINT;
|
|
-- Battle planet ID
|
|
bp_id BIGINT;
|
|
-- Record for list operations
|
|
rec RECORD;
|
|
-- Protagonist ID
|
|
prot_id BIGINT;
|
|
-- Fleet status record ID
|
|
bf_id BIGINT;
|
|
-- Planet owner
|
|
po_id INT;
|
|
BEGIN
|
|
-- Create main battle record
|
|
INSERT INTO battles.battles ( location_id , first_tick )
|
|
VALUES ( l_id , tick )
|
|
RETURNING id INTO b_id;
|
|
|
|
-- Create planet record
|
|
INSERT INTO battles.planets( battle_id , tick_identifier , change_type , name )
|
|
SELECT b_id , tick , 'INIT'::battle_planet_change , n.name
|
|
FROM naming.map_names n WHERE n.id = l_id
|
|
RETURNING id INTO bp_id;
|
|
|
|
-- Insert list of initial buildings
|
|
INSERT INTO battles.buildings ( planet_id , building_id , change )
|
|
SELECT bp_id , b.building_id , b.amount
|
|
FROM verse.planet_buildings b
|
|
INNER JOIN tech.buildings bd ON bd.buildable_id = b.building_id
|
|
WHERE b.planet_id = l_id AND bd.output_type = 'DEF' AND b.amount > 0;
|
|
|
|
-- Insert defensive power
|
|
INSERT INTO battles.defences ( battle_id , tick_identifier , power )
|
|
SELECT b_id , tick , floor( verse.adjust_production(
|
|
verse.get_raw_production( p.name_id , 'DEF' ) , ph.current / p.population ) )
|
|
FROM verse.planets p
|
|
INNER JOIN verse.planet_happiness ph ON ph.planet_id = p.name_id
|
|
WHERE p.name_id = l_id;
|
|
|
|
-- Add protagonists and initial status
|
|
SELECT INTO po_id empire_id FROM emp.planets ep WHERE ep.planet_id = l_id;
|
|
FOR rec IN SELECT f.owner_id AS id , f.attacking AS mode
|
|
FROM fleets.fleets f
|
|
INNER JOIN naming.empire_names n ON n.id = f.owner_id
|
|
LEFT OUTER JOIN fleets.movements m ON m.fleet_id = f.id
|
|
WHERE f.location_id = l_id AND m.fleet_id IS NULL
|
|
UNION SELECT ep.empire_id AS id , FALSE AS mode
|
|
FROM emp.planets ep
|
|
WHERE ep.planet_id = l_id
|
|
LOOP
|
|
prot_id := battles.goc_protagonist( b_id , rec.id , rec.mode , tick );
|
|
IF po_id = rec.id THEN
|
|
INSERT INTO battles.planet_ownership ( protagonist_id )
|
|
VALUES ( prot_id );
|
|
END IF;
|
|
|
|
-- Insert fleets
|
|
INSERT INTO battles.fleets ( protagonist_id , tick_identifier , change_type )
|
|
VALUES ( prot_id , tick , 'INIT'::battle_fleet_change )
|
|
RETURNING id INTO bf_id;
|
|
INSERT INTO battles.ships ( fleet_id , ship_id , change )
|
|
SELECT bf_id , s.ship_id , sum( s.amount )
|
|
FROM fleets.fleets f
|
|
INNER JOIN fleets.ships s ON s.fleet_id = f.id
|
|
LEFT OUTER JOIN fleets.movements m ON m.fleet_id = f.id
|
|
WHERE f.owner_id = rec.id AND f.location_id = l_id
|
|
AND m.fleet_id IS NULL AND f.status <> 'DEPLOYING'
|
|
GROUP BY s.ship_id;
|
|
END LOOP;
|
|
|
|
RETURN b_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Adds fleet ship changes
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.add_fleet_change( f_id BIGINT , s_id INT , l_change INT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
BEGIN
|
|
LOOP
|
|
UPDATE battles.ships SET change = change + l_change
|
|
WHERE fleet_id = f_id AND ship_id = s_id;
|
|
EXIT WHEN FOUND;
|
|
|
|
BEGIN
|
|
INSERT INTO battles.ships ( fleet_id , ship_id , change )
|
|
VALUES ( f_id , s_id , l_change );
|
|
EXIT;
|
|
EXCEPTION
|
|
WHEN unique_violation THEN
|
|
-- Do nothing
|
|
END;
|
|
END LOOP;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Gets or create a planet's change record
|
|
--
|
|
-- Parameters:
|
|
-- b_id Battle identifier
|
|
-- ctype Change type
|
|
-- tick Tick identifier
|
|
--
|
|
-- Returns:
|
|
-- the planet's change record
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.goc_planet( b_id BIGINT , ctype battle_planet_change , tick BIGINT )
|
|
RETURNS BIGINT
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
pr_id BIGINT;
|
|
BEGIN
|
|
LOOP
|
|
SELECT INTO pr_id id FROM battles.planets
|
|
WHERE battle_id = b_id AND change_type = ctype
|
|
AND tick_identifier = tick;
|
|
EXIT WHEN FOUND;
|
|
|
|
BEGIN
|
|
INSERT INTO battles.planets( battle_id , tick_identifier , change_type )
|
|
VALUES ( b_id , tick , ctype )
|
|
RETURNING id INTO pr_id;
|
|
EXIT;
|
|
EXCEPTION
|
|
WHEN unique_violation THEN
|
|
-- Do nothing
|
|
END;
|
|
END LOOP;
|
|
|
|
RETURN pr_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Records building changes
|
|
--
|
|
-- Parameters:
|
|
-- pcr_id Planet change record
|
|
-- bt_id Building type
|
|
-- amount Amount
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.record_building_change( pcr_id BIGINT , bt_id INT , amount INT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
BEGIN
|
|
LOOP
|
|
UPDATE battles.buildings
|
|
SET change = change + amount
|
|
WHERE planet_id = pcr_id AND building_id = bt_id;
|
|
EXIT WHEN FOUND;
|
|
|
|
BEGIN
|
|
INSERT INTO battles.buildings ( planet_id , building_id , change )
|
|
VALUES ( pcr_id , bt_id , amount );
|
|
EXIT;
|
|
EXCEPTION
|
|
WHEN unique_violation THEN
|
|
-- Do nothing
|
|
END;
|
|
END LOOP;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Adds buildings to a planet's buildings list
|
|
--
|
|
-- Parameters:
|
|
-- p_id Planet identifier
|
|
-- bt_id Building type
|
|
-- amount Amount of buildings to add
|
|
-- tick Current tick identifier
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.add_buildings( p_id INT , bt_id INT , amount INT , tick BIGINT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
b_id BIGINT;
|
|
pcr_id BIGINT;
|
|
BEGIN
|
|
-- Check building type
|
|
PERFORM buildable_id FROM tech.buildings WHERE buildable_id = bt_id AND output_type = 'DEF';
|
|
IF NOT FOUND THEN
|
|
RETURN;
|
|
END IF;
|
|
|
|
-- Get battle identifier
|
|
SELECT INTO b_id id FROM battles.battles
|
|
WHERE location_id = p_id AND last_tick IS NULL;
|
|
IF NOT FOUND THEN
|
|
RETURN;
|
|
END IF;
|
|
|
|
-- Record changes
|
|
pcr_id := battles.goc_planet( b_id , 'BUILD' , tick );
|
|
PERFORM battles.record_building_change( pcr_id , bt_id , amount );
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Removes buildings to a planet's buildings list
|
|
--
|
|
-- Parameters:
|
|
-- p_id Planet identifier
|
|
-- bt_id Building type
|
|
-- amount Amount of buildings to remove
|
|
-- bdmg Whether the losses have been caused by battle damage
|
|
-- tick Current tick identifier
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.remove_buildings( p_id INT , bt_id INT , amount INT , bdmg BOOLEAN , tick BIGINT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
b_id BIGINT;
|
|
pcr_id BIGINT;
|
|
BEGIN
|
|
-- Check building type
|
|
PERFORM buildable_id FROM tech.buildings WHERE buildable_id = bt_id AND output_type = 'DEF';
|
|
IF NOT FOUND THEN
|
|
RETURN;
|
|
END IF;
|
|
|
|
-- Get battle identifier
|
|
SELECT INTO b_id id FROM battles.battles
|
|
WHERE location_id = p_id AND last_tick IS NULL;
|
|
IF NOT FOUND THEN
|
|
RETURN;
|
|
END IF;
|
|
|
|
-- Record changes
|
|
pcr_id := battles.goc_planet( b_id , ( CASE WHEN bdmg THEN 'BATTLE' ELSE 'DESTROY' END )::battle_planet_change , tick );
|
|
PERFORM battles.record_building_change( pcr_id , bt_id , -amount );
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Adds a record when the planet is renamed
|
|
--
|
|
-- Parameters:
|
|
-- p_id Planet identifier
|
|
-- nnm New name
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.rename_planet( p_id INT , nnm TEXT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
b_id BIGINT;
|
|
pcr_id BIGINT;
|
|
BEGIN
|
|
-- Get battle identifier
|
|
SELECT INTO b_id id FROM battles.battles
|
|
WHERE location_id = p_id AND last_tick IS NULL;
|
|
IF NOT FOUND THEN
|
|
RETURN;
|
|
END IF;
|
|
|
|
-- Record change
|
|
pcr_id := battles.goc_planet( b_id , 'RENAME' , sys.get_tick() );
|
|
UPDATE battles.planets
|
|
SET name = nnm
|
|
WHERE id = pcr_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Gets or creates a fleet change record
|
|
--
|
|
-- Parameters:
|
|
-- bp_id Protagonist identifier
|
|
-- tick Tick identifier
|
|
-- ctype Change type
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.goc_fleet_change( bp_id BIGINT , tick BIGINT , ctype battle_fleet_change )
|
|
RETURNS BIGINT
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
bf_id BIGINT;
|
|
BEGIN
|
|
LOOP
|
|
SELECT INTO bf_id id FROM battles.fleets
|
|
WHERE protagonist_id = bp_id AND tick_identifier = tick AND change_type = ctype;
|
|
EXIT WHEN FOUND;
|
|
|
|
BEGIN
|
|
INSERT INTO battles.fleets ( protagonist_id , tick_identifier , change_type )
|
|
VALUES ( bp_id , tick , ctype )
|
|
RETURNING id INTO bf_id;
|
|
EXIT;
|
|
EXCEPTION
|
|
WHEN unique_violation THEN
|
|
-- Do nothing
|
|
END;
|
|
END LOOP;
|
|
RETURN bf_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Adds a fleet to the battle
|
|
--
|
|
-- Parameters:
|
|
-- b_id Battle identifier
|
|
-- f_id Fleet identifier
|
|
-- dep Whether the fleet was added at the end of its deployment phase
|
|
-- tick Current tick
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.add_fleet( b_id BIGINT , f_id BIGINT , dep BOOLEAN , tick BIGINT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
e_id INT;
|
|
f_att BOOLEAN;
|
|
bp_id BIGINT;
|
|
bf_id BIGINT;
|
|
ctype battle_fleet_change;
|
|
rec RECORD;
|
|
BEGIN
|
|
-- Get owner's protagonist record
|
|
SELECT INTO e_id , f_att owner_id , attacking
|
|
FROM fleets.fleets
|
|
WHERE id = f_id;
|
|
bp_id := battles.goc_protagonist( b_id , e_id , f_att , tick );
|
|
|
|
-- Try getting or creating the fleet's record
|
|
ctype := ( CASE WHEN dep THEN 'BUILD' ELSE 'ARRIVE' END );
|
|
bf_id := battles.goc_fleet_change( bp_id , tick , ctype );
|
|
|
|
-- Insert or update fleet ships
|
|
FOR rec IN SELECT ship_id , amount FROM fleets.ships WHERE fleet_id = f_id
|
|
LOOP
|
|
PERFORM battles.add_fleet_change( bf_id , rec.ship_id , rec.amount );
|
|
END LOOP;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Removes a fleet from the battle
|
|
--
|
|
-- Parameters:
|
|
-- b_id Battle identifier
|
|
-- f_id Fleet identifier
|
|
-- ctype Change type
|
|
-- tick Current tick
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.remove_fleet( b_id BIGINT , f_id BIGINT , ctype battle_fleet_change , tick BIGINT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
e_id INT;
|
|
f_att BOOLEAN;
|
|
bp_id BIGINT;
|
|
bf_id BIGINT;
|
|
rec RECORD;
|
|
BEGIN
|
|
-- Get owner's protagonist record
|
|
SELECT INTO e_id , f_att owner_id , attacking
|
|
FROM fleets.fleets
|
|
WHERE id = f_id;
|
|
bp_id := battles.goc_protagonist( b_id , e_id , f_att , tick );
|
|
|
|
-- Try getting or creating the fleet's record
|
|
bf_id := battles.goc_fleet_change( bp_id , tick , ctype );
|
|
|
|
-- Insert or update fleet ships
|
|
FOR rec IN SELECT ship_id , amount FROM fleets.ships WHERE fleet_id = f_id
|
|
LOOP
|
|
PERFORM battles.add_fleet_change( bf_id , rec.ship_id , - rec.amount );
|
|
END LOOP;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Sets an empire's mode
|
|
--
|
|
-- Parameters:
|
|
-- b_id Battle identifier
|
|
-- e_id Empire identifier
|
|
-- att Whether the empire is attacking or defending
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.set_mode( b_id BIGINT , e_id INT , att BOOLEAN )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
tick BIGINT;
|
|
p_id BIGINT;
|
|
BEGIN
|
|
SELECT INTO p_id p.id
|
|
FROM battles.empires e
|
|
INNER JOIN battles.protagonists p ON p.empire_id = e.id
|
|
WHERE e.empire_id = e_id AND p.battle_id = b_id;
|
|
IF NOT FOUND THEN
|
|
RETURN;
|
|
END IF;
|
|
|
|
tick := sys.get_tick( );
|
|
LOOP
|
|
DELETE FROM battles.status_changes
|
|
WHERE protagonist_id = p_id AND tick_identifier = tick AND attacking = ( NOT att );
|
|
EXIT WHEN FOUND;
|
|
|
|
BEGIN
|
|
INSERT INTO battles.status_changes ( protagonist_id , tick_identifier , attacking )
|
|
VALUES ( p_id , tick , att );
|
|
EXIT;
|
|
EXCEPTION
|
|
WHEN unique_violation THEN
|
|
-- Do nothing
|
|
END;
|
|
|
|
PERFORM * FROM battles.status_changes
|
|
WHERE protagonist_id = p_id AND tick_identifier = tick AND attacking = att;
|
|
EXIT WHEN FOUND;
|
|
END LOOP;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Inflicts damage to one side of the engagement
|
|
--
|
|
-- Parameters:
|
|
-- b_id Battle identifer
|
|
-- dmg Amount of damage to inflict
|
|
-- att Whether damage is being inflicted to attacking fleets
|
|
-- tick Current tick identifier
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.inflict_damage( b_id BIGINT , dmg REAL , att BOOLEAN , tick BIGINT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
DECLARE
|
|
planet INT;
|
|
tot_power BIGINT;
|
|
st_power BIGINT;
|
|
rec RECORD;
|
|
BEGIN
|
|
-- Get total power from fleets
|
|
SELECT INTO tot_power sum( fs.power )
|
|
FROM battles.battles b
|
|
LEFT OUTER JOIN fleets.fleets f ON f.location_id = b.location_id AND f.attacking = att
|
|
LEFT OUTER JOIN fleets.stats_view fs ON fs.id = f.id
|
|
LEFT OUTER JOIN fleets.movements m ON m.fleet_id = f.id
|
|
WHERE b.id = b_id AND m.fleet_id IS NULL;
|
|
IF tot_power IS NULL THEN
|
|
tot_power := 0;
|
|
END IF;
|
|
SELECT INTO planet location_id FROM battles.battles WHERE id = b_id;
|
|
|
|
-- If damage is being inflicted to defence forces, handle defence buildings
|
|
IF NOT att THEN
|
|
st_power := battles.get_defence_power( b_id , tick );
|
|
tot_power := tot_power + st_power;
|
|
PERFORM sys.write_sql_log( 'BattleUpdate' , 'TRACE'::log_level , 'About to inflict planet damage; total power: ' || tot_power
|
|
|| '; planet power: ' || st_power || '; computed damage: ' || ( dmg * st_power / tot_power )::REAL );
|
|
IF st_power <> 0 THEN
|
|
PERFORM verse.inflict_battle_damage( planet , st_power , ( dmg * st_power / tot_power )::REAL , b_id , tick );
|
|
END IF;
|
|
END IF;
|
|
|
|
-- Inflict damage to fleets
|
|
FOR rec IN SELECT f.id , fs.power
|
|
FROM battles.battles b
|
|
INNER JOIN fleets.fleets f ON f.location_id = b.location_id
|
|
INNER JOIN fleets.stats_view fs ON fs.id = f.id
|
|
LEFT OUTER JOIN fleets.movements m ON m.fleet_id = f.id
|
|
WHERE b.id = b_id AND m.fleet_id IS NULL AND f.attacking = att
|
|
LOOP
|
|
PERFORM fleets.inflict_battle_damage( rec.id , ( dmg * rec.power / tot_power )::REAL , b_id , tick );
|
|
END LOOP;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
|
|
--
|
|
-- Computes the size/power of a planet's defences at a given tick
|
|
--
|
|
|
|
CREATE TYPE planet_defence_size AS (
|
|
item_id INT ,
|
|
amount INT
|
|
);
|
|
|
|
CREATE OR REPLACE FUNCTION battles.get_defence_size( b_id BIGINT , tick BIGINT )
|
|
RETURNS SETOF planet_defence_size
|
|
STRICT STABLE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
SELECT bb.building_id AS item_id , sum( bb.change )::INT AS amount
|
|
FROM battles.planets bp
|
|
INNER JOIN battles.buildings bb ON bb.planet_id = bp.id
|
|
WHERE bp.battle_id = $1 AND bp.tick_identifier <= $2
|
|
GROUP BY bb.building_id;
|
|
$$ LANGUAGE SQL;
|
|
|
|
|
|
CREATE OR REPLACE FUNCTION battles.set_defence_power( b_id BIGINT , tick BIGINT , pw BIGINT )
|
|
RETURNS VOID
|
|
STRICT VOLATILE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
BEGIN
|
|
IF battles.get_defence_power( b_id , tick ) <> pw
|
|
THEN
|
|
LOOP
|
|
UPDATE battles.defences SET power = pw
|
|
WHERE battle_id = b_id AND tick_identifier = tick;
|
|
EXIT WHEN FOUND;
|
|
|
|
BEGIN
|
|
INSERT INTO battles.defences ( battle_id , tick_identifier , power )
|
|
VALUES ( b_id , tick , pw );
|
|
EXIT;
|
|
EXCEPTION
|
|
WHEN unique_violation THEN
|
|
-- EMPTY
|
|
END;
|
|
END LOOP;
|
|
END IF;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
|
|
CREATE OR REPLACE FUNCTION battles.get_defence_power( b_id BIGINT , tick BIGINT )
|
|
RETURNS BIGINT
|
|
STRICT STABLE
|
|
SECURITY DEFINER
|
|
AS $$
|
|
SELECT power FROM battles.defences
|
|
WHERE battle_id = $1 AND tick_identifier = (
|
|
SELECT max( tick_identifier ) FROM battles.defences
|
|
WHERE battle_id = $1 AND tick_identifier <= $2
|
|
);
|
|
$$ LANGUAGE SQL;
|
|
|
|
|
|
|
|
--
|
|
-- Lists battle protagonists in a specific mode at a given tick
|
|
--
|
|
|
|
CREATE OR REPLACE FUNCTION battles.get_protagonists_with_mode( b_id BIGINT , tick BIGINT , mode BOOLEAN )
|
|
RETURNS SETOF BIGINT
|
|
STRICT STABLE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
SELECT protagonist_id
|
|
FROM battles.status_changes c
|
|
INNER JOIN (
|
|
SELECT sc.protagonist_id , max( sc.tick_identifier ) AS tick_identifier
|
|
FROM battles.status_changes sc
|
|
INNER JOIN battles.protagonists p ON p.id = sc.protagonist_id
|
|
WHERE sc.tick_identifier <= $2 AND p.battle_id = $1
|
|
GROUP BY protagonist_id
|
|
) x USING ( protagonist_id , tick_identifier )
|
|
WHERE c.attacking = $3;
|
|
$$ LANGUAGE SQL;
|
|
|
|
|
|
|
|
--
|
|
-- Computes the size/power of fleets in a specific mode at a given tick
|
|
--
|
|
|
|
CREATE TYPE battle_fleet_size AS (
|
|
protagonist_id BIGINT ,
|
|
ship_id INT ,
|
|
amount INT
|
|
);
|
|
|
|
CREATE OR REPLACE FUNCTION battles.get_fleets_composition( b_id BIGINT , tick BIGINT )
|
|
RETURNS SETOF battle_fleet_size
|
|
STRICT STABLE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
SELECT bp.id AS protagonist_id , bs.ship_id AS item_id , sum( bs.change )::INT AS amount
|
|
FROM battles.fleets bf
|
|
INNER JOIN battles.protagonists bp ON bp.id = bf.protagonist_id
|
|
INNER JOIN battles.ships bs ON bs.fleet_id = bf.id
|
|
WHERE bp.battle_id = $1 AND bf.tick_identifier <= $2
|
|
GROUP BY bp.id , bs.ship_id;
|
|
$$ LANGUAGE SQL;
|
|
|
|
|
|
CREATE TYPE battle_fleet_power AS (
|
|
protagonist_id BIGINT ,
|
|
power BIGINT
|
|
);
|
|
|
|
|
|
CREATE OR REPLACE FUNCTION battles.get_fleets_power( b_id BIGINT , tick BIGINT )
|
|
RETURNS SETOF battle_fleet_power
|
|
STRICT STABLE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
SELECT ds.protagonist_id , ( CASE
|
|
WHEN sum( ds.amount * s.power ) IS NULL THEN
|
|
0
|
|
ELSE
|
|
sum( ds.amount * s.power )
|
|
END ) AS power
|
|
FROM battles.get_fleets_composition( $1 , $2 ) ds
|
|
INNER JOIN tech.ships s ON s.buildable_id = ds.ship_id
|
|
GROUP BY ds.protagonist_id;
|
|
$$ LANGUAGE SQL;
|
|
|
|
|
|
CREATE OR REPLACE FUNCTION battles.get_biggest_fleet_owner( b_id BIGINT , tick BIGINT )
|
|
RETURNS INT
|
|
STRICT STABLE
|
|
SECURITY INVOKER
|
|
AS $$
|
|
SELECT e.empire_id
|
|
FROM battles.get_fleets_power( $1 , $2 ) fp
|
|
INNER JOIN battles.protagonists bp ON bp.id = fp.protagonist_id
|
|
INNER JOIN battles.empires e ON e.id = bp.empire_id
|
|
ORDER BY fp.power DESC
|
|
LIMIT 1;
|
|
$$ LANGUAGE SQL;
|
|
|
|
|
|
CREATE OR REPLACE FUNCTION battles.get_fleets_power( b_id BIGINT , tick BIGINT , mode BOOLEAN )
|
|
RETURNS BIGINT
|
|
STRICT STABLE
|
|
SECURITY DEFINER
|
|
AS $$
|
|
SELECT ( CASE
|
|
WHEN sum( fp.power ) IS NULL THEN
|
|
0
|
|
ELSE
|
|
sum( fp.power )
|
|
END )::BIGINT
|
|
FROM battles.get_fleets_power( $1 , $2 ) fp
|
|
INNER JOIN battles.get_protagonists_with_mode( $1 , $2 , $3 ) pm
|
|
ON fp.protagonist_id = pm;
|
|
$$ LANGUAGE SQL;
|