| rev | 
   line source | 
| 
jbe@453
 | 
     1 BEGIN;
 | 
| 
jbe@453
 | 
     2 
 | 
| 
jbe@453
 | 
     3 CREATE OR REPLACE VIEW "liquid_feedback_version" AS
 | 
| 
jbe@453
 | 
     4   SELECT * FROM (VALUES ('3.1.0', 3, 1, 0))
 | 
| 
jbe@453
 | 
     5   AS "subquery"("string", "major", "minor", "revision");
 | 
| 
jbe@453
 | 
     6 
 | 
| 
jbe@453
 | 
     7 ALTER TABLE "member" DROP CONSTRAINT "authority_requires_uid_and_vice_versa";
 | 
| 
jbe@453
 | 
     8 ALTER TABLE "member" ADD CONSTRAINT "authority_requires_uid_and_vice_versa" CHECK (
 | 
| 
jbe@453
 | 
     9   ("authority" NOTNULL) = ("authority_uid" NOTNULL) );
 | 
| 
jbe@453
 | 
    10 
 | 
| 
jbe@453
 | 
    11 DROP TABLE "member_application";
 | 
| 
jbe@453
 | 
    12 DROP TYPE "application_access_level";
 | 
| 
jbe@453
 | 
    13 
 | 
| 
jbe@453
 | 
    14 ALTER TABLE "policy" ADD COLUMN "min_admission_time" INTERVAL DEFAULT '0';
 | 
| 
jbe@453
 | 
    15 ALTER TABLE "policy" ALTER COLUMN "min_admission_time" DROP DEFAULT;
 | 
| 
jbe@453
 | 
    16 ALTER TABLE "policy" RENAME COLUMN "admission_time" TO "max_admission_time";
 | 
| 
jbe@453
 | 
    17 
 | 
| 
jbe@453
 | 
    18 ALTER TABLE "policy" DROP CONSTRAINT "timing";
 | 
| 
jbe@453
 | 
    19 ALTER TABLE "policy" ADD CONSTRAINT "timing" CHECK (
 | 
| 
jbe@453
 | 
    20   ( "polling" = FALSE AND
 | 
| 
jbe@453
 | 
    21     "min_admission_time" NOTNULL AND "max_admission_time" NOTNULL AND
 | 
| 
jbe@453
 | 
    22     "discussion_time" NOTNULL AND
 | 
| 
jbe@453
 | 
    23     "verification_time" NOTNULL AND
 | 
| 
jbe@453
 | 
    24     "voting_time" NOTNULL ) OR
 | 
| 
jbe@453
 | 
    25   ( "polling" = TRUE AND
 | 
| 
jbe@453
 | 
    26     "min_admission_time" ISNULL AND "max_admission_time" ISNULL AND
 | 
| 
jbe@453
 | 
    27     "discussion_time" NOTNULL AND
 | 
| 
jbe@453
 | 
    28     "verification_time" NOTNULL AND
 | 
| 
jbe@453
 | 
    29     "voting_time" NOTNULL ) OR
 | 
| 
jbe@453
 | 
    30   ( "polling" = TRUE AND
 | 
| 
jbe@453
 | 
    31     "min_admission_time" ISNULL AND "max_admission_time" ISNULL AND
 | 
| 
jbe@453
 | 
    32     "discussion_time" ISNULL AND
 | 
| 
jbe@453
 | 
    33     "verification_time" ISNULL AND
 | 
| 
jbe@453
 | 
    34     "voting_time" ISNULL ) );
 | 
| 
jbe@453
 | 
    35 
 | 
| 
jbe@453
 | 
    36 ALTER TABLE "policy" DROP CONSTRAINT "issue_quorum_if_and_only_if_not_polling";
 | 
| 
jbe@453
 | 
    37 ALTER TABLE "policy" ADD CONSTRAINT "issue_quorum_if_and_only_if_not_polling" CHECK (
 | 
| 
jbe@453
 | 
    38   "polling" = ("issue_quorum_num" ISNULL) AND
 | 
| 
jbe@453
 | 
    39   "polling" = ("issue_quorum_den" ISNULL) );
 | 
| 
jbe@453
 | 
    40 
 | 
