• Tom Lane's avatar
    Improve our ability to regurgitate SQL-syntax function calls. · 40c24bfe
    Tom Lane authored
    The SQL spec calls out nonstandard syntax for certain function calls,
    for example substring() with numeric position info is supposed to be
    spelled "SUBSTRING(string FROM start FOR count)".  We accept many
    of these things, but up to now would not print them in the same format,
    instead simplifying down to "substring"(string, start, count).
    That's long annoyed me because it creates an interoperability
    problem: we're gratuitously injecting Postgres-specific syntax into
    what might otherwise be a perfectly spec-compliant view definition.
    However, the real reason for addressing it right now is to support
    a planned change in the semantics of EXTRACT() a/k/a date_part().
    When we switch that to returning numeric, we'll have the parser
    translate EXTRACT() to some new function name (might as well be
    "extract" if you ask me) and then teach ruleutils.c to reverse-list
    that per SQL spec.  In this way existing calls to date_part() will
    continue to have the old semantics.
    
    To implement this, invent a new CoercionForm value COERCE_SQL_SYNTAX,
    and make the parser insert that rather than COERCE_EXPLICIT_CALL when
    the input has SQL-spec decoration.  (But if the input has the form of
    a plain function call, continue to mark it COERCE_EXPLICIT_CALL, even
    if it's calling one of these functions.)  Then ruleutils.c recognizes
    COERCE_SQL_SYNTAX as a cue to emit SQL call syntax.  It can know
    which decoration to emit using hard-wired knowledge about the
    functions that could be called this way.  (While this solution isn't
    extensible without manual additions, neither is the grammar, so this
    doesn't seem unmaintainable.)  Notice that this solution will
    reverse-list a function call with SQL decoration only if it was
    entered that way; so dump-and-reload will not by itself produce any
    changes in the appearance of views.
    
    This requires adding a CoercionForm field to struct FuncCall.
    (I couldn't resist the temptation to rearrange that struct's
    field order a tad while I was at it.)  FuncCall doesn't appear
    in stored rules, so that change isn't a reason for a catversion
    bump, but I did one anyway because the new enum value for
    CoercionForm fields could confuse old backend code.
    
    Possible future work:
    
    * Perhaps CoercionForm should now be renamed to DisplayForm,
    or something like that, to reflect its more general meaning.
    This'd require touching a couple hundred places, so it's not
    clear it's worth the code churn.
    
    * The SQLValueFunction node type, which was invented partly for
    the same goal of improving SQL-compatibility of view output,
    could perhaps be replaced with regular function calls marked
    with COERCE_SQL_SYNTAX.  It's unclear if this would be a net
    code savings, however.
    
    Discussion: https://postgr.es/m/42b73d2d-da12-ba9f-570a-420e0cce19d9@phystech.edu
    40c24bfe
makefuncs.c 18 KB