risingwave_common/session_config/
search_path.rs

1// Copyright 2025 RisingWave Labs
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7//     http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15use std::convert::Infallible;
16use std::str::FromStr;
17
18use super::SESSION_CONFIG_LIST_SEP;
19use crate::catalog::{DEFAULT_SCHEMA_NAME, PG_CATALOG_SCHEMA_NAME, RW_CATALOG_SCHEMA_NAME};
20
21pub const USER_NAME_WILD_CARD: &str = "\"$user\"";
22
23/// see <https://www.postgresql.org/docs/14/runtime-config-client.html#GUC-SEARCH-PATH>
24///
25/// 1. when we `select` or `drop` object and don't give a specified schema, it will search the
26///    object from the valid items in schema `rw_catalog`, `pg_catalog` and `search_path`. If schema
27///    `rw_catalog` and `pg_catalog` are not in `search_path`, we will search them firstly. If they're
28///    in `search_path`, we will follow the order in `search_path`.
29///
30/// 2. when we `create` a `source` or `mv` and don't give a specified schema, it will use the first
31///    valid schema in `search_path`.
32///
33/// 3. when we `create` a `index` or `sink`, it will use the schema of the associated table.
34#[derive(Clone, Debug, PartialEq)]
35pub struct SearchPath {
36    origin_str: String,
37    /// The path will implicitly includes `rw_catalog` and `pg_catalog` if user does specify them.
38    path: Vec<String>,
39    real_path: Vec<String>,
40}
41
42impl SearchPath {
43    pub fn real_path(&self) -> &[String] {
44        &self.real_path
45    }
46
47    pub fn path(&self) -> &[String] {
48        &self.path
49    }
50}
51
52impl Default for SearchPath {
53    fn default() -> Self {
54        [USER_NAME_WILD_CARD, DEFAULT_SCHEMA_NAME]
55            .join(SESSION_CONFIG_LIST_SEP)
56            .parse()
57            .unwrap()
58    }
59}
60
61impl FromStr for SearchPath {
62    type Err = Infallible;
63
64    fn from_str(s: &str) -> Result<Self, Self::Err> {
65        let paths = s.split(SESSION_CONFIG_LIST_SEP).map(|path| path.trim());
66        let mut real_path = vec![];
67        for p in paths {
68            let p = p.trim();
69            if !p.is_empty() {
70                real_path.push(p.to_owned());
71            }
72        }
73        let string = real_path.join(SESSION_CONFIG_LIST_SEP);
74
75        let mut path = real_path.clone();
76        let rw_catalog = RW_CATALOG_SCHEMA_NAME.to_owned();
77        if !real_path.contains(&rw_catalog) {
78            path.insert(0, rw_catalog);
79        }
80
81        let pg_catalog = PG_CATALOG_SCHEMA_NAME.to_owned();
82        if !real_path.contains(&pg_catalog) {
83            path.insert(0, pg_catalog);
84        }
85
86        Ok(Self {
87            origin_str: string,
88            path,
89            real_path,
90        })
91    }
92}
93
94impl std::fmt::Display for SearchPath {
95    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
96        write!(f, "{}", self.origin_str)
97    }
98}