| 
jbe@453
 | 
    41 COMMENT ON COLUMN "policy"."polling" IS 'TRUE = special policy for non-user-generated issues without issue quorum, where certain initiatives (those having the "polling" flag set) do not need to pass the initiative quorum; "min_admission_time" and "max_admission_time" MUST be set to NULL, the other timings may be set to NULL altogether, allowing individual timing for those issues';
 | 
| 
jbe@453
 | 
    42 COMMENT ON COLUMN "policy"."min_admission_time" IS 'Minimum duration of issue state ''admission''; Minimum time an issue stays open';
 | 
| 
jbe@453
 | 
    43 
 | 
| 
jbe@453
 | 
    44 ALTER TABLE "issue" ADD COLUMN "min_admission_time" INTERVAL DEFAULT '0';
 | 
| 
jbe@453
 | 
    45 ALTER TABLE "issue" ALTER COLUMN "min_admission_time" DROP DEFAULT;
 | 
| 
jbe@453
 | 
    46 ALTER TABLE "issue" RENAME COLUMN "admission_time" TO "max_admission_time";
 | 
| 
jbe@453
 | 
    47 
 | 
| 
jbe@453
 | 
    48 ALTER TABLE "issue" DROP CONSTRAINT "admission_time_not_null_unless_instantly_accepted";
 | 
| 
jbe@453
 | 
    49 ALTER TABLE "issue" ADD CONSTRAINT "admission_time_not_null_unless_instantly_accepted" CHECK (
 | 
| 
jbe@453
 | 
    50   ("min_admission_time" NOTNULL) = ("max_admission_time" NOTNULL) AND
 | 
| 
jbe@453
 | 
    51   ("min_admission_time" NOTNULL OR ("accepted" NOTNULL AND "accepted" = "created")) );
 | 
| 
jbe@453
 | 
    52 
 | 
| 
jbe@453
 | 
    53 ALTER TABLE "issue" DROP CONSTRAINT "set_both_or_none_of_snapshot_and_latest_snapshot_event";
 | 
| 
jbe@453
 | 
    54 ALTER TABLE "issue" ADD CONSTRAINT "set_both_or_none_of_snapshot_and_latest_snapshot_event" CHECK (
 | 
| 
jbe@453
 | 
    55   (("snapshot" NOTNULL) = ("latest_snapshot_event" NOTNULL)) );
 | 
| 
jbe@453
 | 
    56 
 | 
| 
jbe@453
 | 
    57 COMMENT ON COLUMN "issue"."closed" IS 'Point in time, when "max_admission_time" or "voting_time" have elapsed, and issue is no longer active; Frontends must ensure that for closed issues additionally to the restrictions for half_frozen and fully_frozen issues a) no voter is added or removed to/from the direct_voter table, b) no votes are added, modified or removed.';
 | 
| 
jbe@453
 | 
    58 COMMENT ON COLUMN "issue"."min_admission_time" IS 'Copied from "policy" table at creation of issue';
 | 
| 
jbe@453
 | 
    59 
 | 
| 
jbe@453
 | 
    60 DROP TRIGGER "update_text_search_data" ON "initiative";
 | 
| 
jbe@453
 | 
    61 ALTER TABLE "initiative" DROP COLUMN "discussion_url";
 | 
| 
jbe@453
 | 
    62 
 | 
| 
jbe@453
 | 
    63 ALTER TABLE "initiative" DROP CONSTRAINT "all_or_none_of_revoked_and_revoked_by_member_id_must_be_null";
 | 
| 
jbe@453
 | 
    64 ALTER TABLE "initiative" ADD CONSTRAINT "all_or_none_of_revoked_and_revoked_by_member_id_must_be_null" CHECK (
 | 
| 
jbe@453
 | 
    65   ("revoked" NOTNULL) = ("revoked_by_member_id" NOTNULL) );
 | 
| 
jbe@453
 | 
    66 
 | 
| 
jbe@453
 | 
    67  CREATE TRIGGER "update_text_search_data"
 | 
| 
jbe@453
 | 
    68    BEFORE INSERT OR UPDATE ON "initiative"
 | 
| 
jbe@453
 | 
    69    FOR EACH ROW EXECUTE PROCEDURE
 | 
