pgrx_pg_sys/submodules/
oids.rs

1//LICENSE Portions Copyright 2019-2021 ZomboDB, LLC.
2//LICENSE
3//LICENSE Portions Copyright 2021-2023 Technology Concepts & Design, Inc.
4//LICENSE
5//LICENSE Portions Copyright 2023-2023 PgCentral Foundation, Inc. <contact@pgcentral.org>
6//LICENSE
7//LICENSE All rights reserved.
8//LICENSE
9//LICENSE Use of this source code is governed by the MIT license that can be found in the LICENSE file.
10#![allow(non_camel_case_types)]
11use crate as pg_sys;
12use crate::BuiltinOid;
13use crate::Datum;
14use pgrx_sql_entity_graph::metadata::{
15    ArgumentError, Returns, ReturnsError, SqlMapping, SqlTranslatable,
16};
17
18/// An [object identifier][pg_docs_oid] in Postgres.
19///
20/// This is meant to be understood purely by equality. There is no sensible "order" for Oids.
21///
22/// # Notes
23/// `Default` shall return a sensical Oid, not necessarily a useful one.
24/// Currently, this means that it returns the invalid Oid.
25///
26/// [pg_docs_oid]: https://www.postgresql.org/docs/current/datatype-oid.html
27#[repr(transparent)]
28#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
29#[derive(serde::Deserialize, serde::Serialize)]
30pub struct Oid(pub(crate) u32);
31
32impl Oid {
33    pub const INVALID: Oid = Oid(0);
34
35    /// Generate an Oid from an arbitrary u32.
36    /// # Safety
37    /// This allows you to create an Oid that Postgres won't recognize and throw it into Postgres.
38    /// Don't.
39    ///
40    /// You should know what kind of object the identifier would imply before creating it.
41    /// Postgres may sometimes call very different internal functions based on an Oid input.
42    /// The extension programming interface of Postgres can reach deep, calling functions
43    /// that assume the caller should be trusted like Postgres would trust itself. Because it is.
44    /// Postgres tables can also change at runtime, so if an Oid is not a [BuiltinOid],
45    /// what Postgres does based on an Oid can change dynamically.
46    ///
47    /// The existence of this `unsafe` requirement to create *arbitrary* Oids does not, itself,
48    /// constitute a promise any Oid from Postgres or PGRX is guaranteed to be valid or sensical.
49    /// There are many existing problems in the way of this, for example:
50    /// - `Oid` includes the guaranteed-wrong values [Oid::INVALID]
51    /// - Postgres may return arbitrary-seeming Oids, like [BuiltinOid::UNKNOWNOID]
52    /// - an Oid can arrive in Rust from a table a non-superuser can write
53    /// - PGRX mostly relies on Rust's type system instead of the dynamic typing of Postgres,
54    ///   thus often deliberately does not bother to remember what OID something had.
55    ///
56    /// So this function is merely a reminder. Even for extensions that work with many Oids,
57    /// it is not typical to need to create one from an arbitrary `u32`. Prefer to use a constant,
58    /// or a [BuiltinOid], or to obtain one from querying Postgres, or simply use [Oid::INVALID].
59    /// Marking it as an `unsafe fn` is an invitation to get an Oid from more trustworthy sources.
60    /// This includes [Oid::INVALID], or [BuiltinOid], or by directly calling into Postgres.
61    /// An `unsafe fn` is not an officer of the law empowered to indict C programs for felonies,
62    /// nor cite SQL statements for misdemeanors, nor even truly stop you from foolishness.
63    /// Even "trustworthy" is meant here in a similar sense to how raw pointers can be "trustworthy".
64    /// Often, you should still check if it's null.
65    #[deprecated(since = "0.11.2", note = "safely converts via SPI, use pg_sys::Oid::from(u32)")]
66    pub const unsafe fn from_u32_unchecked(id: u32) -> Oid {
67        Oid(id)
68    }
69
70    /// Gets an Oid from a u32 if it is a valid builtin declared by Postgres
71    pub const fn from_builtin(id: u32) -> Result<Oid, NotBuiltinOid> {
72        match BuiltinOid::from_u32(id) {
73            Ok(oid) => Ok(oid.value()),
74            Err(e) => Err(e),
75        }
76    }
77
78    pub const fn as_u32(self) -> u32 {
79        self.0
80    }
81}
82
83impl Default for Oid {
84    fn default() -> Oid {
85        Oid::INVALID
86    }
87}
88
89/// De facto available via SPI
90impl From<u32> for Oid {
91    fn from(word: u32) -> Oid {
92        Oid(word)
93    }
94}
95
96impl From<Oid> for u32 {
97    fn from(oid: Oid) -> u32 {
98        oid.0
99    }
100}
101
102impl From<Oid> for crate::Datum {
103    fn from(oid: Oid) -> Self {
104        Datum::from(oid.0)
105    }
106}
107
108impl From<BuiltinOid> for Oid {
109    fn from(builtin: BuiltinOid) -> Oid {
110        builtin.value()
111    }
112}
113
114unsafe impl SqlTranslatable for Oid {
115    fn argument_sql() -> Result<SqlMapping, ArgumentError> {
116        Ok(SqlMapping::literal("oid"))
117    }
118    fn return_sql() -> Result<Returns, ReturnsError> {
119        Ok(Returns::One(SqlMapping::literal("oid")))
120    }
121}
122
123// Actually implemented inside pgXX_oids.rs
124pub type PgBuiltInOids = BuiltinOid;
125
126#[derive(Debug, Clone, Copy)]
127pub enum NotBuiltinOid {
128    /// the invalid OID
129    Invalid,
130    /// not a known, builtin OID
131    Ambiguous,
132    /// value too large to be a valid OID in this Postgres version
133    TooBig,
134}
135
136impl TryFrom<u32> for BuiltinOid {
137    type Error = NotBuiltinOid;
138    fn try_from(uint: u32) -> Result<BuiltinOid, NotBuiltinOid> {
139        BuiltinOid::from_u32(uint)
140    }
141}
142
143impl TryFrom<Oid> for BuiltinOid {
144    type Error = NotBuiltinOid;
145    fn try_from(oid: Oid) -> Result<BuiltinOid, NotBuiltinOid> {
146        BuiltinOid::from_u32(oid.0)
147    }
148}
149
150impl TryFrom<crate::Datum> for BuiltinOid {
151    type Error = NotBuiltinOid;
152    fn try_from(datum: crate::Datum) -> Result<BuiltinOid, NotBuiltinOid> {
153        let uint = u32::try_from(datum.value()).map_err(|_| NotBuiltinOid::TooBig)?;
154        BuiltinOid::from_u32(uint)
155    }
156}
157
158impl BuiltinOid {
159    pub const fn value(self) -> pg_sys::Oid {
160        Oid(self as u32)
161    }
162
163    pub fn oid(self) -> PgOid {
164        PgOid::from(self)
165    }
166}
167
168#[derive(Copy, Clone, Eq, PartialEq, Hash, Debug)]
169pub enum PgOid {
170    Invalid,
171    Custom(Oid),
172    BuiltIn(BuiltinOid),
173}
174
175impl PgOid {
176    pub const fn from_untagged(oid: Oid) -> PgOid {
177        match BuiltinOid::from_u32(oid.0) {
178            Ok(builtin) => PgOid::BuiltIn(builtin),
179            Err(NotBuiltinOid::Invalid) => PgOid::Invalid,
180            Err(NotBuiltinOid::Ambiguous) => PgOid::Custom(oid),
181            _ => unsafe { core::hint::unreachable_unchecked() },
182        }
183    }
184}
185
186impl From<BuiltinOid> for PgOid {
187    fn from(builtin: BuiltinOid) -> PgOid {
188        PgOid::BuiltIn(builtin)
189    }
190}
191
192impl From<Oid> for PgOid {
193    fn from(oid: Oid) -> PgOid {
194        PgOid::from_untagged(oid)
195    }
196}
197
198impl PgOid {
199    #[inline]
200    pub const fn value(self) -> pg_sys::Oid {
201        match self {
202            PgOid::Invalid => pg_sys::InvalidOid,
203            PgOid::Custom(custom) => custom,
204            PgOid::BuiltIn(builtin) => builtin.value(),
205        }
206    }
207}