पोस्टग्रेज में एनम टाइप वैल्यू कैसे हटाएं?


109

मैं एक एनुम प्रकार मान को कैसे हटाऊं जो मैंने पोस्टग्रेजल में बनाया था?

create type admin_level1 as enum('classifier', 'moderator', 'god');

जैसे मैं moderatorसूची से हटाना चाहता हूं ।

मुझे डॉक्स पर कुछ भी नहीं मिल रहा है।

मैं Postgresql 9.3.4 का उपयोग कर रहा हूं।


4
drop type admin_level1?
बरील

1
अंगूठे का नियम: हर के लिए create xxxहै एकdrop xxx
a_horse_with_no_name

IMO चयनित उत्तर को दूसरे में बदलना होगा।
रोमन पोडलिनोव

जवाबों:


180

आप किसी अन्य प्रकार की तरह enum प्रकारों को हटाते हैं (छोड़ते हैं) DROP TYPE:

DROP TYPE admin_level1;

क्या यह संभव है कि आप वास्तव में पूछ रहे हैं कि किसी एनम प्रकार से एक व्यक्तिगत मूल्य कैसे निकाला जाए ? यदि हां, तो आप नहीं कर सकते। यह समर्थित नहीं है :

यद्यपि enumप्रकार मुख्य रूप से मूल्यों के स्थिर सेट के लिए अभिप्रेत हैं, एक मौजूदा एनम प्रकार में नए मूल्यों को जोड़ने के लिए समर्थन है, और मूल्यों का नाम बदलने के लिए (देखें ALTER TYPE)। मौजूदा मानों को एनम प्रकार से हटाया नहीं जा सकता है, और न ही ऐसे मूल्यों के क्रम को बदला जा सकता है, एनम प्रकार को छोड़ने और फिर से बनाने के लिए।

आपको मान के बिना एक नया प्रकार बनाना होगा, नए प्रकार का उपयोग करने के लिए पुराने प्रकार के सभी मौजूदा उपयोगों को रूपांतरित करें, फिर पुराने प्रकार को छोड़ दें।

उदाहरण के लिए

CREATE TYPE admin_level1 AS ENUM ('classifier', 'moderator');

CREATE TABLE blah (
    user_id integer primary key,
    power admin_level1 not null
);

INSERT INTO blah(user_id, power) VALUES (1, 'moderator'), (10, 'classifier');

ALTER TYPE admin_level1 ADD VALUE 'god';

INSERT INTO blah(user_id, power) VALUES (42, 'god');

-- .... oops, maybe that was a bad idea

CREATE TYPE admin_level1_new AS ENUM ('classifier', 'moderator');

-- Remove values that won't be compatible with new definition
-- You don't have to delete, you might update instead
DELETE FROM blah WHERE power = 'god';

-- Convert to new type, casting via text representation
ALTER TABLE blah 
  ALTER COLUMN power TYPE admin_level1_new 
    USING (power::text::admin_level1_new);

-- and swap the types
DROP TYPE admin_level1;

ALTER TYPE admin_level1_new RENAME TO admin_level1;

1
ये जबरदस्त है! इसके साथ मैं एलेम्बिक माइग्रेशन समस्या को हल करने में कामयाब रहा। मैं नए enum प्रकार को जोड़ नहीं सकता क्योंकि(psycopg2.InternalError) ALTER TYPE ... ADD cannot run inside a transaction block
कारंथन

disable_ddl_transaction जोड़ें! माइग्रेशन फ़ाइल के शीर्ष पर।
Chell

वाराह से शक्ति प्राप्त करें, जहां = 'भगवान'; मेरे मामले में काम नहीं कर रहा
अंकित

1
टीबीएच मुझे समझ में नहीं आता है कि इस उत्तर को क्यों चुना गया। यह उत्तर सही नहीं है! आप निर्दिष्ट लेबल के साथ pg_enum से मान हटा सकते हैं।
रोमन पॉडलिनोव

2
@RomanPoelinov प्रत्यक्ष सूची हेरफेर मैं अपने जोखिम पर हूँ। ऐसे कारण हैं जो पोस्टग्रैम्स मूल रूप से एनम मानों को हटाने का समर्थन नहीं करते हैं। एक असमर्थित और असुरक्षित कैटलॉग हैक की तुलना में यह "सही नहीं" कैसे है?
क्रेग रिंगर