| 
jbe@453
 | 
    70    tsvector_update_trigger('text_search_data', 'pg_catalog.simple', "name");
 | 
| 
jbe@453
 | 
    71 
 | 
| 
jbe@453
 | 
    72 ALTER TABLE "event" DROP CONSTRAINT "null_constraints_for_issue_state_changed";
 | 
| 
jbe@453
 | 
    73 ALTER TABLE "event" DROP CONSTRAINT "null_constraints_for_initiative_creation_or_revocation_or_new_draft";
 | 
| 
jbe@453
 | 
    74 ALTER TABLE "event" DROP CONSTRAINT "null_constraints_for_suggestion_creation";
 | 
| 
jbe@453
 | 
    75 
 | 
| 
jbe@453
 | 
    76 ALTER TABLE "event" ADD CONSTRAINT "null_constr_for_issue_state_changed" CHECK (
 | 
| 
jbe@453
 | 
    77   "event" != 'issue_state_changed' OR (
 | 
| 
jbe@453
 | 
    78     "member_id"     ISNULL  AND
 | 
| 
jbe@453
 | 
    79     "issue_id"      NOTNULL AND
 | 
| 
jbe@453
 | 
    80     "state"         NOTNULL AND
 | 
| 
jbe@453
 | 
    81     "initiative_id" ISNULL  AND
 | 
| 
jbe@453
 | 
    82     "draft_id"      ISNULL  AND
 | 
| 
jbe@453
 | 
    83     "suggestion_id" ISNULL  ) );
 | 
| 
jbe@453
 | 
    84 ALTER TABLE "event" ADD CONSTRAINT "null_constr_for_initiative_creation_or_revocation_or_new_draft" CHECK (
 | 
| 
jbe@453
 | 
    85   "event" NOT IN (
 | 
| 
jbe@453
 | 
    86     'initiative_created_in_new_issue',
 | 
| 
jbe@453
 | 
    87     'initiative_created_in_existing_issue',
 | 
| 
jbe@453
 | 
    88     'initiative_revoked',
 | 
| 
jbe@453
 | 
    89     'new_draft_created'
 | 
| 
jbe@453
 | 
    90   ) OR (
 | 
| 
jbe@453
 | 
    91     "member_id"     NOTNULL AND
 | 
| 
jbe@453
 | 
    92     "issue_id"      NOTNULL AND
 | 
| 
jbe@453
 | 
    93     "state"         NOTNULL AND
 | 
| 
jbe@453
 | 
    94     "initiative_id" NOTNULL AND
 | 
| 
jbe@453
 | 
    95     "draft_id"      NOTNULL AND
 | 
| 
jbe@453
 | 
    96     "suggestion_id" ISNULL  ) );
 | 
| 
jbe@453
 | 
    97 ALTER TABLE "event" ADD CONSTRAINT "null_constr_for_suggestion_creation" CHECK (
 | 
| 
jbe@453
 | 
    98   "event" != 'suggestion_created' OR (
 | 
| 
jbe@453
 | 
    99     "member_id"     NOTNULL AND
 | 
| 
jbe@453
 | 
   100     "issue_id"      NOTNULL AND
 | 
| 
jbe@453
 | 
   101     "state"         NOTNULL AND
 | 
| 
jbe@453
 | 
   102     "initiative_id" NOTNULL AND
 | 
| 
jbe@453
 | 
   103     "draft_id"      ISNULL  AND
 | 
| 
jbe@453
 | 
   104     "suggestion_id" NOTNULL ) );
 | 
| 
jbe@453
 | 
   105 
 | 
| 
jbe@453
 | 
   106 CREATE OR REPLACE FUNCTION "copy_timings_trigger"()
 | 
| 
jbe@453
 | 
   107   RETURNS TRIGGER
 | 
| 
jbe@453
 | 
   108   LANGUAGE 'plpgsql' VOLATILE AS $$
 | 
| 
jbe@453
 | 
   109     DECLARE
 | 
| 
jbe@453
 | 
   110       "policy_row" "policy"%ROWTYPE;
 | 
| 
jbe@453
 | 
   111     BEGIN
 | 
| 
jbe@453
 | 
   112       SELECT * INTO "policy_row" FROM "policy"
 | 
