Estou escrevendo um trabalho para transformar dados de um design antigo em um novo design. Nesse processo, preciso levar o ID de uma inserção para uma tabela separada e usá-lo em uma inserção para a tabela de destino, como tal:
CREATE TABLE t1 {
t1_id BIGSERIAL,
col1 VARCHAR
};
CREATE TABLE t2 {
t2_id BIGSERIAL,
col2 VARCHAR, -- renamed from col1 to avoid confusion
t1_id BIGINT REFERENCES t1.t1_id
};
Eu tenho o SQL definido que corresponde ao seguinte formulário:
WITH ins AS (
INSERT INTO t1 (t1_id) VALUES (DEFAULT) RETURNING t1_id
) INSERT INTO t2
(col1, t1_id)
SELECT
a.val1, (SELECT * FROM ins)
FROM t3 a;
Eu queria que isso executasse o SELECT * FROM ins
para todas as linhas do SELECT
.. mas, em vez disso, ele é executado apenas uma vez e usa esse valor para todas as linhas no SELECT
. Como posso reestruturar meu SQL para obter o comportamento desejado?
edit4
t1 acaba assim:
1,<NULL>
(1 row)
t2 acaba assim:
10,'a',1
11,'b',1 -- problem with id from t1 being 1
12,'c',1 -- problem with id from t1 being 1
.
.
Como eu quero que o t1 seja:
1,<NULL>
2,<NULL>
3,<NULL>
.
.
Como eu quero que o t2 seja:
10,'a',1
11,'b',2 -- id from t1 of 2
12,'c',3 -- id from t1 of 3
.
.
edit Para abordar o que a_horse_with_no_name disse, eu também tentei isso (com o mesmo resultado):
WITH ins AS (
INSERT INTO t1 (t1_id) VALUES (DEFAULT) RETURNING t1_id
) INSERT INTO t2
(col1, t1_id)
SELECT
a.val1, b.t1_id
FROM t3 a
JOIN ins b ON TRUE;
edit2
Eu apenas tentei fazer referência direta ao apropriado SEQUENCE
na minha consulta e isso funciona - mas não gosto muito dessa solução (principalmente porque não gosto de nomes de objetos codificados). Se houver QUALQUER outra solução do que referenciar diretamente o nome do que SEQUENCE
eu apreciaria. :)
edit3
Suponho que outra solução seria fazer uso de a PROCEDURE
para fazer o INSERT
CTE, em vez de ... mas ainda assim eu apreciaria opções / sugestões.
fonte
ins
et3
t1
e não está fornecendo nenhum valort1.col1
. Onde os dados devem chegar para essa coluna? Estát1.col1
relacionado comt2.col1
?INSERT INTO t1 (t1_id) VALUES (DEFAULT)
insere apenas 1 linhat1
. Portanto, não importa se você colocains
aFROM
cláusula na e a juntat3
ou não. Você pode nos mostrar como inserir 2 (ou mais) linhast1
? E mais importante, como você sabe em qual dos 2 (ou mais)t1.id
valores corresponderia as linhas inseridast2
?Respostas:
Eu não entendo por que você precisa de 2 tabelas se elas têm apenas 1-1. Mas aqui está (
pk
é a chave primária det3
):Se o seu t3 for o resultado de uma SELECT em vez de uma tabela preexistente, você poderá implementá-lo dessa forma para não precisar repetir a consulta t3 duas vezes:
fonte
t2_id
. Parece que você pode usar ot2(t1_id)
como o PK det2
.ERROR: syntax error at or near "DEFAULT" LINE 2: DEFAULT AS contact_detail_id
DEFAULT
que não pode ser usado dessa maneira. Não retornando ot.pk