LCOV - differential code coverage report
Current view: top level - src/bin/psql - prompt.c (source / functions) Coverage Total Hit UNC LBC UIC UBC GBC GIC GNC CBC EUB ECB
Current: Differential Code Coverage HEAD vs 15 Lines: 38.0 % 187 71 2 3 14 97 4 13 54 15 14
Current Date: 2023-04-08 15:15:32 Functions: 100.0 % 1 1 1
Baseline: 15
Baseline Date: 2023-04-08 15:09:40
Legend: Lines: hit not hit

           TLA  Line data    Source code
       1                 : /*
       2                 :  * psql - the PostgreSQL interactive terminal
       3                 :  *
       4                 :  * Copyright (c) 2000-2023, 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                 :  * %> - database server port number
      35                 :  * %n - database user name
      36                 :  * %/ - current database
      37                 :  * %~ - like %/ but "~" when database name equals user name
      38                 :  * %w - whitespace of the same width as the most recent output of PROMPT1
      39                 :  * %# - "#" if superuser, ">" otherwise
      40                 :  * %R - in prompt1 normally =, or ^ if single line mode,
      41                 :  *          or a ! if session is not connected to a database;
      42                 :  *      in prompt2 -, *, ', or ";
      43                 :  *      in prompt3 nothing
      44                 :  * %x - transaction status: empty, *, !, ? (unknown or no connection)
      45                 :  * %l - The line number inside the current statement, starting from 1.
      46                 :  * %? - the error code of the last query (not yet implemented)
      47                 :  * %% - a percent sign
      48                 :  *
      49                 :  * %[0-9]          - the character with the given decimal code
      50                 :  * %0[0-7]         - the character with the given octal code
      51                 :  * %0x[0-9A-Fa-f]  - the character with the given hexadecimal code
      52                 :  *
      53                 :  * %`command`      - The result of executing command in /bin/sh with trailing
      54                 :  *                   newline stripped.
      55                 :  * %:name:         - The value of the psql variable 'name'
      56                 :  * (those will not be rescanned for more escape sequences!)
      57                 :  *
      58                 :  * %[ ... %]       - tell readline that the contained text is invisible
      59                 :  *
      60                 :  * If the application-wide prompts become NULL somehow, the returned string
      61                 :  * will be empty (not NULL!).
      62                 :  *--------------------------
      63                 :  */
      64                 : 
      65                 : char *
      66 CBC          53 : get_prompt(promptStatus_t status, ConditionalStack cstack)
      67                 : {
      68                 : #define MAX_PROMPT_SIZE 256
      69                 :     static char destination[MAX_PROMPT_SIZE + 1];
      70                 :     char        buf[MAX_PROMPT_SIZE + 1];
      71              53 :     bool        esc = false;
      72                 :     const char *p;
      73              53 :     const char *prompt_string = "? ";
      74                 :     static size_t last_prompt1_width = 0;
      75                 : 
      76              53 :     switch (status)
      77                 :     {
      78              51 :         case PROMPT_READY:
      79              51 :             prompt_string = pset.prompt1;
      80              51 :             break;
      81                 : 
      82               2 :         case PROMPT_CONTINUE:
      83                 :         case PROMPT_SINGLEQUOTE:
      84                 :         case PROMPT_DOUBLEQUOTE:
      85                 :         case PROMPT_DOLLARQUOTE:
      86                 :         case PROMPT_COMMENT:
      87                 :         case PROMPT_PAREN:
      88               2 :             prompt_string = pset.prompt2;
      89               2 :             break;
      90                 : 
      91 UBC           0 :         case PROMPT_COPY:
      92               0 :             prompt_string = pset.prompt3;
      93               0 :             break;
      94                 :     }
      95                 : 
      96 CBC          53 :     destination[0] = '\0';
      97                 : 
      98              53 :     for (p = prompt_string;
      99             530 :          *p && strlen(destination) < sizeof(destination) - 1;
     100             477 :          p++)
     101                 :     {
     102             477 :         memset(buf, 0, sizeof(buf));
     103             477 :         if (esc)
     104                 :         {
     105             212 :             switch (*p)
     106                 :             {
     107                 :                     /* Current database */
     108              53 :                 case '/':
     109              53 :                     if (pset.db)
     110              53 :                         strlcpy(buf, PQdb(pset.db), sizeof(buf));
     111              53 :                     break;
     112 UBC           0 :                 case '~':
     113               0 :                     if (pset.db)
     114                 :                     {
     115                 :                         const char *var;
     116                 : 
     117               0 :                         if (strcmp(PQdb(pset.db), PQuser(pset.db)) == 0 ||
     118               0 :                             ((var = getenv("PGDATABASE")) && strcmp(var, PQdb(pset.db)) == 0))
     119               0 :                             strlcpy(buf, "~", sizeof(buf));
     120                 :                         else
     121               0 :                             strlcpy(buf, PQdb(pset.db), sizeof(buf));
     122                 :                     }
     123               0 :                     break;
     124                 : 
     125                 :                     /* Whitespace of the same width as the last PROMPT1 */
     126               0 :                 case 'w':
     127               0 :                     if (pset.db)
     128               0 :                         memset(buf, ' ',
     129               0 :                                Min(last_prompt1_width, sizeof(buf) - 1));
     130               0 :                     break;
     131                 : 
     132                 :                     /* DB server hostname (long/short) */
     133               0 :                 case 'M':
     134                 :                 case 'm':
     135               0 :                     if (pset.db)
     136                 :                     {
     137               0 :                         const char *host = PQhost(pset.db);
     138                 : 
     139                 :                         /* INET socket */
     140               0 :                         if (host && host[0] && !is_unixsock_path(host))
     141                 :                         {
     142               0 :                             strlcpy(buf, host, sizeof(buf));
     143               0 :                             if (*p == 'm')
     144               0 :                                 buf[strcspn(buf, ".")] = '\0';
     145                 :                         }
     146                 :                         /* UNIX socket */
     147                 :                         else
     148                 :                         {
     149               0 :                             if (!host
     150               0 :                                 || strcmp(host, DEFAULT_PGSOCKET_DIR) == 0
     151               0 :                                 || *p == 'm')
     152               0 :                                 strlcpy(buf, "[local]", sizeof(buf));
     153                 :                             else
     154               0 :                                 snprintf(buf, sizeof(buf), "[local:%s]", host);
     155                 :                         }
     156                 :                     }
     157               0 :                     break;
     158                 :                     /* DB server port number */
     159               0 :                 case '>':
     160               0 :                     if (pset.db && PQport(pset.db))
     161               0 :                         strlcpy(buf, PQport(pset.db), sizeof(buf));
     162               0 :                     break;
     163                 :                     /* DB server user name */
     164               0 :                 case 'n':
     165               0 :                     if (pset.db)
     166               0 :                         strlcpy(buf, session_username(), sizeof(buf));
     167               0 :                     break;
     168                 :                     /* backend pid */
     169               0 :                 case 'p':
     170               0 :                     if (pset.db)
     171                 :                     {
     172               0 :                         int         pid = PQbackendPID(pset.db);
     173                 : 
     174               0 :                         if (pid)
     175               0 :                             snprintf(buf, sizeof(buf), "%d", pid);
     176                 :                     }
     177               0 :                     break;
     178                 : 
     179               0 :                 case '0':
     180                 :                 case '1':
     181                 :                 case '2':
     182                 :                 case '3':
     183                 :                 case '4':
     184                 :                 case '5':
     185                 :                 case '6':
     186                 :                 case '7':
     187               0 :                     *buf = (char) strtol(p, unconstify(char **, &p), 8);
     188               0 :                     --p;
     189               0 :                     break;
     190 CBC          53 :                 case 'R':
     191                 :                     switch (status)
     192                 :                     {
     193              51 :                         case PROMPT_READY:
     194              51 :                             if (cstack != NULL && !conditional_active(cstack))
     195 UBC           0 :                                 buf[0] = '@';
     196 CBC          51 :                             else if (!pset.db)
     197 UBC           0 :                                 buf[0] = '!';
     198 CBC          51 :                             else if (!pset.singleline)
     199              51 :                                 buf[0] = '=';
     200                 :                             else
     201 UBC           0 :                                 buf[0] = '^';
     202 CBC          51 :                             break;
     203               1 :                         case PROMPT_CONTINUE:
     204               1 :                             buf[0] = '-';
     205               1 :                             break;
     206 UBC           0 :                         case PROMPT_SINGLEQUOTE:
     207               0 :                             buf[0] = '\'';
     208               0 :                             break;
     209               0 :                         case PROMPT_DOUBLEQUOTE:
     210               0 :                             buf[0] = '"';
     211               0 :                             break;
     212               0 :                         case PROMPT_DOLLARQUOTE:
     213               0 :                             buf[0] = '$';
     214               0 :                             break;
     215               0 :                         case PROMPT_COMMENT:
     216               0 :                             buf[0] = '*';
     217               0 :                             break;
     218 CBC           1 :                         case PROMPT_PAREN:
     219               1 :                             buf[0] = '(';
     220               1 :                             break;
     221 UBC           0 :                         default:
     222               0 :                             buf[0] = '\0';
     223               0 :                             break;
     224                 :                     }
     225 CBC          53 :                     break;
     226                 : 
     227              53 :                 case 'x':
     228              53 :                     if (!pset.db)
     229 UBC           0 :                         buf[0] = '?';
     230                 :                     else
     231 CBC          53 :                         switch (PQtransactionStatus(pset.db))
     232                 :                         {
     233              53 :                             case PQTRANS_IDLE:
     234              53 :                                 buf[0] = '\0';
     235              53 :                                 break;
     236 UBC           0 :                             case PQTRANS_ACTIVE:
     237                 :                             case PQTRANS_INTRANS:
     238               0 :                                 buf[0] = '*';
     239               0 :                                 break;
     240               0 :                             case PQTRANS_INERROR:
     241               0 :                                 buf[0] = '!';
     242               0 :                                 break;
     243               0 :                             default:
     244               0 :                                 buf[0] = '?';
     245               0 :                                 break;
     246                 :                         }
     247 CBC          53 :                     break;
     248                 : 
     249 UBC           0 :                 case 'l':
     250               0 :                     snprintf(buf, sizeof(buf), UINT64_FORMAT, pset.stmt_lineno);
     251               0 :                     break;
     252                 : 
     253               0 :                 case '?':
     254                 :                     /* not here yet */
     255               0 :                     break;
     256                 : 
     257 CBC          53 :                 case '#':
     258              53 :                     if (is_superuser())
     259              53 :                         buf[0] = '#';
     260                 :                     else
     261 UBC           0 :                         buf[0] = '>';
     262 CBC          53 :                     break;
     263                 : 
     264                 :                     /* execute command */
     265 UBC           0 :                 case '`':
     266                 :                     {
     267               0 :                         int         cmdend = strcspn(p + 1, "`");
     268               0 :                         char       *file = pnstrdup(p + 1, cmdend);
     269                 :                         FILE       *fd;
     270                 : 
     271 UNC           0 :                         fflush(NULL);
     272               0 :                         fd = popen(file, "r");
     273 UBC           0 :                         if (fd)
     274 EUB             :                         {
     275 UBC           0 :                             if (fgets(buf, sizeof(buf), fd) == NULL)
     276 UIC           0 :                                 buf[0] = '\0';
     277 UBC           0 :                             pclose(fd);
     278 EUB             :                         }
     279                 : 
     280                 :                         /* strip trailing newline and carriage return */
     281 UIC           0 :                         (void) pg_strip_crlf(buf);
     282                 : 
     283 UBC           0 :                         free(file);
     284 UIC           0 :                         p += cmdend + 1;
     285 UBC           0 :                         break;
     286 EUB             :                     }
     287                 : 
     288                 :                     /* interpolate variable */
     289 UIC           0 :                 case ':':
     290                 :                     {
     291 UBC           0 :                         int         nameend = strcspn(p + 1, ":");
     292 UIC           0 :                         char       *name = pnstrdup(p + 1, nameend);
     293 EUB             :                         const char *val;
     294                 : 
     295 UIC           0 :                         val = GetVariable(pset.vars, name);
     296               0 :                         if (val)
     297 UBC           0 :                             strlcpy(buf, val, sizeof(buf));
     298               0 :                         free(name);
     299               0 :                         p += nameend + 1;
     300               0 :                         break;
     301 EUB             :                     }
     302                 : 
     303 UIC           0 :                 case '[':
     304                 :                 case ']':
     305 EUB             : #if defined(USE_READLINE) && defined(RL_PROMPT_START_IGNORE)
     306                 : 
     307                 :                     /*
     308                 :                      * readline >=4.0 undocumented feature: non-printing
     309                 :                      * characters in prompt strings must be marked as such, in
     310                 :                      * order to properly display the line during editing.
     311                 :                      */
     312 UIC           0 :                     buf[0] = (*p == '[') ? RL_PROMPT_START_IGNORE : RL_PROMPT_END_IGNORE;
     313               0 :                     buf[1] = '\0';
     314 EUB             : #endif                          /* USE_READLINE */
     315 UBC           0 :                     break;
     316                 : 
     317               0 :                 default:
     318 UIC           0 :                     buf[0] = *p;
     319 UBC           0 :                     buf[1] = '\0';
     320               0 :                     break;
     321 EUB             :             }
     322 GBC         212 :             esc = false;
     323                 :         }
     324 CBC         265 :         else if (*p == '%')
     325 GIC         212 :             esc = true;
     326 ECB             :         else
     327                 :         {
     328 GIC          53 :             buf[0] = *p;
     329              53 :             buf[1] = '\0';
     330 CBC          53 :             esc = false;
     331 ECB             :         }
     332                 : 
     333 GIC         477 :         if (!esc)
     334             265 :             strlcat(destination, buf, sizeof(destination));
     335 ECB             :     }
     336                 : 
     337                 :     /* Compute the visible width of PROMPT1, for PROMPT2's %w */
     338 GIC          53 :     if (prompt_string == pset.prompt1)
     339                 :     {
     340 CBC          51 :         char       *p = destination;
     341 GIC          51 :         char       *end = p + strlen(p);
     342 CBC          51 :         bool        visible = true;
     343 ECB             : 
     344 CBC          51 :         last_prompt1_width = 0;
     345 GIC         612 :         while (*p)
     346 ECB             :         {
     347                 : #if defined(USE_READLINE) && defined(RL_PROMPT_START_IGNORE)
     348 GIC         561 :             if (*p == RL_PROMPT_START_IGNORE)
     349                 :             {
     350 LBC           0 :                 visible = false;
     351 UIC           0 :                 ++p;
     352 EUB             :             }
     353 GBC         561 :             else if (*p == RL_PROMPT_END_IGNORE)
     354                 :             {
     355 LBC           0 :                 visible = true;
     356 UIC           0 :                 ++p;
     357 EUB             :             }
     358                 :             else
     359                 : #endif
     360                 :             {
     361                 :                 int         chlen,
     362                 :                             chwidth;
     363                 : 
     364 GIC         561 :                 chlen = PQmblen(p, pset.encoding);
     365             561 :                 if (p + chlen > end)
     366 LBC           0 :                     break;      /* Invalid string */
     367 ECB             : 
     368 GBC         561 :                 if (visible)
     369                 :                 {
     370 CBC         561 :                     chwidth = PQdsplen(p, pset.encoding);
     371                 : 
     372             561 :                     if (*p == '\n')
     373 UIC           0 :                         last_prompt1_width = 0;
     374 CBC         561 :                     else if (chwidth > 0)
     375 GBC         561 :                         last_prompt1_width += chwidth;
     376 ECB             :                 }
     377                 : 
     378 GIC         561 :                 p += chlen;
     379                 :             }
     380 ECB             :         }
     381                 :     }
     382                 : 
     383 GIC          53 :     return destination;
     384                 : }
        

Generated by: LCOV version v1.16-55-g56c0a2a