| 
jbe@453
 | 
   113         WHERE "id" = NEW."policy_id";
 | 
| 
jbe@453
 | 
   114       IF NEW."min_admission_time" ISNULL THEN
 | 
| 
jbe@453
 | 
   115         NEW."min_admission_time" := "policy_row"."min_admission_time";
 | 
| 
jbe@453
 | 
   116       END IF;
 | 
| 
jbe@453
 | 
   117       IF NEW."max_admission_time" ISNULL THEN
 | 
| 
jbe@453
 | 
   118         NEW."max_admission_time" := "policy_row"."max_admission_time";
 | 
| 
jbe@453
 | 
   119       END IF;
 | 
| 
jbe@453
 | 
   120       IF NEW."discussion_time" ISNULL THEN
 | 
| 
jbe@453
 | 
   121         NEW."discussion_time" := "policy_row"."discussion_time";
 | 
| 
jbe@453
 | 
   122       END IF;
 | 
| 
jbe@453
 | 
   123       IF NEW."verification_time" ISNULL THEN
 | 
| 
jbe@453
 | 
   124         NEW."verification_time" := "policy_row"."verification_time";
 | 
| 
jbe@453
 | 
   125       END IF;
 | 
| 
jbe@453
 | 
   126       IF NEW."voting_time" ISNULL THEN
 | 
| 
jbe@453
 | 
   127         NEW."voting_time" := "policy_row"."voting_time";
 | 
| 
jbe@453
 | 
   128       END IF;
 | 
| 
jbe@453
 | 
   129       RETURN NEW;
 | 
| 
jbe@453
 | 
   130     END;
 | 
| 
jbe@453
 | 
   131   $$;
 | 
| 
jbe@453
 | 
   132 
 | 
| 
jbe@453
 | 
   133 CREATE OR REPLACE FUNCTION "check_issue"
 | 
| 
jbe@453
 | 
   134   ( "issue_id_p" "issue"."id"%TYPE,
 | 
| 
jbe@453
 | 
   135     "persist"    "check_issue_persistence" )
 | 
| 
jbe@453
 | 
   136   RETURNS "check_issue_persistence"
 | 
| 
jbe@453
 | 
   137   LANGUAGE 'plpgsql' VOLATILE AS $$
 | 
| 
jbe@453
 | 
   138     DECLARE
 | 
| 
jbe@453
 | 
   139       "issue_row"      "issue"%ROWTYPE;
 | 
| 
jbe@453
 | 
   140       "policy_row"     "policy"%ROWTYPE;
 | 
| 
jbe@453
 | 
   141       "initiative_row" "initiative"%ROWTYPE;
 | 
| 
jbe@453
 | 
   142       "state_v"        "issue_state";
 | 
| 
jbe@453
 | 
   143     BEGIN
 | 
| 
jbe@453
 | 
   144       PERFORM "require_transaction_isolation"();
 | 
| 
jbe@453
 | 
   145       IF "persist" ISNULL THEN
 | 
| 
jbe@453
 | 
   146         SELECT * INTO "issue_row" FROM "issue" WHERE "id" = "issue_id_p"
 | 
| 
jbe@453
 | 
   147           FOR UPDATE;
 | 
| 
jbe@453
 | 
   148         IF "issue_row"."closed" NOTNULL THEN
 | 
| 
jbe@453
 | 
   149           RETURN NULL;
 | 
| 
jbe@453
 | 
   150         END IF;
 | 
| 
jbe@453
 | 
   151         "persist"."state" := "issue_row"."state";
 | 
| 
jbe@453
 | 
   152         IF
 | 
| 
jbe@453
 | 
   153           ( "issue_row"."state" = 'admission' AND now() >=
 | 
| 
jbe@453
 | 
   154             "issue_row"."created" + "issue_row"."max_admission_time" ) OR
 | 
| 
jbe@453
 | 
   155           ( "issue_row"."state" = 'discussion' AND now() >=
 | 
| 
jbe@453
 | 
   156             "issue_row"."accepted" + "issue_row"."discussion_time" ) OR
 | 
| 
jbe@453
 | 
   157           ( "issue_row"."state" = 'verification' AND now() >=
 | 
| 
jbe@453
 | 
   158             "issue_row"."half_frozen" + "issue_row"."verification_time" ) OR
 | 
