summaryrefslogtreecommitdiffstats
path: root/raddb/mods-config/sql/ippool/oracle/procedure.sql
blob: e48323697783b0c35f04556aa93b0c02e6f6cb03 (plain)
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
--
-- A stored procedure to reallocate a user's previous address, otherwise
-- provide a free address.
--
-- Using this SP reduces the usual set dialogue of queries to a single
-- query:
--
--   BEGIN; SELECT FOR UPDATE; UPDATE; COMMIT;  ->  SELECT sp() FROM dual
--
-- The stored procedure is executed on an database instance within a single
-- round trip which often leads to reduced deadlocking and significant
-- performance improvements especially on multi-master clusters, perhaps even
-- by an order of magnitude or more.
--
-- To use this stored procedure the corresponding queries.conf statements must
-- be configured as follows:
--
-- allocate_begin = ""
-- allocate_find = "\
--         SELECT fr_allocate_previous_or_new_framedipaddress( \
--                 '%{control:${pool_name}}', \
--                 '%{User-Name}', \
--                 '%{%{Calling-Station-Id}:-0}', \
--                 '%{NAS-IP-Address}', \
--                 '${pool_key}', \
--                 ${lease_duration} \
--         ) FROM dual"
-- allocate_update = ""
-- allocate_commit = ""
--

CREATE OR REPLACE FUNCTION fr_allocate_previous_or_new_framedipaddress (
        v_pool_name IN VARCHAR2,
        v_username IN VARCHAR2,
        v_callingstationid IN VARCHAR2,
        v_nasipaddress IN VARCHAR2,
        v_pool_key IN VARCHAR2,
        v_lease_duration IN INTEGER
)
RETURN varchar2 IS
        PRAGMA AUTONOMOUS_TRANSACTION;
        r_address varchar2(15);
BEGIN

        -- Reissue an existing IP address lease when re-authenticating a session
        --
          BEGIN
                SELECT framedipaddress INTO r_address FROM radippool WHERE id IN (
                        SELECT id FROM (
                                SELECT *
                                FROM radippool
                                WHERE pool_name = v_pool_name
                                        AND expiry_time > current_timestamp
                                        AND username = v_username
                                        AND callingstationid = v_callingstationid
                        ) WHERE ROWNUM <= 1
                ) FOR UPDATE SKIP LOCKED;
          EXCEPTION
                    WHEN NO_DATA_FOUND THEN
                        r_address := NULL;
          END;

        -- Reissue an user's previous IP address, provided that the lease is
        -- available (i.e. enable sticky IPs)
        --
        -- When using this SELECT you should delete the one above. You must also
        -- set allocate_clear = "" in queries.conf to persist the associations
        -- for expired leases.
        --
        -- BEGIN
        --         SELECT framedipaddress INTO r_address FROM radippool WHERE id IN (
        --                 SELECT id FROM (
        --                         SELECT *
        --                         FROM radippool
        --                         WHERE pool_name = v_pool_name
        --                                 AND username = v_username
        --                                 AND callingstationid = v_callingstationid
        --                 ) WHERE ROWNUM <= 1
        --         ) FOR UPDATE SKIP LOCKED;
        -- EXCEPTION
        --         WHEN NO_DATA_FOUND THEN
        --              r_address := NULL;
        -- END;

        -- If we didn't reallocate a previous address then pick the least
        -- recently used address from the pool which maximises the likelihood
        -- of re-assigning the other addresses to their recent user
        --
        IF r_address IS NULL THEN
                BEGIN
                        SELECT framedipaddress INTO r_address FROM radippool WHERE id IN (
                                SELECT id FROM (
                                        SELECT *
                                        FROM radippool
                                        WHERE pool_name = v_pool_name
                                        AND expiry_time < CURRENT_TIMESTAMP
                                        ORDER BY expiry_time
                                       ) WHERE ROWNUM <= 1
                        ) FOR UPDATE SKIP LOCKED;
                EXCEPTION
                        WHEN NO_DATA_FOUND THEN
                                r_address := NULL;
                END;
        END IF;

        -- Return nothing if we failed to allocated an address
        --
        IF r_address IS NULL THEN
                COMMIT;
                RETURN r_address;
        END IF;

        -- Update the pool having allocated an IP address
        --
        UPDATE radippool
        SET
                nasipaddress = v_nasipaddress,
                pool_key = v_pool_key,
                callingstationid = v_callingstationid,
                username = v_username,
                expiry_time = CURRENT_TIMESTAMP + v_lease_duration * INTERVAL '1' SECOND(1)
        WHERE framedipaddress = r_address;

        -- Return the address that we allocated
        COMMIT;
        RETURN r_address;

END;
/