41

यहाँ बहुत अच्छी तरह से लिखा गया है:

http://blog.yo1.dog/updating-enum-values-in-postgresql-the-safe-and-easy-way/

मौजूदा प्रकार का नाम बदलें

ALTER TYPE status_enum RENAME TO status_enum_old;

नए प्रकार बनाएं

CREATE TYPE status_enum AS ENUM('queued', 'running', 'done');

नए प्रकार का उपयोग करने के लिए कॉलम अपडेट करें

ALTER TABLE job ALTER COLUMN job_status TYPE status_enum USING job_status::text::status_enum;

पुराने प्रकार को हटा दें

DROP TYPE status_enum_old;

यह लिंक अब 503 लौटाता है।
ओलिवर इवांस

32

यदि आप enum type का आइटम हटाना चाहते हैं, तो आपको PostgreSQL के सिस्टम टेबल पर काम करना होगा।

इस आदेश के साथ, आप सभी आइटम एनम प्रकार प्रदर्शित कर सकते हैं।

चयन करें * pg_enum से;

फिर जाँचें कि खोज किया गया मूल्य अद्वितीय है। Rekoru हटाने के दौरान विशिष्टता को बढ़ाने के लिए 'enumtypid' के अलावा 'enumtypid' पास होना चाहिए।

यह कमांड एनम प्रकार में प्रविष्टि को हटाता है, जहां 'अद्वितीय' आपका मूल्य है।

DELETE FROM pg_enum en WHERE en.enumtypid = 124 और en.enumlabel = 'अद्वितीय';

नोट जिस उदाहरण का मैंने वर्णन किया है, उसका उपयोग तब किया जाना चाहिए, जब संयोग से हम enum प्रकार में नया मान जोड़ते हैं, और फिर भी हमने डेटाबेस में कहीं भी इसका उपयोग नहीं किया है।


20
यह एक बहुत ही खतरनाक ऑपरेशन है , लेकिन यदि आप जानते हैं कि आप क्या कर रहे हैं, तो यह बहुत ही त्वरित और एक प्रकार का मूल्य निकालने में सफल है। सबसे पहले, सुनिश्चित करें कि कोई भी तालिका एनुम मान का उपयोग नहीं कर रही है जिसे आप निकालना चाहते हैं। यदि आप नहीं करते हैं, तो आप एनम वैल्यू को संदर्भित करने वाली सभी तालिकाओं को बुरी तरह से तोड़ देंगे (उदाहरण के लिए ऐसी तालिका से चयन करेंगे ERROR: invalid internal value for enumऔर कोई परिणाम प्राप्त नहीं करेंगे ।)
क्लिंट पचल

5
यह सही है, यह सबसे महत्वपूर्ण पहलू है जिसे ध्यान में रखा जाना चाहिए। मेरे द्वारा वर्णित उदाहरण का उपयोग किया जाना चाहिए, जब संयोग से हम Enum प्रकार में नया मान जोड़ते हैं, और फिर भी हमने डेटाबेस में कहीं भी इसका उपयोग नहीं किया है।
एलकुड्रो

1
यह देखते हुए कि यह आदेश कितना खतरनाक है DELETE FROM pg_enum en WHERE en.enumtypid=124 AND en.enumlabel='unigue';, नोट बोल्ड में होना चाहिए, कमांड नहीं। यदि आपने किसी तालिका में मान का उपयोग किया है, तो आप इससे उबर नहीं सकते। आप उन पंक्तियों को अपडेट नहीं कर सकते जिनमें मूल्य सम्‍मिलित हैं, आप परिवर्तित नहीं कर सकते। केवल तरीका पूरी पंक्ति को हटाना है।
सिल्वेन

8

जो लोग एनम मूल्यों को संशोधित करना चाहते हैं, उनके लिए इसे फिर से बनाना एकमात्र व्यवहार्य और सुरक्षित समाधान लगता है।

इसमें अस्थायी रूप से एनम कॉलम को एक स्ट्रिंग प्रारूप में बदलना, एनम को फिर से बनाना और फिर स्ट्रिंग कॉलम को एनम प्रकार में वापस जोड़ना है।

