postgresql/contrib/vacuumlo/vacuumlo.c

246 lines
5.6 KiB
C
Raw Normal View History

/*-------------------------------------------------------------------------
*
* vacuumlo.c
* This removes orphaned large objects from a database.
*
* Portions Copyright (c) 1996-2001, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* IDENTIFICATION
* $Header: /cvsroot/pgsql/contrib/vacuumlo/vacuumlo.c,v 1.10 2001/09/17 02:30:54 inoue Exp $
*
*-------------------------------------------------------------------------
*/
#include <stdio.h>
#include <stdlib.h>
2000-06-16 02:55:34 +08:00
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include "libpq-fe.h"
#include "libpq/libpq-fs.h"
2000-11-22 01:54:21 +08:00
#define atooid(x) ((Oid) strtoul((x), NULL, 10))
#define BUFSIZE 1024
1999-05-26 00:15:34 +08:00
int vacuumlo(char *, int);
/*
2000-11-22 01:54:21 +08:00
* This vacuums LOs of one database. It returns 0 on success, -1 on failure.
*/
1999-05-26 00:15:34 +08:00
int
vacuumlo(char *database, int verbose)
{
1999-05-26 00:15:34 +08:00
PGconn *conn;
PGresult *res,
*res2;
char buf[BUFSIZE];
2000-11-22 01:54:21 +08:00
int matched;
int deleted;
1999-05-26 00:15:34 +08:00
int i;
conn = PQsetdb(NULL, NULL, NULL, NULL, database);
/* check to see that the backend connection was successfully made */
if (PQstatus(conn) == CONNECTION_BAD)
{
2000-11-22 01:54:21 +08:00
fprintf(stderr, "Connection to database '%s' failed:\n", database);
1999-05-26 00:15:34 +08:00
fprintf(stderr, "%s", PQerrorMessage(conn));
2000-11-22 01:54:21 +08:00
PQfinish(conn);
1999-05-26 00:15:34 +08:00
return -1;
}
1999-05-26 00:15:34 +08:00
if (verbose)
fprintf(stdout, "Connected to %s\n", database);
/*
2000-11-22 01:54:21 +08:00
* First we create and populate the LO temp table
1999-05-26 00:15:34 +08:00
*/
buf[0] = '\0';
strcat(buf, "SELECT DISTINCT loid AS lo ");
1999-05-26 00:15:34 +08:00
strcat(buf, "INTO TEMP TABLE vacuum_l ");
strcat(buf, "FROM pg_largeobject ");
2000-11-22 01:54:21 +08:00
res = PQexec(conn, buf);
if (PQresultStatus(res) != PGRES_COMMAND_OK)
{
fprintf(stderr, "Failed to create temp table:\n");
fprintf(stderr, "%s", PQerrorMessage(conn));
PQclear(res);
PQfinish(conn);
return -1;
}
PQclear(res);
2001-03-22 12:01:46 +08:00
2000-11-22 01:54:21 +08:00
/*
* Vacuum the temp table so that planner will generate decent plans
* for the DELETEs below.
*/
buf[0] = '\0';
strcat(buf, "VACUUM ANALYZE vacuum_l ");
res = PQexec(conn, buf);
if (PQresultStatus(res) != PGRES_COMMAND_OK)
{
2000-11-22 01:54:21 +08:00
fprintf(stderr, "Failed to vacuum temp table:\n");
fprintf(stderr, "%s", PQerrorMessage(conn));
PQclear(res);
PQfinish(conn);
return -1;
1999-05-26 00:15:34 +08:00
}
PQclear(res);
/*
2000-11-22 01:54:21 +08:00
* Now find any candidate tables who have columns of type oid.
*
2001-03-22 12:01:46 +08:00
* NOTE: the temp table formed above is ignored, because its real table
* name will be pg_something. Also, pg_largeobject will be ignored.
* If either of these were scanned, obviously we'd end up with nothing
* to delete...
2000-11-22 01:54:21 +08:00
*
2001-03-22 12:01:46 +08:00
* NOTE: the system oid column is ignored, as it has attnum < 1. This
* shouldn't matter for correctness, but it saves time.
1999-05-26 00:15:34 +08:00
*/
buf[0] = '\0';
strcat(buf, "SELECT c.relname, a.attname ");
strcat(buf, "FROM pg_class c, pg_attribute a, pg_type t ");
strcat(buf, "WHERE a.attnum > 0 ");
strcat(buf, " AND a.attrelid = c.oid ");
strcat(buf, " AND a.atttypid = t.oid ");
strcat(buf, " AND t.typname in ('oid', 'lo') ");
2000-11-22 01:54:21 +08:00
strcat(buf, " AND c.relkind = 'r'");
1999-05-26 00:15:34 +08:00
strcat(buf, " AND c.relname NOT LIKE 'pg_%'");
2000-11-22 01:54:21 +08:00
res = PQexec(conn, buf);
if (PQresultStatus(res) != PGRES_TUPLES_OK)
1999-05-26 00:15:34 +08:00
{
2000-11-22 01:54:21 +08:00
fprintf(stderr, "Failed to find OID columns:\n");
fprintf(stderr, "%s", PQerrorMessage(conn));
PQclear(res);
PQfinish(conn);
return -1;
1999-05-26 00:15:34 +08:00
}
2000-11-22 01:54:21 +08:00
1999-05-26 00:15:34 +08:00
for (i = 0; i < PQntuples(res); i++)
{
char *table,
*field;
table = PQgetvalue(res, i, 0);
field = PQgetvalue(res, i, 1);
if (verbose)
2000-11-22 01:54:21 +08:00
fprintf(stdout, "Checking %s in %s\n", field, table);
/*
2001-03-22 12:01:46 +08:00
* We use a DELETE with implicit join for efficiency. This is a
* Postgres-ism and not portable to other DBMSs, but then this
* whole program is a Postgres-ism.
2000-11-22 01:54:21 +08:00
*/
sprintf(buf, "DELETE FROM vacuum_l WHERE lo = \"%s\".\"%s\" ",
table, field);
res2 = PQexec(conn, buf);
1999-05-26 00:15:34 +08:00
if (PQresultStatus(res2) != PGRES_COMMAND_OK)
{
2000-11-22 01:54:21 +08:00
fprintf(stderr, "Failed to check %s in table %s:\n",
field, table);
fprintf(stderr, "%s", PQerrorMessage(conn));
1999-05-26 00:15:34 +08:00
PQclear(res2);
PQclear(res);
PQfinish(conn);
return -1;
}
PQclear(res2);
}
1999-05-26 00:15:34 +08:00
PQclear(res);
2000-11-22 01:54:21 +08:00
/*
2001-03-22 12:01:46 +08:00
* Run the actual deletes in a single transaction. Note that this
2000-11-22 01:54:21 +08:00
* would be a bad idea in pre-7.1 Postgres releases (since rolling
2001-03-22 12:01:46 +08:00
* back a table delete used to cause problems), but it should be safe
* now.
2000-11-22 01:54:21 +08:00
*/
1999-05-26 00:15:34 +08:00
res = PQexec(conn, "begin");
PQclear(res);
/*
* Finally, those entries remaining in vacuum_l are orphans.
*/
buf[0] = '\0';
strcat(buf, "SELECT lo ");
strcat(buf, "FROM vacuum_l");
2000-11-22 01:54:21 +08:00
res = PQexec(conn, buf);
if (PQresultStatus(res) != PGRES_TUPLES_OK)
{
2000-11-22 01:54:21 +08:00
fprintf(stderr, "Failed to read temp table:\n");
fprintf(stderr, "%s", PQerrorMessage(conn));
PQclear(res);
1999-05-26 00:15:34 +08:00
PQfinish(conn);
return -1;
}
2000-11-22 01:54:21 +08:00
1999-05-26 00:15:34 +08:00
matched = PQntuples(res);
2000-11-22 01:54:21 +08:00
deleted = 0;
1999-05-26 00:15:34 +08:00
for (i = 0; i < matched; i++)
{
2000-11-22 01:54:21 +08:00
Oid lo = atooid(PQgetvalue(res, i, 0));
1999-05-26 00:15:34 +08:00
if (verbose)
{
2000-11-22 01:54:21 +08:00
fprintf(stdout, "\rRemoving lo %6u ", lo);
1999-05-26 00:15:34 +08:00
fflush(stdout);
}
if (lo_unlink(conn, lo) < 0)
2000-11-22 01:54:21 +08:00
{
fprintf(stderr, "\nFailed to remove lo %u: ", lo);
fprintf(stderr, "%s", PQerrorMessage(conn));
}
else
deleted++;
}
1999-05-26 00:15:34 +08:00
PQclear(res);
/*
* That's all folks!
*/
res = PQexec(conn, "end");
PQclear(res);
2000-11-22 01:54:21 +08:00
1999-05-26 00:15:34 +08:00
PQfinish(conn);
if (verbose)
2000-11-22 01:54:21 +08:00
fprintf(stdout, "\rRemoved %d large objects from %s.\n",
deleted, database);
1999-05-26 00:15:34 +08:00
return 0;
}
int
main(int argc, char **argv)
{
1999-05-26 00:15:34 +08:00
int verbose = 0;
int arg;
int rc = 0;
if (argc < 2)
{
1999-05-26 00:15:34 +08:00
fprintf(stderr, "Usage: %s [-v] database_name [db2 ... dbn]\n",
argv[0]);
exit(1);
}
1999-05-26 00:15:34 +08:00
for (arg = 1; arg < argc; arg++)
{
if (strcmp("-v", argv[arg]) == 0)
verbose = !verbose;
else
2000-11-22 01:54:21 +08:00
rc += (vacuumlo(argv[arg], verbose) != 0);
1999-05-26 00:15:34 +08:00
}
return rc;
}