Commit 3c1e9fd2 authored by Tom Lane's avatar Tom Lane

Fix sample INSTR() functions in the plpgsql documentation.

These functions are stated to be Oracle-compatible, but they weren't.
Yugo Nagata noticed that while our code returns zero for a zero or
negative fourth parameter (occur_index), Oracle throws an error.
Further testing by me showed that there was also a discrepancy in the
interpretation of a negative third parameter (beg_index): Oracle thinks
that a negative beg_index indicates the last place where the target
substring can *begin*, whereas our code thinks it is the last place
where the target can *end*.

Adjust the sample code to behave like Oracle in both these respects.
Also change it to be a CDATA[] section, simplifying copying-and-pasting
out of the documentation source file.  And fix minor problems in the
introductory comment, which wasn't very complete or accurate.

Back-patch to all supported branches.  Although this patch only touches
documentation, we should probably call it out as a bug fix in the next
minor release notes, since users who have adopted the functions will
likely want to update their versions.

Yugo Nagata and Tom Lane

Discussion: https://postgr.es/m/20171229191705.c0b43a8c.nagata@sraoss.co.jp
parent 70d6226e
...@@ -5647,27 +5647,29 @@ $$ LANGUAGE plpgsql STRICT IMMUTABLE; ...@@ -5647,27 +5647,29 @@ $$ LANGUAGE plpgsql STRICT IMMUTABLE;
<primary><function>instr</function> function</primary> <primary><function>instr</function> function</primary>
</indexterm> </indexterm>
<programlisting> <programlisting><![CDATA[
-- --
-- instr functions that mimic Oracle's counterpart -- instr functions that mimic Oracle's counterpart
-- Syntax: instr(string1, string2, [n], [m]) where [] denotes optional parameters. -- Syntax: instr(string1, string2 [, n [, m]])
-- where [] denotes optional parameters.
-- --
-- Searches string1 beginning at the nth character for the mth occurrence -- Search string1, beginning at the nth character, for the mth occurrence
-- of string2. If n is negative, search backwards. If m is not passed, -- of string2. If n is negative, search backwards, starting at the abs(n)'th
-- assume 1 (search starts at first character). -- character from the end of string1.
-- If n is not passed, assume 1 (search starts at first character).
-- If m is not passed, assume 1 (find first occurrence).
-- Returns starting index of string2 in string1, or 0 if string2 is not found.
-- --
CREATE FUNCTION instr(varchar, varchar) RETURNS integer AS $$ CREATE FUNCTION instr(varchar, varchar) RETURNS integer AS $$
DECLARE
pos integer;
BEGIN BEGIN
pos:= instr($1, $2, 1); RETURN instr($1, $2, 1);
RETURN pos;
END; END;
$$ LANGUAGE plpgsql STRICT IMMUTABLE; $$ LANGUAGE plpgsql STRICT IMMUTABLE;
CREATE FUNCTION instr(string varchar, string_to_search varchar, beg_index integer) CREATE FUNCTION instr(string varchar, string_to_search_for varchar,
beg_index integer)
RETURNS integer AS $$ RETURNS integer AS $$
DECLARE DECLARE
pos integer NOT NULL DEFAULT 0; pos integer NOT NULL DEFAULT 0;
...@@ -5676,25 +5678,23 @@ DECLARE ...@@ -5676,25 +5678,23 @@ DECLARE
length integer; length integer;
ss_length integer; ss_length integer;
BEGIN BEGIN
IF beg_index &gt; 0 THEN IF beg_index > 0 THEN
temp_str := substring(string FROM beg_index); temp_str := substring(string FROM beg_index);
pos := position(string_to_search IN temp_str); pos := position(string_to_search_for IN temp_str);
IF pos = 0 THEN IF pos = 0 THEN
RETURN 0; RETURN 0;
ELSE ELSE
RETURN pos + beg_index - 1; RETURN pos + beg_index - 1;
END IF; END IF;
ELSIF beg_index &lt; 0 THEN ELSIF beg_index < 0 THEN
ss_length := char_length(string_to_search); ss_length := char_length(string_to_search_for);
length := char_length(string); length := char_length(string);
beg := length + beg_index - ss_length + 2; beg := length + 1 + beg_index;
WHILE beg &gt; 0 LOOP WHILE beg > 0 LOOP
temp_str := substring(string FROM beg FOR ss_length); temp_str := substring(string FROM beg FOR ss_length);
pos := position(string_to_search IN temp_str); IF string_to_search_for = temp_str THEN
IF pos &gt; 0 THEN
RETURN beg; RETURN beg;
END IF; END IF;
...@@ -5709,7 +5709,7 @@ END; ...@@ -5709,7 +5709,7 @@ END;
$$ LANGUAGE plpgsql STRICT IMMUTABLE; $$ LANGUAGE plpgsql STRICT IMMUTABLE;
CREATE FUNCTION instr(string varchar, string_to_search varchar, CREATE FUNCTION instr(string varchar, string_to_search_for varchar,
beg_index integer, occur_index integer) beg_index integer, occur_index integer)
RETURNS integer AS $$ RETURNS integer AS $$
DECLARE DECLARE
...@@ -5721,39 +5721,32 @@ DECLARE ...@@ -5721,39 +5721,32 @@ DECLARE
length integer; length integer;
ss_length integer; ss_length integer;
BEGIN BEGIN
IF beg_index &gt; 0 THEN IF occur_index <= 0 THEN
beg := beg_index; RAISE 'argument ''%'' is out of range', occur_index
temp_str := substring(string FROM beg_index); USING ERRCODE = '22003';
END IF;
IF beg_index > 0 THEN
beg := beg_index - 1;
FOR i IN 1..occur_index LOOP FOR i IN 1..occur_index LOOP
pos := position(string_to_search IN temp_str);
IF i = 1 THEN
beg := beg + pos - 1;
ELSE
beg := beg + pos;
END IF;
temp_str := substring(string FROM beg + 1); temp_str := substring(string FROM beg + 1);
pos := position(string_to_search_for IN temp_str);
IF pos = 0 THEN
RETURN 0;
END IF;
beg := beg + pos;
END LOOP; END LOOP;
IF pos = 0 THEN RETURN beg;
RETURN 0; ELSIF beg_index < 0 THEN
ELSE ss_length := char_length(string_to_search_for);
RETURN beg;
END IF;
ELSIF beg_index &lt; 0 THEN
ss_length := char_length(string_to_search);
length := char_length(string); length := char_length(string);
beg := length + beg_index - ss_length + 2; beg := length + 1 + beg_index;
WHILE beg &gt; 0 LOOP WHILE beg > 0 LOOP
temp_str := substring(string FROM beg FOR ss_length); temp_str := substring(string FROM beg FOR ss_length);
pos := position(string_to_search IN temp_str); IF string_to_search_for = temp_str THEN
IF pos &gt; 0 THEN
occur_number := occur_number + 1; occur_number := occur_number + 1;
IF occur_number = occur_index THEN IF occur_number = occur_index THEN
RETURN beg; RETURN beg;
END IF; END IF;
...@@ -5768,6 +5761,7 @@ BEGIN ...@@ -5768,6 +5761,7 @@ BEGIN
END IF; END IF;
END; END;
$$ LANGUAGE plpgsql STRICT IMMUTABLE; $$ LANGUAGE plpgsql STRICT IMMUTABLE;
]]>
</programlisting> </programlisting>
</sect2> </sect2>
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment