2000-06-16 03:05:22 +08:00
|
|
|
/* -------------------------------------------------------------------------
|
|
|
|
* pg_dumplo
|
|
|
|
*
|
2004-11-29 07:49:49 +08:00
|
|
|
* $PostgreSQL: pgsql/contrib/pg_dumplo/main.c,v 1.22 2004/11/28 23:49:49 tgl Exp $
|
2000-06-16 03:05:22 +08:00
|
|
|
*
|
|
|
|
* Karel Zak 1999-2000
|
|
|
|
* -------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
2001-02-10 10:31:31 +08:00
|
|
|
#include "postgres_fe.h"
|
|
|
|
|
2001-03-22 12:01:46 +08:00
|
|
|
#include <errno.h>
|
2002-09-06 05:01:16 +08:00
|
|
|
#include <unistd.h>
|
2000-06-16 03:05:22 +08:00
|
|
|
|
2002-09-06 05:01:16 +08:00
|
|
|
#include "libpq-fe.h"
|
|
|
|
#include "libpq/libpq-fs.h"
|
2000-06-16 03:05:22 +08:00
|
|
|
|
2001-03-22 12:01:46 +08:00
|
|
|
#include "pg_dumplo.h"
|
2000-11-22 08:00:55 +08:00
|
|
|
|
2003-01-08 05:42:38 +08:00
|
|
|
#ifndef HAVE_STRDUP
|
|
|
|
#include "strdup.h"
|
2000-06-16 03:05:22 +08:00
|
|
|
#endif
|
|
|
|
|
2003-01-08 05:42:38 +08:00
|
|
|
#include "getopt_long.h"
|
2003-08-08 05:11:58 +08:00
|
|
|
|
2004-11-28 02:51:08 +08:00
|
|
|
#ifndef HAVE_INT_OPTRESET
|
2003-08-04 08:43:34 +08:00
|
|
|
int optreset;
|
2003-01-08 05:42:38 +08:00
|
|
|
#endif
|
2000-06-16 03:05:22 +08:00
|
|
|
|
2001-03-22 12:01:46 +08:00
|
|
|
char *progname = NULL;
|
|
|
|
|
|
|
|
int main(int argc, char **argv);
|
2000-06-16 03:05:22 +08:00
|
|
|
static void usage(void);
|
2001-03-22 12:01:46 +08:00
|
|
|
static void parse_lolist(LODumpMaster * pgLO);
|
|
|
|
|
2000-06-16 03:05:22 +08:00
|
|
|
|
|
|
|
/*-----
|
|
|
|
* The mother of all C functions
|
|
|
|
*-----
|
|
|
|
*/
|
2001-03-22 12:01:46 +08:00
|
|
|
int
|
2000-06-16 03:05:22 +08:00
|
|
|
main(int argc, char **argv)
|
2001-03-22 12:01:46 +08:00
|
|
|
{
|
|
|
|
LODumpMaster _pgLO,
|
|
|
|
*pgLO = &_pgLO;
|
|
|
|
char *pwd = NULL;
|
|
|
|
|
|
|
|
pgLO->argv = argv;
|
|
|
|
pgLO->argc = argc;
|
|
|
|
pgLO->action = 0;
|
|
|
|
pgLO->lolist = NULL;
|
|
|
|
pgLO->user = NULL;
|
|
|
|
pgLO->db = NULL;
|
|
|
|
pgLO->host = NULL;
|
2001-11-13 01:44:14 +08:00
|
|
|
pgLO->port = NULL;
|
2001-03-22 12:01:46 +08:00
|
|
|
pgLO->space = NULL;
|
|
|
|
pgLO->index = NULL;
|
|
|
|
pgLO->remove = FALSE;
|
|
|
|
pgLO->quiet = FALSE;
|
|
|
|
pgLO->counter = 0;
|
2000-06-16 03:05:22 +08:00
|
|
|
pgLO->lolist_start = 0;
|
2001-03-22 12:01:46 +08:00
|
|
|
|
2000-06-16 03:05:22 +08:00
|
|
|
progname = argv[0];
|
|
|
|
|
2001-03-22 14:16:21 +08:00
|
|
|
/*
|
2000-06-16 03:05:22 +08:00
|
|
|
* Parse ARGV
|
|
|
|
*/
|
2001-03-22 12:01:46 +08:00
|
|
|
if (argc > 1)
|
|
|
|
{
|
|
|
|
int arg;
|
|
|
|
extern int optind;
|
|
|
|
int l_index = 0;
|
|
|
|
static struct option l_opt[] = {
|
|
|
|
{"help", no_argument, 0, 'h'},
|
|
|
|
{"user", required_argument, 0, 'u'},
|
|
|
|
{"pwd", required_argument, 0, 'p'},
|
|
|
|
{"db", required_argument, 0, 'd'},
|
|
|
|
{"host", required_argument, 0, 'h'},
|
2001-11-13 01:44:14 +08:00
|
|
|
{"port", required_argument, 0, 'o'},
|
2001-03-22 12:01:46 +08:00
|
|
|
{"space", required_argument, 0, 's'},
|
|
|
|
{"import", no_argument, 0, 'i'},
|
|
|
|
{"export", no_argument, 0, 'e'},
|
|
|
|
{"remove", no_argument, 0, 'r'},
|
|
|
|
{"quiet", no_argument, 0, 'q'},
|
|
|
|
{"all", no_argument, 0, 'a'},
|
|
|
|
{"show", no_argument, 0, 'w'},
|
|
|
|
{NULL, 0, 0, 0}
|
|
|
|
};
|
|
|
|
|
2001-11-13 01:44:14 +08:00
|
|
|
while ((arg = getopt_long(argc, argv, "?aeho:u:p:qd:l:t:irs:w", l_opt, &l_index)) != -1)
|
2001-03-22 12:01:46 +08:00
|
|
|
{
|
|
|
|
switch (arg)
|
|
|
|
{
|
|
|
|
case '?':
|
|
|
|
case 'h':
|
|
|
|
usage();
|
2000-11-22 08:00:55 +08:00
|
|
|
exit(RE_OK);
|
2001-03-22 12:01:46 +08:00
|
|
|
case 'u':
|
|
|
|
pgLO->user = strdup(optarg);
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
pgLO->host = strdup(optarg);
|
|
|
|
break;
|
2001-11-13 01:44:14 +08:00
|
|
|
case 'o':
|
|
|
|
pgLO->port = strdup(optarg);
|
|
|
|
break;
|
2001-03-22 12:01:46 +08:00
|
|
|
case 'p':
|
|
|
|
pwd = strdup(optarg);
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
pgLO->db = strdup(optarg);
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
pgLO->space = strdup(optarg);
|
|
|
|
break;
|
|
|
|
case 'i':
|
|
|
|
pgLO->action = ACTION_IMPORT;
|
|
|
|
break;
|
|
|
|
case 'l':
|
|
|
|
pgLO->action = ACTION_EXPORT_ATTR;
|
|
|
|
pgLO->lolist_start = optind - 1;
|
|
|
|
parse_lolist(pgLO);
|
|
|
|
break;
|
|
|
|
case 'e':
|
|
|
|
case 'a':
|
2000-11-22 08:00:55 +08:00
|
|
|
pgLO->action = ACTION_EXPORT_ALL;
|
|
|
|
break;
|
|
|
|
case 'w':
|
|
|
|
pgLO->action = ACTION_SHOW;
|
2001-03-22 12:01:46 +08:00
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
pgLO->remove = TRUE;
|
|
|
|
break;
|
|
|
|
case 'q':
|
|
|
|
pgLO->quiet = TRUE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
fprintf(stderr, "%s: bad arg -%c\n", progname, arg);
|
|
|
|
usage();
|
|
|
|
exit(RE_ERROR);
|
2000-11-22 08:00:55 +08:00
|
|
|
}
|
2001-03-22 12:01:46 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-06-16 03:05:22 +08:00
|
|
|
usage();
|
2001-03-22 12:01:46 +08:00
|
|
|
exit(RE_ERROR);
|
|
|
|
}
|
2000-06-16 03:05:22 +08:00
|
|
|
|
2001-10-25 13:50:21 +08:00
|
|
|
/*
|
2000-06-16 03:05:22 +08:00
|
|
|
* Check space
|
|
|
|
*/
|
2004-11-29 07:49:49 +08:00
|
|
|
if (pgLO->space==NULL && pgLO->action != ACTION_SHOW)
|
2001-03-22 12:01:46 +08:00
|
|
|
{
|
|
|
|
if (!(pgLO->space = getenv("PWD")))
|
2004-11-29 07:49:49 +08:00
|
|
|
pgLO->space = ".";
|
2000-06-16 03:05:22 +08:00
|
|
|
}
|
2001-03-22 12:01:46 +08:00
|
|
|
|
|
|
|
if (!pgLO->action)
|
|
|
|
{
|
2000-06-19 21:54:50 +08:00
|
|
|
fprintf(stderr, "%s: What do you want - export or import?\n", progname);
|
|
|
|
exit(RE_ERROR);
|
|
|
|
}
|
2001-03-22 12:01:46 +08:00
|
|
|
|
2001-10-25 13:50:21 +08:00
|
|
|
/*
|
2000-06-16 03:05:22 +08:00
|
|
|
* Make connection
|
|
|
|
*/
|
2001-11-13 01:44:14 +08:00
|
|
|
pgLO->conn = PQsetdbLogin(pgLO->host, pgLO->port, NULL, NULL, pgLO->db,
|
2001-03-22 12:01:46 +08:00
|
|
|
pgLO->user, pwd);
|
|
|
|
|
|
|
|
if (PQstatus(pgLO->conn) == CONNECTION_BAD)
|
|
|
|
{
|
2000-11-22 08:00:55 +08:00
|
|
|
fprintf(stderr, "%s (connection): %s\n", progname, PQerrorMessage(pgLO->conn));
|
|
|
|
exit(RE_ERROR);
|
2001-03-22 12:01:46 +08:00
|
|
|
}
|
2000-11-22 08:00:55 +08:00
|
|
|
pgLO->host = PQhost(pgLO->conn) ? PQhost(pgLO->conn) : "localhost";
|
2001-03-22 12:01:46 +08:00
|
|
|
pgLO->db = PQdb(pgLO->conn);
|
|
|
|
pgLO->user = PQuser(pgLO->conn);
|
|
|
|
|
|
|
|
|
2001-10-25 13:50:21 +08:00
|
|
|
/*
|
2000-06-16 03:05:22 +08:00
|
|
|
* Init index file
|
|
|
|
*/
|
2001-03-22 12:01:46 +08:00
|
|
|
if (pgLO->action != ACTION_SHOW)
|
2000-06-16 03:05:22 +08:00
|
|
|
index_file(pgLO);
|
2001-03-22 12:01:46 +08:00
|
|
|
|
2002-10-19 02:41:22 +08:00
|
|
|
PQexec(pgLO->conn, "SET search_path = public");
|
|
|
|
|
2000-06-16 03:05:22 +08:00
|
|
|
PQexec(pgLO->conn, "BEGIN");
|
|
|
|
|
2001-03-22 12:01:46 +08:00
|
|
|
switch (pgLO->action)
|
|
|
|
{
|
|
|
|
|
2000-06-16 03:05:22 +08:00
|
|
|
case ACTION_SHOW:
|
|
|
|
case ACTION_EXPORT_ALL:
|
|
|
|
load_lolist(pgLO);
|
2000-11-22 08:00:55 +08:00
|
|
|
/* FALL THROUGH */
|
2001-03-22 12:01:46 +08:00
|
|
|
|
2000-06-16 03:05:22 +08:00
|
|
|
case ACTION_EXPORT_ATTR:
|
|
|
|
pglo_export(pgLO);
|
2001-03-22 12:01:46 +08:00
|
|
|
if (!pgLO->quiet)
|
|
|
|
{
|
2000-06-16 03:05:22 +08:00
|
|
|
if (pgLO->action == ACTION_SHOW)
|
2000-06-19 21:54:50 +08:00
|
|
|
printf("\nDatabase '%s' contains %d large objects.\n\n", pgLO->db, pgLO->counter);
|
2001-03-22 12:01:46 +08:00
|
|
|
else
|
2000-06-16 03:05:22 +08:00
|
|
|
printf("\nExported %d large objects.\n\n", pgLO->counter);
|
|
|
|
}
|
|
|
|
break;
|
2001-03-22 12:01:46 +08:00
|
|
|
|
2000-06-16 03:05:22 +08:00
|
|
|
case ACTION_IMPORT:
|
|
|
|
pglo_import(pgLO);
|
|
|
|
if (!pgLO->quiet)
|
|
|
|
printf("\nImported %d large objects.\n\n", pgLO->counter);
|
2001-03-22 12:01:46 +08:00
|
|
|
break;
|
|
|
|
}
|
2000-06-16 03:05:22 +08:00
|
|
|
|
|
|
|
PQexec(pgLO->conn, "COMMIT");
|
2001-03-22 12:01:46 +08:00
|
|
|
PQfinish(pgLO->conn);
|
|
|
|
|
2000-06-16 03:05:22 +08:00
|
|
|
if (pgLO->action != ACTION_SHOW)
|
|
|
|
fclose(pgLO->index);
|
2001-03-22 12:01:46 +08:00
|
|
|
|
2000-06-16 03:05:22 +08:00
|
|
|
exit(RE_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2001-03-22 12:01:46 +08:00
|
|
|
parse_lolist(LODumpMaster * pgLO)
|
2000-06-16 03:05:22 +08:00
|
|
|
{
|
2001-03-22 12:01:46 +08:00
|
|
|
LOlist *ll;
|
2004-11-29 07:49:49 +08:00
|
|
|
char **d, *loc, *loc2,
|
|
|
|
buff[MAX_SCHEMA_NAME + MAX_TABLE_NAME + MAX_ATTR_NAME + 3];
|
2000-06-16 03:05:22 +08:00
|
|
|
|
|
|
|
pgLO->lolist = (LOlist *) malloc(pgLO->argc * sizeof(LOlist));
|
2001-03-22 12:01:46 +08:00
|
|
|
|
|
|
|
if (!pgLO->lolist)
|
|
|
|
{
|
2000-06-16 03:05:22 +08:00
|
|
|
fprintf(stderr, "%s: can't allocate memory\n", progname);
|
|
|
|
exit(RE_ERROR);
|
|
|
|
}
|
2001-03-22 12:01:46 +08:00
|
|
|
|
|
|
|
for (d = pgLO->argv + pgLO->lolist_start, ll = pgLO->lolist;
|
|
|
|
*d != NULL;
|
|
|
|
d++, ll++)
|
|
|
|
{
|
|
|
|
|
2004-11-29 07:49:49 +08:00
|
|
|
strncpy(buff, *d, MAX_SCHEMA_NAME + MAX_TABLE_NAME + MAX_ATTR_NAME + 2);
|
2001-03-22 12:01:46 +08:00
|
|
|
|
2004-11-29 07:49:49 +08:00
|
|
|
if ((loc = strchr(buff, '.')) == NULL || *(loc+1)=='\0')
|
2001-03-22 12:01:46 +08:00
|
|
|
{
|
2004-11-29 07:49:49 +08:00
|
|
|
fprintf(stderr, "%s: '%s' is bad 'table.attr' or 'schema.table.attr'\n", progname, buff);
|
2001-03-22 12:01:46 +08:00
|
|
|
exit(RE_ERROR);
|
2000-06-16 03:05:22 +08:00
|
|
|
}
|
2004-11-29 07:49:49 +08:00
|
|
|
loc2 = strchr(loc+1, '.');
|
2000-06-16 03:05:22 +08:00
|
|
|
*loc = '\0';
|
2004-11-29 07:49:49 +08:00
|
|
|
|
|
|
|
if (loc2)
|
|
|
|
{
|
|
|
|
/* "schema.table.attr"
|
|
|
|
*/
|
|
|
|
*loc2 = '\0';
|
|
|
|
ll->lo_schema = strdup(buff);
|
|
|
|
ll->lo_table = strdup(loc+1);
|
|
|
|
ll->lo_attr = strdup(loc2+1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ll->lo_schema = strdup("public");
|
|
|
|
ll->lo_table = strdup(buff);
|
|
|
|
ll->lo_attr = strdup(loc+1);
|
|
|
|
}
|
2000-06-16 03:05:22 +08:00
|
|
|
}
|
|
|
|
ll++;
|
|
|
|
ll->lo_table = ll->lo_attr = (char *) NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-03-22 12:01:46 +08:00
|
|
|
static void
|
2004-10-25 10:15:02 +08:00
|
|
|
usage(void)
|
2000-06-16 03:05:22 +08:00
|
|
|
{
|
2003-01-08 05:42:38 +08:00
|
|
|
printf("\npg_dumplo %s - PostgreSQL large objects dump\n", PG_VERSION);
|
2001-03-22 12:01:46 +08:00
|
|
|
puts("pg_dumplo [option]\n\n"
|
|
|
|
"-h --help this help\n"
|
|
|
|
"-u --user=<username> username for connection to server\n"
|
|
|
|
"-p --password=<password> password for connection to server\n"
|
|
|
|
"-d --db=<database> database name\n"
|
|
|
|
"-t --host=<hostname> server hostname\n"
|
2002-09-05 04:31:48 +08:00
|
|
|
"-o --port=<port> database server port (default: 5432)\n"
|
2001-03-22 12:01:46 +08:00
|
|
|
"-s --space=<dir> directory with dump tree (for export/import)\n"
|
|
|
|
"-i --import import large obj dump tree to DB\n"
|
|
|
|
"-e --export export (dump) large obj to dump tree\n"
|
2004-11-29 07:49:49 +08:00
|
|
|
"-l <schema.table.attr ...> dump attribute (columns) with LO to dump tree\n"
|
2001-03-22 12:01:46 +08:00
|
|
|
"-a --all dump all LO in DB (default)\n"
|
|
|
|
"-r --remove if is set '-i' try remove old LO\n"
|
|
|
|
"-q --quiet run quietly\n"
|
|
|
|
"-w --show not dump, but show all LO in DB\n"
|
2003-01-08 05:42:38 +08:00
|
|
|
"\n"
|
2001-03-22 12:01:46 +08:00
|
|
|
"Example (dump): pg_dumplo -d my_db -s /my_dump/dir -l t1.a t1.b t2.a\n"
|
|
|
|
" pg_dumplo -a -d my_db -s /my_dump/dir\n"
|
|
|
|
"Example (import): pg_dumplo -i -d my_db -s /my_dump/dir\n"
|
|
|
|
"Example (show): pg_dumplo -w -d my_db\n\n"
|
|
|
|
"Note: * option '-l' must be last option!\n"
|
2004-11-29 07:49:49 +08:00
|
|
|
" * default schema is \"public\"\n"
|
|
|
|
" * option '-i' without option '-r' make new large obj in DB\n"
|
2001-03-22 12:01:46 +08:00
|
|
|
" not rewrite old, the '-i' UPDATE oid numbers in table.attr only!\n"
|
2004-11-29 07:49:49 +08:00
|
|
|
" * if option -s is not set, pg_dumplo uses $PWD or \".\"\n"
|
2001-03-22 12:01:46 +08:00
|
|
|
); /* puts() */
|
2000-06-16 03:05:22 +08:00
|
|
|
}
|