Como o número 1 exige a cópia de dados do mestre para o filho enquanto ele estiver em um ambiente de produção ativo, eu pessoalmente fui com o número 2 (criando um novo mestre). Isso evita interrupções na tabela original enquanto ela estiver ativamente em uso e, se houver algum problema, eu posso excluir facilmente o novo mestre sem problemas e continuar usando a tabela original. Aqui estão as etapas para fazer isso:
Crie uma nova tabela mestre.
CREATE TABLE new_master (
id serial,
counter integer,
dt_created DATE DEFAULT CURRENT_DATE NOT NULL
);
Crie filhos que herdam do mestre.
CREATE TABLE child_2014 (
CONSTRAINT pk_2014 PRIMARY KEY (id),
CONSTRAINT ck_2014 CHECK ( dt_created < DATE '2015-01-01' )
) INHERITS (new_master);
CREATE INDEX idx_2014 ON child_2014 (dt_created);
CREATE TABLE child_2015 (
CONSTRAINT pk_2015 PRIMARY KEY (id),
CONSTRAINT ck_2015 CHECK ( dt_created >= DATE '2015-01-01' AND dt_created < DATE '2016-01-01' )
) INHERITS (new_master);
CREATE INDEX idx_2015 ON child_2015 (dt_created);
...
Copie todos os dados históricos para a nova tabela mestre
INSERT INTO child_2014 (id,counter,dt_created)
SELECT id,counter,dt_created
from old_master
where dt_created < '01/01/2015'::date;
Pausar temporariamente novas inserções / atualizações no banco de dados de produção
Copie os dados mais recentes para a nova tabela mestre
INSERT INTO child_2015 (id,counter,dt_created)
SELECT id,counter,dt_created
from old_master
where dt_created >= '01/01/2015'::date AND dt_created < '01/01/2016'::date;
Renomeie as tabelas para que new_master se torne o banco de dados de produção.
ALTER TABLE old_master RENAME TO old_master_backup;
ALTER TABLE new_master RENAME TO old_master;
Inclua a função para instruções INSERT em old_master para que os dados sejam passados para a partição correta.
CREATE OR REPLACE FUNCTION fn_insert() RETURNS TRIGGER AS $$
BEGIN
IF ( NEW.dt_created >= DATE '2015-01-01' AND
NEW.dt_created < DATE '2016-01-01' ) THEN
INSERT INTO child_2015 VALUES (NEW.*);
ELSIF ( NEW.dt_created < DATE '2015-01-01' ) THEN
INSERT INTO child_2014 VALUES (NEW.*);
ELSE
RAISE EXCEPTION 'Date out of range';
END IF;
RETURN NULL;
END;
$$
LANGUAGE plpgsql;
Adicione gatilho para que a função seja chamada INSERTS
CREATE TRIGGER tr_insert BEFORE INSERT ON old_master
FOR EACH ROW EXECUTE PROCEDURE fn_insert();
Defina a exclusão de restrição como ON
SET constraint_exclusion = on;
Reativar UPDATES e INSERTS no banco de dados de produção
Configure trigger ou cron para que novas partições sejam criadas e a função seja atualizada para atribuir novos dados à correção da partição. Consulte este artigo para exemplos de código
Excluir old_master_backup
DELETE FROM ONLY master_table
é a solução.