| 
jbe@453
 | 
   159           ( "issue_row"."state" = 'voting' AND now() >=
 | 
| 
jbe@453
 | 
   160             "issue_row"."fully_frozen" + "issue_row"."voting_time" )
 | 
| 
jbe@453
 | 
   161         THEN
 | 
| 
jbe@453
 | 
   162           "persist"."phase_finished" := TRUE;
 | 
| 
jbe@453
 | 
   163         ELSE
 | 
| 
jbe@453
 | 
   164           "persist"."phase_finished" := FALSE;
 | 
| 
jbe@453
 | 
   165         END IF;
 | 
| 
jbe@453
 | 
   166         IF
 | 
| 
jbe@453
 | 
   167           NOT EXISTS (
 | 
| 
jbe@453
 | 
   168             -- all initiatives are revoked
 | 
| 
jbe@453
 | 
   169             SELECT NULL FROM "initiative"
 | 
| 
jbe@453
 | 
   170             WHERE "issue_id" = "issue_id_p" AND "revoked" ISNULL
 | 
| 
jbe@453
 | 
   171           ) AND (
 | 
| 
jbe@453
 | 
   172             -- and issue has not been accepted yet
 | 
| 
jbe@453
 | 
   173             "persist"."state" = 'admission' OR
 | 
| 
jbe@453
 | 
   174             -- or verification time has elapsed
 | 
| 
jbe@453
 | 
   175             ( "persist"."state" = 'verification' AND
 | 
| 
jbe@453
 | 
   176               "persist"."phase_finished" ) OR
 | 
| 
jbe@453
 | 
   177             -- or no initiatives have been revoked lately
 | 
| 
jbe@453
 | 
   178             NOT EXISTS (
 | 
| 
jbe@453
 | 
   179               SELECT NULL FROM "initiative"
 | 
| 
jbe@453
 | 
   180               WHERE "issue_id" = "issue_id_p"
 | 
| 
jbe@453
 | 
   181               AND now() < "revoked" + "issue_row"."verification_time"
 | 
| 
jbe@453
 | 
   182             )
 | 
| 
jbe@453
 | 
   183           )
 | 
| 
jbe@453
 | 
   184         THEN
 | 
| 
jbe@453
 | 
   185           "persist"."issue_revoked" := TRUE;
 | 
| 
jbe@453
 | 
   186         ELSE
 | 
| 
jbe@453
 | 
   187           "persist"."issue_revoked" := FALSE;
 | 
| 
jbe@453
 | 
   188         END IF;
 | 
| 
jbe@453
 | 
   189         IF "persist"."phase_finished" OR "persist"."issue_revoked" THEN
 | 
| 
jbe@453
 | 
   190           UPDATE "issue" SET "phase_finished" = now()
 | 
| 
jbe@453
 | 
   191             WHERE "id" = "issue_row"."id";
 | 
| 
jbe@453
 | 
   192           RETURN "persist";
 | 
| 
jbe@453
 | 
   193         ELSIF
 | 
| 
jbe@453
 | 
   194           "persist"."state" IN ('admission', 'discussion', 'verification')
 | 
| 
jbe@453
 | 
   195         THEN
 | 
| 
jbe@453
 | 
   196           RETURN "persist";
 | 
| 
jbe@453
 | 
   197         ELSE
 | 
| 
jbe@453
 | 
   198           RETURN NULL;
 | 
| 
jbe@453
 | 
   199         END IF;
 | 
| 
jbe@453
 | 
   200       END IF;
 | 
| 
jbe@453
 | 
   201       IF
 | 
| 
jbe@453
 | 
   202         "persist"."state" IN ('admission', 'discussion', 'verification') AND
 | 
| 
jbe@453
 | 
   203         coalesce("persist"."snapshot_created", FALSE) = FALSE
 | 
| 
jbe@453
 | 
   204       THEN
 | 
| 
jbe@453
 | 
   205         PERFORM "create_snapshot"("issue_id_p");
 | 
| 
jbe@453
 | 
   206         "persist"."snapshot_created" = TRUE;
 | 
