postgresql/contrib/oid2name/oid2name.c

648 lines
15 KiB
C
Raw Normal View History

2001-03-22 12:01:46 +08:00
/*
* oid2name, a PostgreSQL app to map OIDs on the filesystem
* to table and database names.
*
* Originally by
* B. Palmer, bpalmer@crimelabs.net 1-17-2001
*
2010-09-21 04:08:53 +08:00
* contrib/oid2name/oid2name.c
*/
#include "postgres_fe.h"
2001-02-12 21:55:36 +08:00
#include <unistd.h>
#ifdef HAVE_GETOPT_H
#include <getopt.h>
#endif
extern char *optarg;
#include "libpq-fe.h"
/* an extensible array to keep track of elements to show */
typedef struct
{
2005-10-15 10:49:52 +08:00
char **array;
int num;
int alloc;
} eary;
/* these are the opts structures for command line params */
2001-03-22 12:01:46 +08:00
struct options
{
eary *tables;
eary *oids;
eary *filenodes;
bool quiet;
bool systables;
bool indexes;
bool nodb;
bool extended;
bool tablespaces;
2005-10-15 10:49:52 +08:00
char *dbname;
char *hostname;
char *port;
char *username;
const char *progname;
};
/* function prototypes */
static void help(const char *progname);
2001-03-22 12:01:46 +08:00
void get_opts(int, char **, struct options *);
void *myalloc(size_t size);
char *mystrdup(const char *str);
void add_one_elt(char *eltname, eary *eary);
char *get_comma_elts(eary *eary);
PGconn *sql_conn(struct options *);
int sql_exec(PGconn *, const char *sql, bool quiet);
void sql_exec_dumpalldbs(PGconn *, struct options *);
void sql_exec_dumpalltables(PGconn *, struct options *);
void sql_exec_searchtables(PGconn *, struct options *);
void sql_exec_dumpalltbspc(PGconn *, struct options *);
/* function to parse command line options and check for some usage errors. */
2001-03-22 12:01:46 +08:00
void
get_opts(int argc, char **argv, struct options * my_opts)
{
2001-11-16 02:40:52 +08:00
int c;
const char *progname;
progname = get_progname(argv[0]);
2001-03-22 12:01:46 +08:00
/* set the defaults */
my_opts->quiet = false;
my_opts->systables = false;
my_opts->indexes = false;
my_opts->nodb = false;
my_opts->extended = false;
my_opts->tablespaces = false;
my_opts->dbname = NULL;
my_opts->hostname = NULL;
my_opts->port = NULL;
my_opts->username = NULL;
my_opts->progname = progname;
if (argc > 1)
{
if (strcmp(argv[1], "--help") == 0 || strcmp(argv[1], "-?") == 0)
{
help(progname);
exit(0);
}
if (strcmp(argv[1], "--version") == 0 || strcmp(argv[1], "-V") == 0)
{
puts("oid2name (PostgreSQL) " PG_VERSION);
exit(0);
}
}
2001-03-22 12:01:46 +08:00
/* get opts */
while ((c = getopt(argc, argv, "H:p:U:d:t:o:f:qSxish")) != -1)
{
2001-03-22 12:01:46 +08:00
switch (c)
{
/* specify the database */
case 'd':
my_opts->dbname = mystrdup(optarg);
2001-03-22 12:01:46 +08:00
break;
/* specify one tablename to show */
2001-03-22 12:01:46 +08:00
case 't':
add_one_elt(optarg, my_opts->tables);
2001-03-22 12:01:46 +08:00
break;
/* specify one Oid to show */
2001-03-22 12:01:46 +08:00
case 'o':
add_one_elt(optarg, my_opts->oids);
break;
2001-03-22 12:01:46 +08:00
2005-10-15 10:49:52 +08:00
/* specify one filenode to show */
case 'f':
add_one_elt(optarg, my_opts->filenodes);
2001-03-22 12:01:46 +08:00
break;
/* don't show headers */
case 'q':
my_opts->quiet = true;
break;
2001-03-22 12:01:46 +08:00
/* host to connect to */
case 'H':
my_opts->hostname = mystrdup(optarg);
2001-03-22 12:01:46 +08:00
break;
/* port to connect to on remote host */
case 'p':
my_opts->port = mystrdup(optarg);
2001-03-22 12:01:46 +08:00
break;
/* username */
case 'U':
my_opts->username = mystrdup(optarg);
2001-03-22 12:01:46 +08:00
break;
/* display system tables */
case 'S':
my_opts->systables = true;
break;
/* also display indexes */
case 'i':
my_opts->indexes = true;
break;
/* display extra columns */
2001-03-22 12:01:46 +08:00
case 'x':
my_opts->extended = true;
break;
/* dump tablespaces only */
case 's':
my_opts->tablespaces = true;
2001-03-22 12:01:46 +08:00
break;
case 'h':
help(progname);
exit(0);
2001-03-22 12:01:46 +08:00
break;
default:
fprintf(stderr, _("Try \"%s --help\" for more information.\n"), progname);
exit(1);
2001-03-22 12:01:46 +08:00
}
}
}
static void
help(const char *progname)
{
printf("%s helps examining the file structure used by PostgreSQL.\n\n"
"Usage:\n"
" %s [OPTION]...\n"
"\nOptions:\n"
" -d DBNAME database to connect to\n"
" -f FILENODE show info for table with given file node\n"
" -H HOSTNAME database server host or socket directory\n"
" -i show indexes and sequences too\n"
" -o OID show info for table with given OID\n"
" -p PORT database server port number\n"
" -q quiet (don't show headers)\n"
" -s show all tablespaces\n"
" -S show system objects too\n"
" -t TABLE show info for named table\n"
" -U NAME connect as specified database user\n"
" -V, --version output version information, then exit\n"
" -x extended (show additional columns)\n"
" -?, --help show this help, then exit\n"
"\nThe default action is to show all database OIDs.\n\n"
"Report bugs to <pgsql-bugs@postgresql.org>.\n",
progname, progname);
}
void *
myalloc(size_t size)
{
2005-10-15 10:49:52 +08:00
void *ptr = malloc(size);
if (!ptr)
2001-03-22 12:01:46 +08:00
{
fprintf(stderr, "out of memory");
exit(1);
2001-03-22 12:01:46 +08:00
}
return ptr;
}
2001-03-22 12:01:46 +08:00
char *
mystrdup(const char *str)
{
2005-10-15 10:49:52 +08:00
char *result = strdup(str);
if (!result)
{
fprintf(stderr, "out of memory");
exit(1);
}
return result;
}
/*
* add_one_elt
*
* Add one element to a (possibly empty) eary struct.
*/
void
add_one_elt(char *eltname, eary *eary)
{
if (eary->alloc == 0)
2001-03-22 12:01:46 +08:00
{
eary ->alloc = 8;
eary ->array = (char **) myalloc(8 * sizeof(char *));
2001-03-22 12:01:46 +08:00
}
else if (eary->num >= eary->alloc)
2001-03-22 12:01:46 +08:00
{
eary ->alloc *= 2;
eary ->array = (char **)
realloc(eary->array, eary->alloc * sizeof(char *));
if (!eary->array)
{
fprintf(stderr, "out of memory");
exit(1);
}
2001-03-22 12:01:46 +08:00
}
eary ->array[eary->num] = mystrdup(eltname);
eary ->num++;
}
/*
* get_comma_elts
*
* Return the elements of an eary as a (freshly allocated) single string, in
* single quotes, separated by commas and properly escaped for insertion in an
* SQL statement.
*/
char *
get_comma_elts(eary *eary)
{
2005-10-15 10:49:52 +08:00
char *ret,
*ptr;
int i,
length = 0;
if (eary->num == 0)
return mystrdup("");
/*
* PQescapeString wants 2 * length + 1 bytes of breath space. Add two
* chars per element for the single quotes and one for the comma.
*/
for (i = 0; i < eary->num; i++)
length += strlen(eary->array[i]);
ret = (char *) myalloc(length * 2 + 4 * eary->num);
ptr = ret;
for (i = 0; i < eary->num; i++)
2001-03-22 12:01:46 +08:00
{
if (i != 0)
sprintf(ptr++, ",");
sprintf(ptr++, "'");
ptr += PQescapeString(ptr, eary->array[i], strlen(eary->array[i]));
sprintf(ptr++, "'");
2001-03-22 12:01:46 +08:00
}
return ret;
}
/* establish connection with database. */
PGconn *
sql_conn(struct options * my_opts)
{
PGconn *conn;
char *password = NULL;
bool new_pass;
/*
* Start the connection. Loop until we have a password if requested by
* backend.
*/
do
2001-03-22 12:01:46 +08:00
{
#define PARAMS_ARRAY_SIZE 7
const char *keywords[PARAMS_ARRAY_SIZE];
const char *values[PARAMS_ARRAY_SIZE];
keywords[0] = "host";
values[0] = my_opts->hostname;
keywords[1] = "port";
values[1] = my_opts->port;
keywords[2] = "user";
values[2] = my_opts->username;
keywords[3] = "password";
values[3] = password;
keywords[4] = "dbname";
values[4] = my_opts->dbname;
keywords[5] = "fallback_application_name";
values[5] = my_opts->progname;
keywords[6] = NULL;
values[6] = NULL;
new_pass = false;
conn = PQconnectdbParams(keywords, values, true);
if (!conn)
{
fprintf(stderr, "%s: could not connect to database %s\n",
"oid2name", my_opts->dbname);
exit(1);
}
if (PQstatus(conn) == CONNECTION_BAD &&
PQconnectionNeedsPassword(conn) &&
password == NULL)
{
PQfinish(conn);
password = simple_prompt("Password: ", 100, false);
new_pass = true;
}
} while (new_pass);
if (password)
free(password);
2001-03-22 12:01:46 +08:00
/* check to see that the backend connection was successfully made */
if (PQstatus(conn) == CONNECTION_BAD)
{
fprintf(stderr, "%s: could not connect to database %s: %s",
"oid2name", my_opts->dbname, PQerrorMessage(conn));
2001-03-22 12:01:46 +08:00
PQfinish(conn);
exit(1);
}
/* return the conn if good */
return conn;
}
/*
* Actual code to make call to the database and print the output data.
*/
2001-03-22 12:01:46 +08:00
int
sql_exec(PGconn *conn, const char *todo, bool quiet)
{
2001-03-22 12:01:46 +08:00
PGresult *res;
int nfields;
int nrows;
2005-10-15 10:49:52 +08:00
int i,
j,
l;
int *length;
char *pad;
2001-03-22 12:01:46 +08:00
/* make the call */
res = PQexec(conn, todo);
/* check and deal with errors */
if (!res || PQresultStatus(res) > 2)
{
fprintf(stderr, "oid2name: query failed: %s\n", PQerrorMessage(conn));
fprintf(stderr, "oid2name: query was: %s\n", todo);
2001-03-22 12:01:46 +08:00
PQclear(res);
PQfinish(conn);
exit(-1);
}
/* get the number of fields */
nrows = PQntuples(res);
nfields = PQnfields(res);
2001-03-22 12:01:46 +08:00
/* for each field, get the needed width */
length = (int *) myalloc(sizeof(int) * nfields);
for (j = 0; j < nfields; j++)
length[j] = strlen(PQfname(res, j));
2001-03-22 12:01:46 +08:00
for (i = 0; i < nrows; i++)
{
for (j = 0; j < nfields; j++)
{
2005-10-15 10:49:52 +08:00
l = strlen(PQgetvalue(res, i, j));
if (l > length[j])
length[j] = strlen(PQgetvalue(res, i, j));
}
}
2001-03-22 12:01:46 +08:00
/* print a header */
if (!quiet)
2001-03-22 12:01:46 +08:00
{
for (j = 0, l = 0; j < nfields; j++)
{
fprintf(stdout, "%*s", length[j] + 2, PQfname(res, j));
l += length[j] + 2;
}
fprintf(stdout, "\n");
pad = (char *) myalloc(l + 1);
MemSet(pad, '-', l);
pad[l] = '\0';
fprintf(stdout, "%s\n", pad);
free(pad);
}
2001-03-22 12:01:46 +08:00
/* for each row, dump the information */
for (i = 0; i < nrows; i++)
{
for (j = 0; j < nfields; j++)
fprintf(stdout, "%*s", length[j] + 2, PQgetvalue(res, i, j));
fprintf(stdout, "\n");
2001-03-22 12:01:46 +08:00
}
/* cleanup */
2001-03-22 12:01:46 +08:00
PQclear(res);
free(length);
2001-03-22 12:01:46 +08:00
return 0;
}
/*
2005-10-15 10:49:52 +08:00
* Dump all databases. There are no system objects to worry about.
*/
2001-03-22 12:01:46 +08:00
void
2005-10-15 10:49:52 +08:00
sql_exec_dumpalldbs(PGconn *conn, struct options * opts)
{
char todo[1024];
2001-03-22 12:01:46 +08:00
/* get the oid and database name from the system pg_database table */
snprintf(todo, sizeof(todo),
"SELECT d.oid AS \"Oid\", datname AS \"Database Name\", "
2010-02-26 10:01:40 +08:00
"spcname AS \"Tablespace\" FROM pg_catalog.pg_database d JOIN pg_catalog.pg_tablespace t ON "
"(dattablespace = t.oid) ORDER BY 2");
sql_exec(conn, todo, opts->quiet);
}
2005-10-15 10:49:52 +08:00
/*
* Dump all tables, indexes and sequences in the current database.
*/
2001-03-22 12:01:46 +08:00
void
2005-10-15 10:49:52 +08:00
sql_exec_dumpalltables(PGconn *conn, struct options * opts)
{
char todo[1024];
char *addfields = ",c.oid AS \"Oid\", nspname AS \"Schema\", spcname as \"Tablespace\" ";
snprintf(todo, sizeof(todo),
2010-02-26 10:01:40 +08:00
"SELECT pg_catalog.pg_relation_filenode(c.oid) as \"Filenode\", relname as \"Table Name\" %s "
"FROM pg_class c "
2005-10-15 10:49:52 +08:00
" LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace "
2010-02-26 10:01:40 +08:00
" LEFT JOIN pg_catalog.pg_database d ON d.datname = pg_catalog.current_database(),"
" pg_catalog.pg_tablespace t "
"WHERE relkind IN ('r'%s%s) AND "
" %s"
" t.oid = CASE"
" WHEN reltablespace <> 0 THEN reltablespace"
" ELSE dattablespace"
" END "
"ORDER BY relname",
opts->extended ? addfields : "",
opts->indexes ? ", 'i', 'S'" : "",
opts->systables ? ", 't'" : "",
opts->systables ? "" : "n.nspname NOT IN ('pg_catalog', 'information_schema') AND n.nspname !~ '^pg_toast' AND");
sql_exec(conn, todo, opts->quiet);
}
/*
* Show oid, filenode, name, schema and tablespace for each of the
* given objects in the current database.
*/
2001-03-22 12:01:46 +08:00
void
2005-10-15 10:49:52 +08:00
sql_exec_searchtables(PGconn *conn, struct options * opts)
{
char *todo;
2005-10-15 10:49:52 +08:00
char *qualifiers,
*ptr;
char *comma_oids,
*comma_filenodes,
*comma_tables;
bool written = false;
char *addfields = ",c.oid AS \"Oid\", nspname AS \"Schema\", spcname as \"Tablespace\" ";
/* get tables qualifiers, whether names, filenodes, or OIDs */
comma_oids = get_comma_elts(opts->oids);
comma_tables = get_comma_elts(opts->tables);
comma_filenodes = get_comma_elts(opts->filenodes);
/* 80 extra chars for SQL expression */
qualifiers = (char *) myalloc(strlen(comma_oids) + strlen(comma_tables) +
strlen(comma_filenodes) + 80);
ptr = qualifiers;
if (opts->oids->num > 0)
{
ptr += sprintf(ptr, "c.oid IN (%s)", comma_oids);
written = true;
}
if (opts->filenodes->num > 0)
{
if (written)
ptr += sprintf(ptr, " OR ");
ptr += sprintf(ptr, "pg_catalog.pg_relation_filenode(c.oid) IN (%s)", comma_filenodes);
written = true;
}
if (opts->tables->num > 0)
{
if (written)
ptr += sprintf(ptr, " OR ");
sprintf(ptr, "c.relname ~~ ANY (ARRAY[%s])", comma_tables);
}
free(comma_oids);
free(comma_tables);
free(comma_filenodes);
/* now build the query */
todo = (char *) myalloc(650 + strlen(qualifiers));
snprintf(todo, 650 + strlen(qualifiers),
2010-02-26 10:01:40 +08:00
"SELECT pg_catalog.pg_relation_filenode(c.oid) as \"Filenode\", relname as \"Table Name\" %s\n"
"FROM pg_catalog.pg_class c \n"
2005-10-15 10:49:52 +08:00
" LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace \n"
" LEFT JOIN pg_catalog.pg_database d ON d.datname = pg_catalog.current_database(),\n"
" pg_catalog.pg_tablespace t \n"
"WHERE relkind IN ('r', 'i', 'S', 't') AND \n"
" t.oid = CASE\n"
" WHEN reltablespace <> 0 THEN reltablespace\n"
" ELSE dattablespace\n"
" END AND \n"
" (%s) \n"
"ORDER BY relname\n",
opts->extended ? addfields : "",
qualifiers);
free(qualifiers);
sql_exec(conn, todo, opts->quiet);
}
2001-03-22 12:01:46 +08:00
void
2005-10-15 10:49:52 +08:00
sql_exec_dumpalltbspc(PGconn *conn, struct options * opts)
{
2005-10-15 10:49:52 +08:00
char todo[1024];
snprintf(todo, sizeof(todo),
"SELECT oid AS \"Oid\", spcname as \"Tablespace Name\"\n"
"FROM pg_catalog.pg_tablespace");
sql_exec(conn, todo, opts->quiet);
}
2001-03-22 12:01:46 +08:00
int
main(int argc, char **argv)
{
2001-03-22 12:01:46 +08:00
struct options *my_opts;
PGconn *pgconn;
my_opts = (struct options *) myalloc(sizeof(struct options));
my_opts->oids = (eary *) myalloc(sizeof(eary));
my_opts->tables = (eary *) myalloc(sizeof(eary));
my_opts->filenodes = (eary *) myalloc(sizeof(eary));
my_opts->oids->num = my_opts->oids->alloc = 0;
my_opts->tables->num = my_opts->tables->alloc = 0;
my_opts->filenodes->num = my_opts->filenodes->alloc = 0;
2001-03-22 12:01:46 +08:00
/* parse the opts */
get_opts(argc, argv, my_opts);
if (my_opts->dbname == NULL)
{
my_opts->dbname = "postgres";
my_opts->nodb = true;
}
pgconn = sql_conn(my_opts);
/* display only tablespaces */
if (my_opts->tablespaces)
2001-03-22 12:01:46 +08:00
{
if (!my_opts->quiet)
printf("All tablespaces:\n");
sql_exec_dumpalltbspc(pgconn, my_opts);
2001-03-22 12:01:46 +08:00
PQfinish(pgconn);
exit(0);
2001-03-22 12:01:46 +08:00
}
/* display the given elements in the database */
if (my_opts->oids->num > 0 ||
my_opts->tables->num > 0 ||
my_opts->filenodes->num > 0)
2001-03-22 12:01:46 +08:00
{
if (!my_opts->quiet)
printf("From database \"%s\":\n", my_opts->dbname);
sql_exec_searchtables(pgconn, my_opts);
2001-03-22 12:01:46 +08:00
PQfinish(pgconn);
exit(0);
2001-03-22 12:01:46 +08:00
}
/* no elements given; dump the given database */
if (my_opts->dbname && !my_opts->nodb)
2001-03-22 12:01:46 +08:00
{
if (!my_opts->quiet)
printf("From database \"%s\":\n", my_opts->dbname);
sql_exec_dumpalltables(pgconn, my_opts);
2001-03-22 12:01:46 +08:00
PQfinish(pgconn);
exit(0);
2001-03-22 12:01:46 +08:00
}
/* no database either; dump all databases */
if (!my_opts->quiet)
printf("All databases:\n");
sql_exec_dumpalldbs(pgconn, my_opts);
2001-03-22 12:01:46 +08:00
PQfinish(pgconn);
return 0;
}