यहाँ एक उदाहरण है:

ALTER TABLE your_schema.your_table ALTER COLUMN your_column TYPE varchar(255);
ALTER TABLE your_schema.your_table ALTER COLUMN your_column SET DEFAULT('your_default_enum_value');
DROP TYPE your_schema.your_enum_name;
CREATE TYPE your_schema.your_enum_name AS ENUM ('enum1', 'enum2', 'enum3');
ALTER TABLE your_schema.your_table ALTER your_column DROP DEFAULT;
ALTER TABLE your_schema.your_table ALTER COLUMN your_column TYPE your_schema.your_enum_name USING your_enum_name::your_schema.your_column;
ALTER TABLE your_schema.your_table ALTER COLUMN your_column SET DEFAULT('your_default_enum_value');

ALTER TABLE your_schema.your_table ALTER COLUMN your_column TYPE your_schema.your_enum_name USING your_enum_name::your_schema.your_column;होना चाहिएALTER TABLE your_schema.your_table ALTER COLUMN your_column TYPE your_schema.your_enum_name USING your_schema.your_column::your_enum_name;
मैनुअल दारव्यू

7

Postgresql प्रकार से ENUM मान हटाने के लिए निम्नलिखित क्वेरी का उपयोग करें

DELETE FROM pg_enum
WHERE enumlabel = 'moderator'
AND enumtypid = ( SELECT oid FROM pg_type WHERE typname = 'admin_level1');

बस किस प्रकार की जानकारी है और क्या मूल्य है

DELETE FROM pg_enum
WHERE enumlabel = 'ENUM_VALUE'
AND enumtypid = ( SELECT oid FROM pg_type WHERE typname = 'ENUM_TYPE')

आपको मौजूदा मानों को अन्य में बदलना चाहिए। उसके लिए यदि आपको नया मान जोड़ने की आवश्यकता है, तो उपयोग करें:

ALTER TYPE **ENUM_TYPE** ADD VALUE '**ENUM_VALUE2**'; 

हटाने से पहले, नए प्रकार के मूल्य या मौजूदा मूल्य के प्रकार को अपडेट करें।


केवल समस्या है टाइपनाम pg_type में निचला मामला है। इसलिए इसका काम नहीं कर रहा है, जब तक कि लोअर केस enum_type का उपयोग नहीं कर रहा है SELECT oid FROM pg_type WHERE typname = 'enum_type'
fzerorubigd

2

इसे करने का प्रोग्रामेटिक तरीका इस प्रकार है। Https://stackoverflow.com/a/47305844/629272 में दिए गए समान सामान्य कदम उपयुक्त हैं, लेकिन वे मेरे उद्देश्यों के लिए बनाये गए अर्थ से अधिक मैनुअल हैं (एक नीच डाउन माइग्रेशन लिखना)। my_type,, my_type_oldऔर value_to_delete, ज़ाहिर है, उचित रूप में बदला जाना चाहिए।

  1. अपने प्रकार का नाम बदलें।

    ALTER TYPE my_type RENAME TO my_type_old;
  2. अपने पुराने प्रकार से मानों के साथ एक नया प्रकार बनाएं, जिसे आप हटाना चाहते हैं।

    DO $$
    BEGIN
        EXECUTE format(
            'CREATE TYPE my_type AS ENUM (%s)',
            (
                SELECT string_agg(quote_literal(value), ',')
                FROM unnest(enum_range(NULL::my_type_old)) value
                WHERE value <> 'value_to_delete'
            )
        );
    END $$;
  3. सभी मौजूदा कॉलम बदलें जो पुराने प्रकार का उपयोग करते हैं नए का उपयोग करें।

    DO $$
    DECLARE
        column_data record;
        table_name varchar(255);
        column_name varchar(255);
    BEGIN
        FOR column_data IN
            SELECT cols.table_name, cols.column_name
                FROM information_schema.columns cols
                WHERE udt_name = 'my_type_old'
        LOOP
            table_name := column_data.table_name;
            column_name := column_data.column_name;
            EXECUTE format(
                '
                    ALTER TABLE %s
                    ALTER COLUMN %s
                    TYPE my_type
                    USING %s::text::my_type;
                ',
                table_name, column_name, column_name
            );
        END LOOP;
    END $$;
  4. पुराने प्रकार को हटाएं।

    DROP TYPE my_type_old;