| 
jbe@453
 | 
   207         IF "persist"."phase_finished" THEN
 | 
| 
jbe@453
 | 
   208           IF "persist"."state" = 'admission' THEN
 | 
| 
jbe@453
 | 
   209             PERFORM "set_snapshot_event"("issue_id_p", 'end_of_admission');
 | 
| 
jbe@453
 | 
   210           ELSIF "persist"."state" = 'discussion' THEN
 | 
| 
jbe@453
 | 
   211             PERFORM "set_snapshot_event"("issue_id_p", 'half_freeze');
 | 
| 
jbe@453
 | 
   212           ELSIF "persist"."state" = 'verification' THEN
 | 
| 
jbe@453
 | 
   213             PERFORM "set_snapshot_event"("issue_id_p", 'full_freeze');
 | 
| 
jbe@453
 | 
   214             SELECT * INTO "issue_row" FROM "issue" WHERE "id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   215             SELECT * INTO "policy_row" FROM "policy"
 | 
| 
jbe@453
 | 
   216               WHERE "id" = "issue_row"."policy_id";
 | 
| 
jbe@453
 | 
   217             FOR "initiative_row" IN
 | 
| 
jbe@453
 | 
   218               SELECT * FROM "initiative"
 | 
| 
jbe@453
 | 
   219               WHERE "issue_id" = "issue_id_p" AND "revoked" ISNULL
 | 
| 
jbe@453
 | 
   220               FOR UPDATE
 | 
| 
jbe@453
 | 
   221             LOOP
 | 
| 
jbe@453
 | 
   222               IF
 | 
| 
jbe@453
 | 
   223                 "initiative_row"."polling" OR (
 | 
| 
jbe@453
 | 
   224                   "initiative_row"."satisfied_supporter_count" > 0 AND
 | 
| 
jbe@453
 | 
   225                   "initiative_row"."satisfied_supporter_count" *
 | 
| 
jbe@453
 | 
   226                   "policy_row"."initiative_quorum_den" >=
 | 
| 
jbe@453
 | 
   227                   "issue_row"."population" * "policy_row"."initiative_quorum_num"
 | 
| 
jbe@453
 | 
   228                 )
 | 
| 
jbe@453
 | 
   229               THEN
 | 
| 
jbe@453
 | 
   230                 UPDATE "initiative" SET "admitted" = TRUE
 | 
| 
jbe@453
 | 
   231                   WHERE "id" = "initiative_row"."id";
 | 
| 
jbe@453
 | 
   232               ELSE
 | 
| 
jbe@453
 | 
   233                 UPDATE "initiative" SET "admitted" = FALSE
 | 
| 
jbe@453
 | 
   234                   WHERE "id" = "initiative_row"."id";
 | 
| 
jbe@453
 | 
   235               END IF;
 | 
| 
jbe@453
 | 
   236             END LOOP;
 | 
| 
jbe@453
 | 
   237           END IF;
 | 
| 
jbe@453
 | 
   238         END IF;
 | 
| 
jbe@453
 | 
   239         RETURN "persist";
 | 
| 
jbe@453
 | 
   240       END IF;
 | 
| 
jbe@453
 | 
   241       IF
 | 
| 
jbe@453
 | 
   242         "persist"."state" IN ('admission', 'discussion', 'verification') AND
 | 
| 
jbe@453
 | 
   243         coalesce("persist"."harmonic_weights_set", FALSE) = FALSE
 | 
| 
jbe@453
 | 
   244       THEN
 | 
| 
jbe@453
 | 
   245         PERFORM "set_harmonic_initiative_weights"("issue_id_p");
 | 
| 
jbe@453
 | 
   246         "persist"."harmonic_weights_set" = TRUE;
 | 
| 
jbe@453
 | 
   247         IF
 | 
| 
jbe@453
 | 
   248           "persist"."phase_finished" OR
 | 
| 
jbe@453
 | 
   249           "persist"."issue_revoked" OR
 | 
| 
jbe@453
 | 
   250           "persist"."state" = 'admission'
 | 
| 
jbe@453
 | 
   251         THEN
 | 
| 
jbe@453
 | 
   252           RETURN "persist";
 | 
| 
jbe@453
 | 
   253         ELSE
 | 
