LCOV - code coverage report
Current view: top level - src/bin/psql - prompt.c (source / functions) Coverage Total Hit
Test: PostgreSQL 19devel Lines: 33.3 % 216 72
Test Date: 2026-02-17 17:20:33 Functions: 100.0 % 1 1
Legend: Lines:     hit not hit

            Line data    Source code
       1              : /*
       2              :  * psql - the PostgreSQL interactive terminal
       3              :  *
       4              :  * Copyright (c) 2000-2026, PostgreSQL Global Development Group
       5              :  *
       6              :  * src/bin/psql/prompt.c
       7              :  */
       8              : #include "postgres_fe.h"
       9              : 
      10              : #ifdef WIN32
      11              : #include <io.h>
      12              : #include <win32.h>
      13              : #endif
      14              : 
      15              : #include "common.h"
      16              : #include "common/string.h"
      17              : #include "input.h"
      18              : #include "libpq/pqcomm.h"
      19              : #include "prompt.h"
      20              : #include "settings.h"
      21              : 
      22              : /*--------------------------
      23              :  * get_prompt
      24              :  *
      25              :  * Returns a statically allocated prompt made by interpolating certain
      26              :  * tcsh style escape sequences into pset.vars "PROMPT1|2|3".
      27              :  * (might not be completely multibyte safe)
      28              :  *
      29              :  * Defined interpolations are:
      30              :  * %M - database server "hostname.domainname", "[local]" for AF_UNIX
      31              :  *      sockets, "[local:/dir/name]" if not default
      32              :  * %m - like %M, but hostname only (before first dot), or always "[local]"
      33              :  * %p - backend pid
      34              :  * %P - pipeline status: on, off or abort
      35              :  * %> - database server port number
      36              :  * %n - database user name
      37              :  * %S - search_path
      38              :  * %s - service
      39              :  * %/ - current database
      40              :  * %~ - like %/ but "~" when database name equals user name
      41              :  * %w - whitespace of the same width as the most recent output of PROMPT1
      42              :  * %# - "#" if superuser, ">" otherwise
      43              :  * %R - in prompt1 normally =, or ^ if single line mode,
      44              :  *          or a ! if session is not connected to a database;
      45              :  *      in prompt2 -, *, ', or ";
      46              :  *      in prompt3 nothing
      47              :  * %i - "standby" or "primary" depending on the server's in_hot_standby
      48              :  *      status, or "?" if unavailable (empty if unknown)
      49              :  * %x - transaction status: empty, *, !, ? (unknown or no connection)
      50              :  * %l - The line number inside the current statement, starting from 1.
      51              :  * %? - the error code of the last query (not yet implemented)
      52              :  * %% - a percent sign
      53              :  *
      54              :  * %[0-9]          - the character with the given decimal code
      55              :  * %0[0-7]         - the character with the given octal code
      56              :  * %0x[0-9A-Fa-f]  - the character with the given hexadecimal code
      57              :  *
      58              :  * %`command`      - The result of executing command in /bin/sh with trailing
      59              :  *                   newline stripped.
      60              :  * %:name:         - The value of the psql variable 'name'
      61              :  * (those will not be rescanned for more escape sequences!)
      62              :  *
      63              :  * %[ ... %]       - tell readline that the contained text is invisible
      64              :  *
      65              :  * If the application-wide prompts become NULL somehow, the returned string
      66              :  * will be empty (not NULL!).
      67              :  *--------------------------
      68              :  */
      69              : 
      70              : char *
      71           67 : get_prompt(promptStatus_t status, ConditionalStack cstack)
      72              : {
      73              : #define MAX_PROMPT_SIZE 256
      74              :     static char destination[MAX_PROMPT_SIZE + 1];
      75              :     char        buf[MAX_PROMPT_SIZE + 1];
      76           67 :     bool        esc = false;
      77              :     const char *p;
      78           67 :     const char *prompt_string = "? ";
      79              :     static size_t last_prompt1_width = 0;
      80              : 
      81           67 :     switch (status)
      82              :     {
      83           65 :         case PROMPT_READY:
      84           65 :             prompt_string = pset.prompt1;
      85           65 :             break;
      86              : 
      87            2 :         case PROMPT_CONTINUE:
      88              :         case PROMPT_SINGLEQUOTE:
      89              :         case PROMPT_DOUBLEQUOTE:
      90              :         case PROMPT_DOLLARQUOTE:
      91              :         case PROMPT_COMMENT:
      92              :         case PROMPT_PAREN:
      93            2 :             prompt_string = pset.prompt2;
      94            2 :             break;
      95              : 
      96            0 :         case PROMPT_COPY:
      97            0 :             prompt_string = pset.prompt3;
      98            0 :             break;
      99              :     }
     100              : 
     101           67 :     destination[0] = '\0';
     102              : 
     103           67 :     for (p = prompt_string;
     104          670 :          *p && strlen(destination) < sizeof(destination) - 1;
     105          603 :          p++)
     106              :     {
     107          603 :         memset(buf, 0, sizeof(buf));
     108          603 :         if (esc)
     109              :         {
     110          268 :             switch (*p)
     111              :             {
     112              :                     /* Current database */
     113           67 :                 case '/':
     114           67 :                     if (pset.db)
     115           67 :                         strlcpy(buf, PQdb(pset.db), sizeof(buf));
     116           67 :                     break;
     117            0 :                 case '~':
     118            0 :                     if (pset.db)
     119              :                     {
     120              :                         const char *var;
     121              : 
     122            0 :                         if (strcmp(PQdb(pset.db), PQuser(pset.db)) == 0 ||
     123            0 :                             ((var = getenv("PGDATABASE")) && strcmp(var, PQdb(pset.db)) == 0))
     124            0 :                             strlcpy(buf, "~", sizeof(buf));
     125              :                         else
     126            0 :                             strlcpy(buf, PQdb(pset.db), sizeof(buf));
     127              :                     }
     128            0 :                     break;
     129              : 
     130              :                     /* Whitespace of the same width as the last PROMPT1 */
     131            0 :                 case 'w':
     132            0 :                     if (pset.db)
     133            0 :                         memset(buf, ' ',
     134            0 :                                Min(last_prompt1_width, sizeof(buf) - 1));
     135            0 :                     break;
     136              : 
     137              :                     /* DB server hostname (long/short) */
     138            0 :                 case 'M':
     139              :                 case 'm':
     140            0 :                     if (pset.db)
     141              :                     {
     142            0 :                         const char *host = PQhost(pset.db);
     143              : 
     144              :                         /* INET socket */
     145            0 :                         if (host && host[0] && !is_unixsock_path(host))
     146              :                         {
     147            0 :                             strlcpy(buf, host, sizeof(buf));
     148            0 :                             if (*p == 'm')
     149            0 :                                 buf[strcspn(buf, ".")] = '\0';
     150              :                         }
     151              :                         /* UNIX socket */
     152              :                         else
     153              :                         {
     154            0 :                             if (!host
     155            0 :                                 || strcmp(host, DEFAULT_PGSOCKET_DIR) == 0
     156            0 :                                 || *p == 'm')
     157            0 :                                 strlcpy(buf, "[local]", sizeof(buf));
     158              :                             else
     159            0 :                                 snprintf(buf, sizeof(buf), "[local:%s]", host);
     160              :                         }
     161              :                     }
     162            0 :                     break;
     163              :                     /* DB server port number */
     164            0 :                 case '>':
     165            0 :                     if (pset.db && PQport(pset.db))
     166            0 :                         strlcpy(buf, PQport(pset.db), sizeof(buf));
     167            0 :                     break;
     168              :                     /* DB server user name */
     169            0 :                 case 'n':
     170            0 :                     if (pset.db)
     171            0 :                         strlcpy(buf, session_username(), sizeof(buf));
     172            0 :                     break;
     173              :                     /* search_path */
     174            0 :                 case 'S':
     175            0 :                     if (pset.db)
     176              :                     {
     177            0 :                         const char *sp = PQparameterStatus(pset.db, "search_path");
     178              : 
     179              :                         /* Use ? for versions that don't report search_path. */
     180            0 :                         strlcpy(buf, sp ? sp : "?", sizeof(buf));
     181              :                     }
     182            0 :                     break;
     183              :                     /* service name */
     184            0 :                 case 's':
     185              :                     {
     186            0 :                         const char *service_name = GetVariable(pset.vars, "SERVICE");
     187              : 
     188            0 :                         if (service_name)
     189            0 :                             strlcpy(buf, service_name, sizeof(buf));
     190              :                     }
     191            0 :                     break;
     192              :                     /* backend pid */
     193            0 :                 case 'p':
     194            0 :                     if (pset.db)
     195              :                     {
     196            0 :                         int         pid = PQbackendPID(pset.db);
     197              : 
     198            0 :                         if (pid)
     199            0 :                             snprintf(buf, sizeof(buf), "%d", pid);
     200              :                     }
     201            0 :                     break;
     202              :                     /* pipeline status */
     203            0 :                 case 'P':
     204            0 :                     if (pset.db)
     205              :                     {
     206            0 :                         PGpipelineStatus status = PQpipelineStatus(pset.db);
     207              : 
     208            0 :                         if (status == PQ_PIPELINE_ON)
     209            0 :                             strlcpy(buf, "on", sizeof(buf));
     210            0 :                         else if (status == PQ_PIPELINE_ABORTED)
     211            0 :                             strlcpy(buf, "abort", sizeof(buf));
     212              :                         else
     213            0 :                             strlcpy(buf, "off", sizeof(buf));
     214              :                     }
     215            0 :                     break;
     216            0 :                 case '0':
     217              :                 case '1':
     218              :                 case '2':
     219              :                 case '3':
     220              :                 case '4':
     221              :                 case '5':
     222              :                 case '6':
     223              :                 case '7':
     224            0 :                     *buf = (char) strtol(p, unconstify(char **, &p), 8);
     225            0 :                     --p;
     226            0 :                     break;
     227           67 :                 case 'R':
     228           67 :                     switch (status)
     229              :                     {
     230           65 :                         case PROMPT_READY:
     231           65 :                             if (cstack != NULL && !conditional_active(cstack))
     232            0 :                                 buf[0] = '@';
     233           65 :                             else if (!pset.db)
     234            0 :                                 buf[0] = '!';
     235           65 :                             else if (!pset.singleline)
     236           65 :                                 buf[0] = '=';
     237              :                             else
     238            0 :                                 buf[0] = '^';
     239           65 :                             break;
     240            1 :                         case PROMPT_CONTINUE:
     241            1 :                             buf[0] = '-';
     242            1 :                             break;
     243            0 :                         case PROMPT_SINGLEQUOTE:
     244            0 :                             buf[0] = '\'';
     245            0 :                             break;
     246            0 :                         case PROMPT_DOUBLEQUOTE:
     247            0 :                             buf[0] = '"';
     248            0 :                             break;
     249            0 :                         case PROMPT_DOLLARQUOTE:
     250            0 :                             buf[0] = '$';
     251            0 :                             break;
     252            0 :                         case PROMPT_COMMENT:
     253            0 :                             buf[0] = '*';
     254            0 :                             break;
     255            1 :                         case PROMPT_PAREN:
     256            1 :                             buf[0] = '(';
     257            1 :                             break;
     258            0 :                         default:
     259            0 :                             buf[0] = '\0';
     260            0 :                             break;
     261              :                     }
     262           67 :                     break;
     263            0 :                 case 'i':
     264            0 :                     if (pset.db)
     265              :                     {
     266            0 :                         const char *hs = PQparameterStatus(pset.db, "in_hot_standby");
     267              : 
     268            0 :                         if (hs)
     269              :                         {
     270            0 :                             if (strcmp(hs, "on") == 0)
     271            0 :                                 strlcpy(buf, "standby", sizeof(buf));
     272              :                             else
     273            0 :                                 strlcpy(buf, "primary", sizeof(buf));
     274              :                         }
     275              :                         /* Use ? for versions that don't report in_hot_standby */
     276              :                         else
     277            0 :                             buf[0] = '?';
     278              :                     }
     279            0 :                     break;
     280           67 :                 case 'x':
     281           67 :                     if (!pset.db)
     282            0 :                         buf[0] = '?';
     283              :                     else
     284           67 :                         switch (PQtransactionStatus(pset.db))
     285              :                         {
     286           67 :                             case PQTRANS_IDLE:
     287           67 :                                 buf[0] = '\0';
     288           67 :                                 break;
     289            0 :                             case PQTRANS_ACTIVE:
     290              :                             case PQTRANS_INTRANS:
     291            0 :                                 buf[0] = '*';
     292            0 :                                 break;
     293            0 :                             case PQTRANS_INERROR:
     294            0 :                                 buf[0] = '!';
     295            0 :                                 break;
     296            0 :                             default:
     297            0 :                                 buf[0] = '?';
     298            0 :                                 break;
     299              :                         }
     300           67 :                     break;
     301              : 
     302            0 :                 case 'l':
     303            0 :                     snprintf(buf, sizeof(buf), UINT64_FORMAT, pset.stmt_lineno);
     304            0 :                     break;
     305              : 
     306            0 :                 case '?':
     307              :                     /* not here yet */
     308            0 :                     break;
     309              : 
     310           67 :                 case '#':
     311           67 :                     if (is_superuser())
     312           67 :                         buf[0] = '#';
     313              :                     else
     314            0 :                         buf[0] = '>';
     315           67 :                     break;
     316              : 
     317              :                     /* execute command */
     318            0 :                 case '`':
     319              :                     {
     320            0 :                         int         cmdend = strcspn(p + 1, "`");
     321            0 :                         char       *file = pnstrdup(p + 1, cmdend);
     322              :                         FILE       *fd;
     323              : 
     324            0 :                         fflush(NULL);
     325            0 :                         fd = popen(file, "r");
     326            0 :                         if (fd)
     327              :                         {
     328            0 :                             if (fgets(buf, sizeof(buf), fd) == NULL)
     329            0 :                                 buf[0] = '\0';
     330            0 :                             pclose(fd);
     331              :                         }
     332              : 
     333              :                         /* strip trailing newline and carriage return */
     334            0 :                         (void) pg_strip_crlf(buf);
     335              : 
     336            0 :                         free(file);
     337            0 :                         p += cmdend + 1;
     338            0 :                         break;
     339              :                     }
     340              : 
     341              :                     /* interpolate variable */
     342            0 :                 case ':':
     343              :                     {
     344            0 :                         int         nameend = strcspn(p + 1, ":");
     345            0 :                         char       *name = pnstrdup(p + 1, nameend);
     346              :                         const char *val;
     347              : 
     348            0 :                         val = GetVariable(pset.vars, name);
     349            0 :                         if (val)
     350            0 :                             strlcpy(buf, val, sizeof(buf));
     351            0 :                         free(name);
     352            0 :                         p += nameend + 1;
     353            0 :                         break;
     354              :                     }
     355              : 
     356            0 :                 case '[':
     357              :                 case ']':
     358              : #if defined(USE_READLINE) && defined(RL_PROMPT_START_IGNORE)
     359              : 
     360              :                     /*
     361              :                      * readline >=4.0 undocumented feature: non-printing
     362              :                      * characters in prompt strings must be marked as such, in
     363              :                      * order to properly display the line during editing.
     364              :                      */
     365            0 :                     buf[0] = (*p == '[') ? RL_PROMPT_START_IGNORE : RL_PROMPT_END_IGNORE;
     366            0 :                     buf[1] = '\0';
     367              : #endif                          /* USE_READLINE */
     368            0 :                     break;
     369              : 
     370            0 :                 default:
     371            0 :                     buf[0] = *p;
     372            0 :                     buf[1] = '\0';
     373            0 :                     break;
     374              :             }
     375          268 :             esc = false;
     376              :         }
     377          335 :         else if (*p == '%')
     378          268 :             esc = true;
     379              :         else
     380              :         {
     381           67 :             buf[0] = *p;
     382           67 :             buf[1] = '\0';
     383           67 :             esc = false;
     384              :         }
     385              : 
     386          603 :         if (!esc)
     387          335 :             strlcat(destination, buf, sizeof(destination));
     388              :     }
     389              : 
     390              :     /* Compute the visible width of PROMPT1, for PROMPT2's %w */
     391           67 :     if (prompt_string == pset.prompt1)
     392              :     {
     393           65 :         char       *p = destination;
     394           65 :         char       *end = p + strlen(p);
     395           65 :         bool        visible = true;
     396              : 
     397           65 :         last_prompt1_width = 0;
     398          780 :         while (*p)
     399              :         {
     400              : #if defined(USE_READLINE) && defined(RL_PROMPT_START_IGNORE)
     401          715 :             if (*p == RL_PROMPT_START_IGNORE)
     402              :             {
     403            0 :                 visible = false;
     404            0 :                 ++p;
     405              :             }
     406          715 :             else if (*p == RL_PROMPT_END_IGNORE)
     407              :             {
     408            0 :                 visible = true;
     409            0 :                 ++p;
     410              :             }
     411              :             else
     412              : #endif
     413              :             {
     414              :                 int         chlen,
     415              :                             chwidth;
     416              : 
     417          715 :                 chlen = PQmblen(p, pset.encoding);
     418          715 :                 if (p + chlen > end)
     419            0 :                     break;      /* Invalid string */
     420              : 
     421          715 :                 if (visible)
     422              :                 {
     423          715 :                     chwidth = PQdsplen(p, pset.encoding);
     424              : 
     425          715 :                     if (*p == '\n')
     426            0 :                         last_prompt1_width = 0;
     427          715 :                     else if (chwidth > 0)
     428          715 :                         last_prompt1_width += chwidth;
     429              :                 }
     430              : 
     431          715 :                 p += chlen;
     432              :             }
     433              :         }
     434              :     }
     435              : 
     436           67 :     return destination;
     437              : }
        

Generated by: LCOV version 2.0-1