0

यदि आपका डेटासेट इतना बड़ा नहीं है तो आप --column-insertsटेक्स्ट एडिटर के साथ डंप को एडिट कर सकते हैं , वैल्यू हटा सकते हैं और डंप को री-इम्पोर्ट कर सकते हैं


0

V.10 में भी यही समस्या थी। postgres। विलोपन के लिए कुछ प्रक्रियाओं की आवश्यकता होती है, और यदि अनुक्रम सही नहीं है, तो पढ़ने के लिए तालिका के लॉक होने का एक मौका भी होगा।

हटाने के लिए एक सुविधाजनक स्क्रिप्ट लिखी। पहले से ही कई बार अपने प्रदर्शन को साबित किया। हालाँकि, इस प्रक्रिया में हटाए गए मान को नए के साथ बदलना शामिल है (यदि तालिका फ़ील्ड इसे अनुमति देती है तो यह पूर्ण हो सकती है)।

उपयोग करने के लिए, आपको बस 3 मान भरने होंगे।

DO $$
DECLARE
    enumTypeName VARCHAR := 'enum_name'; -- VALUE #1, set yor value!
    enumOldFieldValue varchar := 'old_enum_value'; -- VALUE #2, enum value which have to be deleted
    enumNewFieldValue varchar := null; -- VALUE #3, which new value must be instead of deleted
    sql varchar:='';
    rec record;
BEGIN
    raise info 'Check on old and new enum values.';
    IF exists(select * FROM pg_enum -- check existing of OLD enum value
              WHERE enumtypid = (select oid from pg_type where typName=cast(enumTypeName as varchar) limit 1) and enumlabel=cast(enumOldFieldValue as varchar))
      AND
       (exists(select *
               FROM pg_enum -- check existing of NEW enum value
               WHERE enumtypid = (select oid from pg_type where typName = cast(enumTypeName as varchar) limit 1)
                 and enumlabel = cast(enumNewFieldValue as varchar))
           OR
        enumNewFieldValue IS NULL)
        THEN
            raise info 'Check passed!';

            -- selecting all tables with schemas which has column with enum relation
            create temporary table tmp_table_names
             as SELECT concat(c.table_schema,'.',c.table_name ) as table_name, c.column_name
                FROM information_schema.columns c
                WHERE c.udt_name = cast(enumTypeName as varchar)
                  and c.table_schema=c.udt_schema and data_type = 'USER-DEFINED';

            -- if we have table(s) that uses such enum
            if exists(select * from tmp_table_names)
                then
                    FOR rec in (select table_name, column_name from tmp_table_names) LOOP
                        sql:= format('UPDATE %1$s set %2$s = %3$L where %2$s=%4$L',rec.table_name, rec.column_name, enumNewFieldValue, enumOldFieldValue);
                        raise info 'Update by looping: %', sql;
                        EXECUTE sql;
                    END LOOP;
            end if;

            -- just after changing all old values in all tables we can delete old enum value
            sql := format('DELETE FROM pg_enum WHERE enumtypid = (select oid from pg_type where typName=%1$L limit 1) and enumlabel=%2$L',enumTypeName,enumOldFieldValue);
            raise info 'Delete enum value: %', sql;
            EXECUTE sql;

            drop table  tmp_table_names;
        ELSE
            raise info 'Old or new enum values is missing.';
    end if;
END $$;
  1. सामग्री सूचीबद्ध करें

-1

ENUM से व्यक्तिगत मूल्य को हटाना संभव नहीं है, DROP का एकमात्र संभव समाधान है और आवश्यक मानों के साथ ENUM को फिर से बनाना।


यह बहुत संभव है, जो आप शायद मतलब है "आधिकारिक तौर पर समर्थित नहीं है"।
रिकुडौ_सेनिन

@Rikudou_Sennin क्या आप एक कोड प्रदान करना चाहेंगे जो ENUM से एक सटीक मान निकाल सकता है?
ज़ायतसेव दिमित्री

2
@ZaytsevDmitry यहाँ आप हैं:DELETE FROM pg_enum WHERE enumlabel='saml' AND enumsortorder=4;
रोमन पोडलिनोव
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.