| 
jbe@453
 | 
   254           RETURN NULL;
 | 
| 
jbe@453
 | 
   255         END IF;
 | 
| 
jbe@453
 | 
   256       END IF;
 | 
| 
jbe@453
 | 
   257       IF "persist"."issue_revoked" THEN
 | 
| 
jbe@453
 | 
   258         IF "persist"."state" = 'admission' THEN
 | 
| 
jbe@453
 | 
   259           "state_v" := 'canceled_revoked_before_accepted';
 | 
| 
jbe@453
 | 
   260         ELSIF "persist"."state" = 'discussion' THEN
 | 
| 
jbe@453
 | 
   261           "state_v" := 'canceled_after_revocation_during_discussion';
 | 
| 
jbe@453
 | 
   262         ELSIF "persist"."state" = 'verification' THEN
 | 
| 
jbe@453
 | 
   263           "state_v" := 'canceled_after_revocation_during_verification';
 | 
| 
jbe@453
 | 
   264         END IF;
 | 
| 
jbe@453
 | 
   265         UPDATE "issue" SET
 | 
| 
jbe@453
 | 
   266           "state"          = "state_v",
 | 
| 
jbe@453
 | 
   267           "closed"         = "phase_finished",
 | 
| 
jbe@453
 | 
   268           "phase_finished" = NULL
 | 
| 
jbe@453
 | 
   269           WHERE "id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   270         RETURN NULL;
 | 
| 
jbe@453
 | 
   271       END IF;
 | 
| 
jbe@453
 | 
   272       IF "persist"."state" = 'admission' THEN
 | 
| 
jbe@453
 | 
   273         SELECT * INTO "issue_row" FROM "issue" WHERE "id" = "issue_id_p"
 | 
| 
jbe@453
 | 
   274           FOR UPDATE;
 | 
| 
jbe@453
 | 
   275         SELECT * INTO "policy_row"
 | 
| 
jbe@453
 | 
   276           FROM "policy" WHERE "id" = "issue_row"."policy_id";
 | 
| 
jbe@453
 | 
   277         IF 
 | 
| 
jbe@453
 | 
   278           ( now() >=
 | 
| 
jbe@453
 | 
   279             "issue_row"."created" + "issue_row"."min_admission_time" ) AND
 | 
| 
jbe@453
 | 
   280           EXISTS (
 | 
| 
jbe@453
 | 
   281             SELECT NULL FROM "initiative"
 | 
| 
jbe@453
 | 
   282             WHERE "issue_id" = "issue_id_p"
 | 
| 
jbe@453
 | 
   283             AND "supporter_count" > 0
 | 
| 
jbe@453
 | 
   284             AND "supporter_count" * "policy_row"."issue_quorum_den"
 | 
| 
jbe@453
 | 
   285             >= "issue_row"."population" * "policy_row"."issue_quorum_num"
 | 
| 
jbe@453
 | 
   286           )
 | 
| 
jbe@453
 | 
   287         THEN
 | 
| 
jbe@453
 | 
   288           UPDATE "issue" SET
 | 
| 
jbe@453
 | 
   289             "state"          = 'discussion',
 | 
| 
jbe@453
 | 
   290             "accepted"       = coalesce("phase_finished", now()),
 | 
| 
jbe@453
 | 
   291             "phase_finished" = NULL
 | 
| 
jbe@453
 | 
   292             WHERE "id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   293         ELSIF "issue_row"."phase_finished" NOTNULL THEN
 | 
| 
jbe@453
 | 
   294           UPDATE "issue" SET
 | 
| 
jbe@453
 | 
   295             "state"          = 'canceled_issue_not_accepted',
 | 
| 
jbe@453
 | 
   296             "closed"         = "phase_finished",
 | 
| 
jbe@453
 | 
   297             "phase_finished" = NULL
 | 
| 
jbe@453
 | 
   298             WHERE "id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   299         END IF;
 | 
| 
jbe@453
 | 
   300         RETURN NULL;
 | 
| 
jbe@453
 | 
   301       END IF;
 | 
| 
jbe@453
 | 
   302       IF "persist"."phase_finished" THEN
 | 
| 
jbe@453
 | 
   303         IF "persist"."state" = 'discussion' THEN
 | 
