I'm submitting a ...
PostGraphile version: Docker image "graphile/postgraphile:latest"
PostgreSQL version: Docker image "postgres:11-beta2-alpine"
Start a postgraphile container connected to a PostgreSQL 11 database. In my case also a docker image.
Postgraphile crashes on startup with the following error:
PostGraphile server listening on port 5000 🚀
‣ Connected to Postgres instance postgres://postgresql:5432/database_name
‣ Introspected Postgres schema(s) public
‣ GraphQL endpoint served at http://0.0.0.0:5000/graphql
‣ GraphiQL endpoint served at http://0.0.0.0:5000/graphiql
* * *
error: column pro.proisagg does not exist
at Connection.parseE (/usr/local/lib/node_modules/postgraphile/node_modules/pg/lib/connection.js:553:11)
at Connection.parseMessage (/usr/local/lib/node_modules/postgraphile/node_modules/pg/lib/connection.js:378:19)
at Socket.<anonymous> (/usr/local/lib/node_modules/postgraphile/node_modules/pg/lib/connection.js:119:22)
at Socket.emit (events.js:182:13)
at addChunk (_stream_readable.js:283:12)
at readableAddChunk (_stream_readable.js:264:11)
at Socket.Readable.push (_stream_readable.js:219:10)
at TCP.onread (net.js:635:20)
Postgres reports the following error:
2018-07-10 13:59:08.332 UTC [27] ERROR: column pro.proisagg does not exist at character 2044
2018-07-10 13:59:08.332 UTC [27] HINT: Perhaps you meant to reference the column "pro.prolang".
2018-07-10 13:59:08.332 UTC [27] STATEMENT: -- @see https://www.postgresql.org/docs/9.5/static/catalogs.html
-- @see https://github.com/graphile/postgraphile/blob/master/resources/introspection-query.sql
--
-- ## Parameters
-- - `$1`: An array of strings that represent the namespaces we are
-- introspecting.
with
-- @see https://www.postgresql.org/docs/9.5/static/catalog-pg-namespace.html
namespace as (
select
'namespace' as "kind",
nsp.oid as "id",
nsp.nspname as "name",
dsc.description as "description"
from
pg_catalog.pg_namespace as nsp
left join pg_catalog.pg_description as dsc on dsc.objoid = nsp.oid
where
nsp.nspname = any ($1)
order by
nsp.nspname
),
-- Select all of the remote procedures we can use in this schema. This comes
-- first so that we can select types and classes using the information we get
-- from it.
--
-- @see https://www.postgresql.org/docs/9.6/static/catalog-pg-proc.html
procedure as (
select
'procedure' as "kind",
pro.proname as "name",
dsc.description as "description",
pro.pronamespace as "namespaceId",
pro.proisstrict as "isStrict",
pro.proretset as "returnsSet",
case
when pro.provolatile = 'i' then true
when pro.provolatile = 's' then true
else false
end as "isStable",
pro.prorettype as "returnTypeId",
coalesce(pro.proallargtypes, pro.proargtypes) as "argTypeIds",
coalesce(pro.proargnames, array[]::text[]) as "argNames",
pro.pronargdefaults as "argDefaultsNum"
from
pg_catalog.pg_proc as pro
left join pg_catalog.pg_description as dsc on dsc.objoid = pro.oid
where
pro.pronamespace in (select "id" from namespace) and
-- Currently we don’t support functions with variadic arguments. In the
-- future we may, but for now let’s just ignore functions with variadic
-- arguments.
-- TODO: Variadic arguments.
pro.provariadic = 0 and
-- Filter our aggregate functions and window functions.
pro.proisagg = false and
pro.proiswindow = false and
-- We want to make sure the argument mode for all of our arguments is
-- `IN` which means `proargmodes` will be null.
pro.proargmodes is null and
-- Do not select procedures that create range types. These are utility
-- functions that really don’t need to be exposed in an API.
pro.proname not in (select typ.typname from pg_catalog.pg_type as typ where typ.typtype = 'r' and typ.typnamespace = pro.pronamespace) and
-- Do not expose trigger functions (type trigger has oid 2279)
pro.prorettype <> 2279 and
-- We don't want functions that will clash with GraphQL (treat them as private)
pro.proname not like E'\\_\\_%' and
-- We also don’t want procedures that have been defined in our namespace
-- twice. This leads to duplicate fields in the API which throws an
-- error. In the future we may support this case. For now though, it is
-- too complex.
(select count(pro2.*) from pg_catalog.pg_proc as pro2 where pro2.pronamespace = pro.pronamespace and pro2.proname = pro.proname) = 1
order by
pro.pronamespace, pro.proname
),
-- @see https://www.postgresql.org/docs/9.5/static/catalog-pg-class.html
class as (
select
'class' as "kind",
rel.oid as "id",
rel.relname as "name",
rel.relkind as "classKind",
dsc.description as "description",
rel.relnamespace as "namespaceId",
nsp.nspname as "namespaceName",
rel.reltype as "typeId",
-- Here we determine whether or not we can use this class in a
-- `SELECT`’s `FROM` clause. In order to determine this we look at them
-- `relkind` column, if it is `i` (index) or `c` (composite), we cannot
-- select this class. Otherwise we can.
rel.relkind not in ('i', 'c') as "isSelectable",
-- Here we are determining whether we can insert/update/delete a class.
-- This is helpful as it lets us detect non-updatable views and then
-- exclude them from being inserted/updated/deleted into. For more info
-- on how `pg_catalog.pg_relation_is_updatable` works:
--
-- - https://www.postgresql.org/message-id/CAEZATCV2_qN9P3zbvADwME_TkYf2gR_X2cLQR4R+pqkwxGxqJg@mail.gmail.com
-- - https://github.com/postgres/postgres/blob/2410a2543e77983dab1f63f48b2adcd23dba994e/src/backend/utils/adt/misc.c#L684
-- - https://github.com/postgres/postgres/blob/3aff33aa687e47d52f453892498b30ac98a296af/src/backend/rewrite/rewriteHandler.c#L2351
(pg_catalog.pg_relation_is_updatable(rel.oid, true)::bit(8) & B'00010000') = B'00010000' as "isInsertable",
(pg_catalog.pg_relation_is_updatable(rel.oid, true)::bit(8) & B'00001000') = B'00001000' as "isUpdatable",
(pg_catalog.pg_relation_is_updatable(rel.oid, true)::bit(8) & B'00000100') = B'00000100' as "isDeletable"
from
pg_catalog.pg_class as rel
left join pg_catalog.pg_description as dsc on dsc.objoid = rel.oid and dsc.objsubid = 0 and dsc.classoid = (select oid from pg_catalog.pg_class where relname = 'pg_class')
left join pg_catalog.pg_namespace as nsp on nsp.oid = rel.relnamespace
where
rel.relpersistence in ('p') and
-- We don't want classes that will clash with GraphQL (treat them as private)
rel.relname not like E'\\_\\_%' and
rel.relkind in ('r', 'v', 'm', 'c', 'f')
order by
rel.relnamespace, rel.relname
),
-- @see https://www.postgresql.org/docs/9.5/static/catalog-pg-attribute.html
attribute as (
select
'attribute' as "kind",
att.attrelid as "classId",
att.attnum as "num",
att.attname as "name",
dsc.description as "description",
att.atttypid as "typeId",
att.attnotnull as "isNotNull",
att.atthasdef as "hasDefault"
from
pg_catalog.pg_attribute as att
left join pg_catalog.pg_description as dsc on dsc.objoid = att.attrelid and dsc.objsubid = att.attnum
where
att.attrelid in (select "id" from class) and
att.attnum > 0 and
-- We don't want attributes that will clash with GraphQL (treat them as private)
att.attname not like E'\\_\\_%' and
not att.attisdropped
order by
att.attrelid, att.attnum
),
-- @see https://www.postgresql.org/docs/9.5/static/catalog-pg-type.html
type as (
-- Use another `WITH` statement here, because our `WHERE` clause will need
-- to use it.
with type_all as (
select
'type' as "kind",
typ.oid as "id",
typ.typname as "name",
dsc.description as "description",
typ.typnamespace as "namespaceId",
-- We include the namespace name in types only because we select so
-- many types that are outside of our core set of namespaces. Having
-- the namespace name is super helpful when generating SQL, so
-- conditionally having namespace names for types is a pain.
nsp.nspname as "namespaceName",
typ.typtype as "type",
typ.typcategory as "category",
typ.typnotnull as "domainIsNotNull",
nullif(typ.typelem, 0) as "arrayItemTypeId",
typ.typlen as "typeLength",
(typ.typelem <> 0 and typ.typlen = -1) as "isPgArray",
nullif(typ.typrelid, 0) as "classId",
nullif(typ.typbasetype, 0) as "domainBaseTypeId",
-- If this type is an enum type, let’s select all of its enum variants.
--
-- @see https://www.postgresql.org/docs/9.5/static/catalog-pg-enum.html
case
when typ.typtype = 'e' then array(
select enm.enumlabel
from pg_catalog.pg_enum as enm
where enm.enumtypid = typ.oid
order by enm.enumsortorder
)
else null
end as "enumVariants",
-- If this type is a range type, we want to select the sub type of the
-- range.
--
-- @see https://www.postgresql.org/docs/9.6/static/catalog-pg-range.html
case
when typ.typtype = 'r' then (
select rng.rngsubtype
from pg_catalog.pg_range as rng
where rng.rngtypid = typ.oid
limit 1
)
else null
end as "rangeSubTypeId"
from
pg_catalog.pg_type as typ
left join pg_catalog.pg_description as dsc on dsc.objoid = typ.oid
left join pg_catalog.pg_namespace as nsp on nsp.oid = typ.typnamespace
)
select
*
from
type_all as typ
where
typ.id in (select "typeId" from class) or
typ.id in (select "typeId" from attribute) or
typ.id in (select "returnTypeId" from procedure) or
typ.id in (select unnest("argTypeIds") from procedure) or
-- If this type is a base type for *any* domain type, we will include it
-- in our selection. This may mean we fetch more types than we need, but
-- the alternative is to do some funky SQL recursion which would be hard
-- code to read. So we prefer code readability over selecting like 3 or
-- 4 less type rows.
--
-- We also do this for range sub types and array item types.
typ.id in (select "domainBaseTypeId" from type_all) or
typ.id in (select "rangeSubTypeId" from type_all) or
typ.id in (select "arrayItemTypeId" from type_all)
order by
"namespaceId", "name"
),
-- @see https://www.postgresql.org/docs/9.5/static/catalog-pg-constraint.html
"constraint" as (
select distinct on (con.conrelid, con.conkey, con.confrelid, con.confkey)
'constraint' as "kind",
con.conname as "name",
con.contype as "type",
con.conrelid as "classId",
nullif(con.confrelid, 0) as "foreignClassId",
dsc.description as "description",
con.conkey as "keyAttributeNums",
con.confkey as "foreignKeyAttributeNums"
from
pg_catalog.pg_constraint as con
left join pg_catalog.pg_description as dsc on dsc.objoid = con.oid
where
-- Only get constraints for classes we have selected.
con.conrelid in (select "id" from class where "namespaceId" in (select "id" from namespace)) and
case
-- If this is a foreign key constraint, we want to ensure that the
-- foreign class is also in the list of classes we have already
-- selected.
when con.contype = 'f' then con.confrelid in (select "id" from class where "namespaceId" in (select "id" from namespace))
-- Otherwise, this should be true.
else true
end and
-- We only want foreign key, primary key, and unique constraints. We
-- made add support for more constraints in the future.
con.contype in ('f', 'p', 'u')
order by
con.conrelid, con.conkey, con.confrelid, con.confkey, con.conname
)
select row_to_json(x) as object from namespace as x
union all
select row_to_json(x) as object from class as x
union all
select row_to_json(x) as object from attribute as x
union all
select row_to_json(x) as object from type as x
union all
select row_to_json(x) as object from "constraint" as x
union all
select row_to_json(x) as object from procedure as x
;
2018-07-10 13:59:08.367 UTC [27] LOG: could not receive data from client: Connection reset by peer
Postgraphile should start
Thanks for the bug report; it looks like they've removed an attribute we use in the introspection query. I've not touched PG11 yet 😞
We need to mod the introspection query for better PG10 support anyway (e.g. identity columns) so it'll be good to factor this into the same piece of work. 👍
I'm flagging this as PG10-compat even though it's technically PG11-compat.
Just came across this issue too.
postgraphile -c "postgres://localhost:5432/fitness" --watch
PostGraphile v4.0.0-rc.3 server listening on port 5000 🚀
‣ Connected to Postgres instance postgres://localhost:5432/fitness
‣ Introspected Postgres schema(s) public
‣ GraphQL API endpoint served at http://localhost:5000/graphql
‣ GraphiQL IDE endpoint served at http://localhost:5000/graphiql
‣ Documentation: https://graphile.org/postgraphile/introduction/
‣ Please support PostGraphile development: https://graphile.org/donate
* * *
error: column pro.proisagg does not exist
at Connection.parseE (/Users/prisc_000/.config/yarn/global/node_modules/pg/lib/connection.js:545:11)
at Connection.parseMessage (/Users/prisc_000/.config/yarn/global/node_modules/pg/lib/connection.js:370:19)
at Socket.<anonymous> (/Users/prisc_000/.config/yarn/global/node_modules/pg/lib/connection.js:113:22)
at Socket.emit (events.js:180:13)
at addChunk (_stream_readable.js:274:12)
at readableAddChunk (_stream_readable.js:261:11)
at Socket.Readable.push (_stream_readable.js:218:10)
at TCP.onread (net.js:581:20)
PrisBook:instant-api$[master !]
Postgres -V
PrisBook:instant-api$[master !] postgres --version
postgres (PostgreSQL) 11beta1
Is the temporary solution to downgrade to postgres10?
Thank you.
Yeah; I'm afraid so, sorry 😞 If someone wants to take this on, please speak with me first (we should solve some PG10 issues at the same time). 👍
Postgres10 works. Thanks.
Yeah, mostly. It's the GENERATED ALWAYS AS IDENTITY feature we don't support; ref: #631
Hopefully fixed by #294 (not released yet).
@benjie is there a latest channel/package?
Not yet, but soon. I'll post in gitter when it goes out https://gitter.im/graphile/postgraphile