PLPGSQL을 사용하여 현재 search_path에 테이블이 있는지 어떻게 알 수 있습니까?


10

다른 응용 프로그램의 애드온 인 응용 프로그램의 설치 스크립트를 작성 중이므로 다른 응용 프로그램의 테이블이 존재하는지 확인하고 싶습니다. 그렇지 않으면 사용자에게 유용한 오류를주고 싶습니다. 그러나 어떤 스키마가 테이블을 보유하는지 알 수 없습니다.

DO LANGUAGE plpgsql $$
BEGIN
    PERFORM 1
    FROM
        pg_catalog.pg_class c
        JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
    WHERE
        n.nspname = current_setting('search_path')
        AND c.relname = 'foo'
        AND c.relkind = 'r'; -- not sure if I actually need this or not...

    IF NOT FOUND THEN
        RAISE 'This application depends on tables created by another application';
    END IF;
END;
$$;

그러나 기본적으로 current_setting('search_path')포함 "$user",public된 TEXT를 반환하므로 크게 유용하지 않습니다.

내가 생각할 수있는 유일한 다른 것은 테이블에서 선택하고 예외를 포착하는 것입니다. 그것은 일을 할 것이지만, 그것이 매우 우아하다고 생각하지 않으며 사용 하는 것이 비싸다 는 것을 읽었습니다 (아마도 한 번만 실행하기 때문에이 시나리오에서는 괜찮을까요?).

답변:


18

빠르고 더러운

포스트 그레스에서 9.4+ 사용

SELECT to_regclass('foo');

검색 경로에 식별자가 없으면 NULL을 반환합니다.
Postgres 9.3 이상 에서는 다음에 캐스트를regclass 사용하십시오 .

SELECT 'foo'::regclass;

예외가 발생 개체가 발견되지 않는 경우,!

'foo'가 발견 되면 이 표현으로 oid반환됩니다 text. 이는 현재 검색 경로에 따라 스키마로 한정된 테이블 이름 일 뿐이며 필요한 경우 큰 따옴표로 묶습니다.

객체를 찾을 수 없으면 검색 경로의 어느 곳에도 존재하지 않거나 스키마로 한정된 이름 ( ) 이 전혀 없는지 확인할 수 있습니다 schema.foo.

발견 된 경우 두 가지 단점이 있습니다 .

  1. 검색에는 search_path 의 암시 적 스키마 , 즉 pg_catalog및이 포함pg_temp 됩니다. 그러나 목적에 따라 임시 및 시스템 테이블을 제외 할 수 있습니다. (?)

  2. 캐스트 regclass는 시스템 카탈로그의 모든 객체 ( pg_class인덱스, 뷰, 시퀀스 등) 에서 작동합니다 . 테이블뿐만 아니라. 독점 테이블을 독점적으로 찾고있는 것 같습니다. 그러나 이름이 같은 다른 개체에도 문제가있을 수 있습니다. 세부:

느리고 확실

다시 쿼리로 돌아 왔지만 current_setting('search_path')베어 설정을 반환하는를 사용하지 마십시오 . 전용 시스템 정보 기능을 사용하십시오 current_schemas(). 설명서 당 :

current_schemas(boolean) name[]
선택적으로 암시 적 스키마를 포함하여 검색 경로의 스키마 이름

"$user"검색 경로에서 현명하게 해결됩니다. 이름 SESSION_USER이있는 스키마가 존재하지 않으면 스키마는 처음부터 리턴되지 않습니다. 또한 원하는 내용에 따라 암시 적 스키마 ( pg_catalog및 가능하면 pg_temp)를 추가로 출력 할 수는 있지만 현재로서는 원하지 않는 것으로 가정합니다.

DO 
$do$
BEGIN
   IF EXISTS (
      SELECT  -- list can be empty
      FROM   pg_catalog.pg_class c
      JOIN   pg_catalog.pg_namespace n ON n.oid = c.relnamespace
      WHERE  n.nspname = ANY(current_schemas(FALSE))
      AND    n.nspname NOT LIKE 'pg_%'  -- exclude system schemas!
      AND    c.relname = 'foo'
      AND    c.relkind = 'r')           -- you probably need this
   THEN
      RAISE 'This application depends on tables created by another application';
   END IF;
END
$do$;

마지막DO문장을제외한 모든 것을 보여주는 SQL Fiddle
JDBC (SQL Fiddle)에 DO종료 문자가 포함 된 명령문에 문제가 있습니다 .


1

구성 값을 배열로 변환 $user하고 현재 사용자 이름으로 바꿀 수 있습니다 . 그런 다음 where 조건에서 배열을 사용할 수 있습니다.

where n.nspname = any(string_to_array(replace(current_setting('search_path'), '$user', current_user), ','))

0
./sshi.sh vb20deployment controller <<'HERE'
export PGPASSWORD="postgres"
cd logu/postgresql/bin
row=1
tableArray=(table1 table2 table3 table4 table5 table6)

for (( x=0 ; x<=5 ; x++)) ; do        

./psql.bin --port=5432 --username=postgres --host=hostname.rds.amazonaws.com --dbname=mydb -c "SELECT * FROM information_schema.tables WHERE '${tableArray[$x]}' = table_name" | while read -a Record ; do
  row=$((row + 1))
  if [[ $row -gt 3 ]]; then

     echo ${Record[4]}

   fi
done

done


HERE
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.