| 
jbe@453
 | 
   304           UPDATE "issue" SET
 | 
| 
jbe@453
 | 
   305             "state"          = 'verification',
 | 
| 
jbe@453
 | 
   306             "half_frozen"    = "phase_finished",
 | 
| 
jbe@453
 | 
   307             "phase_finished" = NULL
 | 
| 
jbe@453
 | 
   308             WHERE "id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   309           RETURN NULL;
 | 
| 
jbe@453
 | 
   310         END IF;
 | 
| 
jbe@453
 | 
   311         IF "persist"."state" = 'verification' THEN
 | 
| 
jbe@453
 | 
   312           SELECT * INTO "issue_row" FROM "issue" WHERE "id" = "issue_id_p"
 | 
| 
jbe@453
 | 
   313             FOR UPDATE;
 | 
| 
jbe@453
 | 
   314           SELECT * INTO "policy_row" FROM "policy"
 | 
| 
jbe@453
 | 
   315             WHERE "id" = "issue_row"."policy_id";
 | 
| 
jbe@453
 | 
   316           IF EXISTS (
 | 
| 
jbe@453
 | 
   317             SELECT NULL FROM "initiative"
 | 
| 
jbe@453
 | 
   318             WHERE "issue_id" = "issue_id_p" AND "admitted" = TRUE
 | 
| 
jbe@453
 | 
   319           ) THEN
 | 
| 
jbe@453
 | 
   320             UPDATE "issue" SET
 | 
| 
jbe@453
 | 
   321               "state"          = 'voting',
 | 
| 
jbe@453
 | 
   322               "fully_frozen"   = "phase_finished",
 | 
| 
jbe@453
 | 
   323               "phase_finished" = NULL
 | 
| 
jbe@453
 | 
   324               WHERE "id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   325           ELSE
 | 
| 
jbe@453
 | 
   326             UPDATE "issue" SET
 | 
| 
jbe@453
 | 
   327               "state"          = 'canceled_no_initiative_admitted',
 | 
| 
jbe@453
 | 
   328               "fully_frozen"   = "phase_finished",
 | 
| 
jbe@453
 | 
   329               "closed"         = "phase_finished",
 | 
| 
jbe@453
 | 
   330               "phase_finished" = NULL
 | 
| 
jbe@453
 | 
   331               WHERE "id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   332             -- NOTE: The following DELETE statements have effect only when
 | 
| 
jbe@453
 | 
   333             --       issue state has been manipulated
 | 
| 
jbe@453
 | 
   334             DELETE FROM "direct_voter"     WHERE "issue_id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   335             DELETE FROM "delegating_voter" WHERE "issue_id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   336             DELETE FROM "battle"           WHERE "issue_id" = "issue_id_p";
 | 
| 
jbe@453
 | 
   337           END IF;
 | 
| 
jbe@453
 | 
   338           RETURN NULL;
 | 
| 
jbe@453
 | 
   339         END IF;
 | 
| 
jbe@453
 | 
   340         IF "persist"."state" = 'voting' THEN
 | 
| 
jbe@453
 | 
   341           IF coalesce("persist"."closed_voting", FALSE) = FALSE THEN
 | 
| 
jbe@453
 | 
   342             PERFORM "close_voting"("issue_id_p");
 | 
| 
jbe@453
 | 
   343             "persist"."closed_voting" = TRUE;
 | 
| 
jbe@453
 | 
   344             RETURN "persist";
 | 
| 
jbe@453
 | 
   345           END IF;
 | 
| 
jbe@453
 | 
   346           PERFORM "calculate_ranks"("issue_id_p");
 | 
| 
jbe@453
 | 
   347           RETURN NULL;
 | 
| 
jbe@453
 | 
   348         END IF;
 | 
| 
jbe@453
 | 
   349       END IF;
 | 
| 
jbe@453
 | 
   350       RAISE WARNING 'should not happen';
 | 
| 
jbe@453
 | 
   351       RETURN NULL;
 | 
| 
jbe@453
 | 
   352     END;
 | 
| 
jbe@453
 | 
   353   $$;
 | 
| 
jbe@453
 | 
   354 
 | 
| 
jbe@453
 | 
   355 COMMIT;
 |