mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-31 10:30:33 +03:00 
			
		
		
		
	This option turns off autovacuum, prevents non-super-user connections, and enables oid setting hooks in the backend. The code continues to use the old autoavacuum disable settings for servers with earlier catalog versions. This includes a catalog version bump to identify servers that support the -b option.
		
			
				
	
	
		
			398 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			398 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *	pg_upgrade.h
 | |
|  *
 | |
|  *	Copyright (c) 2010-2011, PostgreSQL Global Development Group
 | |
|  *	contrib/pg_upgrade/pg_upgrade.h
 | |
|  */
 | |
| 
 | |
| #include "postgres.h"
 | |
| 
 | |
| #include <unistd.h>
 | |
| #include <assert.h>
 | |
| #include <dirent.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/time.h>
 | |
| 
 | |
| #include "libpq-fe.h"
 | |
| 
 | |
| /* Allocate for null byte */
 | |
| #define USER_NAME_SIZE		128
 | |
| 
 | |
| #define MAX_STRING			1024
 | |
| #define LINE_ALLOC			4096
 | |
| #define QUERY_ALLOC			8192
 | |
| 
 | |
| #define MIGRATOR_API_VERSION	1
 | |
| 
 | |
| #define MESSAGE_WIDTH		"60"
 | |
| 
 | |
| #define OVERWRITE_MESSAGE	"  %-" MESSAGE_WIDTH "." MESSAGE_WIDTH "s\r"
 | |
| #define GET_MAJOR_VERSION(v)	((v) / 100)
 | |
| 
 | |
| #define ALL_DUMP_FILE		"pg_upgrade_dump_all.sql"
 | |
| /* contains both global db information and CREATE DATABASE commands */
 | |
| #define GLOBALS_DUMP_FILE	"pg_upgrade_dump_globals.sql"
 | |
| #define DB_DUMP_FILE		"pg_upgrade_dump_db.sql"
 | |
| 
 | |
| #ifndef WIN32
 | |
| #define pg_copy_file		copy_file
 | |
| #define pg_mv_file			rename
 | |
| #define pg_link_file		link
 | |
| #define RM_CMD				"rm -f"
 | |
| #define RMDIR_CMD			"rm -rf"
 | |
| #define SCRIPT_EXT			"sh"
 | |
| #else
 | |
| #define pg_copy_file		CopyFile
 | |
| #define pg_mv_file			pgrename
 | |
| #define pg_link_file		win32_pghardlink
 | |
| #define sleep(x)			Sleep(x * 1000)
 | |
| #define RM_CMD				"DEL /q"
 | |
| #define RMDIR_CMD			"RMDIR /s/q"
 | |
| #define SCRIPT_EXT			"bat"
 | |
| #define EXE_EXT				".exe"
 | |
| #endif
 | |
| 
 | |
| #define CLUSTER_NAME(cluster)	((cluster) == &old_cluster ? "old" : \
 | |
| 								 (cluster) == &new_cluster ? "new" : "none")
 | |
| 
 | |
| #define atooid(x)  ((Oid) strtoul((x), NULL, 10))
 | |
| 
 | |
| /* OID system catalog preservation added during PG 9.0 development */
 | |
| #define TABLE_SPACE_SUBDIRS_CAT_VER 201001111
 | |
| /* postmaster/postgres -b (binary_upgrade) flag added during PG 9.1 development */
 | |
| #define BINARY_UPGRADE_SERVER_FLAG_CAT_VER 201104251
 | |
| 
 | |
| /*
 | |
|  * Each relation is represented by a relinfo structure.
 | |
|  */
 | |
| typedef struct
 | |
| {
 | |
| 	char		nspname[NAMEDATALEN];	/* namespace name */
 | |
| 	char		relname[NAMEDATALEN];	/* relation name */
 | |
| 	Oid			reloid;			/* relation oid */
 | |
| 	Oid			relfilenode;	/* relation relfile node */
 | |
| 	char		tablespace[MAXPGPATH];	/* relations tablespace path */
 | |
| } RelInfo;
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
| 	RelInfo    *rels;
 | |
| 	int			nrels;
 | |
| } RelInfoArr;
 | |
| 
 | |
| /*
 | |
|  * The following structure represents a relation mapping.
 | |
|  */
 | |
| typedef struct
 | |
| {
 | |
| 	char		old_dir[MAXPGPATH];
 | |
| 	char		new_dir[MAXPGPATH];
 | |
| 
 | |
| 	/*
 | |
| 	 * old/new relfilenodes might differ for pg_largeobject(_metadata) indexes
 | |
| 	 * due to VACUUM FULL or REINDEX.  Other relfilenodes are preserved.
 | |
| 	 */
 | |
| 	Oid			old_relfilenode;
 | |
| 	Oid			new_relfilenode;
 | |
| 	/* the rest are used only for logging and error reporting */
 | |
| 	char		nspname[NAMEDATALEN];	/* namespaces */
 | |
| 	char		relname[NAMEDATALEN];
 | |
| } FileNameMap;
 | |
| 
 | |
| /*
 | |
|  * Structure to store database information
 | |
|  */
 | |
| typedef struct
 | |
| {
 | |
| 	Oid			db_oid;			/* oid of the database */
 | |
| 	char		db_name[NAMEDATALEN];	/* database name */
 | |
| 	char		db_tblspace[MAXPGPATH]; /* database default tablespace path */
 | |
| 	RelInfoArr	rel_arr;		/* array of all user relinfos */
 | |
| } DbInfo;
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
| 	DbInfo	   *dbs;			/* array of db infos */
 | |
| 	int			ndbs;			/* number of db infos */
 | |
| } DbInfoArr;
 | |
| 
 | |
| /*
 | |
|  * The following structure is used to hold pg_control information.
 | |
|  * Rather than using the backend's control structure we use our own
 | |
|  * structure to avoid pg_control version issues between releases.
 | |
|  */
 | |
| typedef struct
 | |
| {
 | |
| 	uint32		ctrl_ver;
 | |
| 	uint32		cat_ver;
 | |
| 	uint32		logid;
 | |
| 	uint32		nxtlogseg;
 | |
| 	uint32		chkpnt_tli;
 | |
| 	uint32		chkpnt_nxtxid;
 | |
| 	uint32		chkpnt_nxtoid;
 | |
| 	uint32		align;
 | |
| 	uint32		blocksz;
 | |
| 	uint32		largesz;
 | |
| 	uint32		walsz;
 | |
| 	uint32		walseg;
 | |
| 	uint32		ident;
 | |
| 	uint32		index;
 | |
| 	uint32		toast;
 | |
| 	bool		date_is_int;
 | |
| 	bool		float8_pass_by_value;
 | |
| 	char	   *lc_collate;
 | |
| 	char	   *lc_ctype;
 | |
| 	char	   *encoding;
 | |
| } ControlData;
 | |
| 
 | |
| /*
 | |
|  * Enumeration to denote link modes
 | |
|  */
 | |
| typedef enum
 | |
| {
 | |
| 	TRANSFER_MODE_COPY,
 | |
| 	TRANSFER_MODE_LINK
 | |
| } transferMode;
 | |
| 
 | |
| /*
 | |
|  * Enumeration to denote pg_log modes
 | |
|  */
 | |
| typedef enum
 | |
| {
 | |
| 	PG_INFO,
 | |
| 	PG_REPORT,
 | |
| 	PG_WARNING,
 | |
| 	PG_FATAL,
 | |
| 	PG_DEBUG
 | |
| } eLogType;
 | |
| 
 | |
| 
 | |
| typedef long pgpid_t;
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * cluster
 | |
|  *
 | |
|  *	information about each cluster
 | |
|  */
 | |
| typedef struct
 | |
| {
 | |
| 	ControlData controldata;	/* pg_control information */
 | |
| 	DbInfoArr	dbarr;			/* dbinfos array */
 | |
| 	char	   *pgdata;			/* pathname for cluster's $PGDATA directory */
 | |
| 	char	   *bindir;			/* pathname for cluster's executable directory */
 | |
| 	unsigned short port;		/* port number where postmaster is waiting */
 | |
| 	uint32		major_version;	/* PG_VERSION of cluster */
 | |
| 	char		major_version_str[64];	/* string PG_VERSION of cluster */
 | |
| 	Oid			pg_database_oid;	/* OID of pg_database relation */
 | |
| 	char	   *libpath;		/* pathname for cluster's pkglibdir */
 | |
| 	char	   *tablespace_suffix;		/* directory specification */
 | |
| } ClusterInfo;
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *	LogOpts
 | |
| */
 | |
| typedef struct
 | |
| {
 | |
| 	char	   *filename;		/* name of log file (may be /dev/null) */
 | |
| 	FILE	   *fd;				/* log FILE */
 | |
| 	bool		debug;			/* TRUE -> log more information */
 | |
| 	FILE	   *debug_fd;		/* debug-level log FILE */
 | |
| 	bool		verbose;		/* TRUE -> be verbose in messages */
 | |
| } LogOpts;
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *	UserOpts
 | |
| */
 | |
| typedef struct
 | |
| {
 | |
| 	bool		check;			/* TRUE -> ask user for permission to make
 | |
| 								 * changes */
 | |
| 	transferMode transfer_mode; /* copy files or link them? */
 | |
| } UserOpts;
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * OSInfo
 | |
|  */
 | |
| typedef struct
 | |
| {
 | |
| 	const char *progname;		/* complete pathname for this program */
 | |
| 	char	   *exec_path;		/* full path to my executable */
 | |
| 	char	   *user;			/* username for clusters */
 | |
| 	char		cwd[MAXPGPATH]; /* current working directory, used for output */
 | |
| 	char	  **tablespaces;	/* tablespaces */
 | |
| 	int			num_tablespaces;
 | |
| 	char	  **libraries;		/* loadable libraries */
 | |
| 	int			num_libraries;
 | |
| 	pgpid_t		postmasterPID;	/* PID of currently running postmaster */
 | |
| 	ClusterInfo *running_cluster;
 | |
| } OSInfo;
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Global variables
 | |
|  */
 | |
| extern LogOpts log_opts;
 | |
| extern UserOpts user_opts;
 | |
| extern ClusterInfo old_cluster,
 | |
| 			new_cluster;
 | |
| extern OSInfo os_info;
 | |
| extern char scandir_file_pattern[];
 | |
| 
 | |
| 
 | |
| /* check.c */
 | |
| 
 | |
| void		output_check_banner(bool *live_check);
 | |
| void check_old_cluster(bool live_check,
 | |
| 				  char **sequence_script_file_name);
 | |
| void		check_new_cluster(void);
 | |
| void		report_clusters_compatible(void);
 | |
| void		issue_warnings(char *sequence_script_file_name);
 | |
| void		output_completion_banner(char *deletion_script_file_name);
 | |
| void		check_cluster_versions(void);
 | |
| void		check_cluster_compatibility(bool live_check);
 | |
| void		create_script_for_old_cluster_deletion(char **deletion_script_file_name);
 | |
| 
 | |
| 
 | |
| /* controldata.c */
 | |
| 
 | |
| void		get_control_data(ClusterInfo *cluster, bool live_check);
 | |
| void check_control_data(ControlData *oldctrl,
 | |
| 				   ControlData *newctrl);
 | |
| 
 | |
| 
 | |
| /* dump.c */
 | |
| 
 | |
| void		generate_old_dump(void);
 | |
| void		split_old_dump(void);
 | |
| 
 | |
| 
 | |
| /* exec.c */
 | |
| 
 | |
| int exec_prog(bool throw_error,
 | |
| 		  const char *cmd,...);
 | |
| void		verify_directories(void);
 | |
| bool		is_server_running(const char *datadir);
 | |
| void		rename_old_pg_control(void);
 | |
| 
 | |
| 
 | |
| /* file.c */
 | |
| 
 | |
| #ifdef PAGE_CONVERSION
 | |
| typedef const char *(*pluginStartup) (uint16 migratorVersion,
 | |
| 								uint16 *pluginVersion, uint16 newPageVersion,
 | |
| 								   uint16 oldPageVersion, void **pluginData);
 | |
| typedef const char *(*pluginConvertFile) (void *pluginData,
 | |
| 								   const char *dstName, const char *srcName);
 | |
| typedef const char *(*pluginConvertPage) (void *pluginData,
 | |
| 								   const char *dstPage, const char *srcPage);
 | |
| typedef const char *(*pluginShutdown) (void *pluginData);
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
| 	uint16		oldPageVersion; /* Page layout version of the old cluster		*/
 | |
| 	uint16		newPageVersion; /* Page layout version of the new cluster		*/
 | |
| 	uint16		pluginVersion;	/* API version of converter plugin */
 | |
| 	void	   *pluginData;		/* Plugin data (set by plugin) */
 | |
| 	pluginStartup startup;		/* Pointer to plugin's startup function */
 | |
| 	pluginConvertFile convertFile;		/* Pointer to plugin's file converter
 | |
| 										 * function */
 | |
| 	pluginConvertPage convertPage;		/* Pointer to plugin's page converter
 | |
| 										 * function */
 | |
| 	pluginShutdown shutdown;	/* Pointer to plugin's shutdown function */
 | |
| } pageCnvCtx;
 | |
| 
 | |
| const char *setupPageConverter(pageCnvCtx **result);
 | |
| #else
 | |
| /* dummy */
 | |
| typedef void *pageCnvCtx;
 | |
| #endif
 | |
| 
 | |
| int			dir_matching_filenames(const struct dirent * scan_ent);
 | |
| int pg_scandir(const char *dirname, struct dirent *** namelist,
 | |
| 		   int (*selector) (const struct dirent *));
 | |
| const char *copyAndUpdateFile(pageCnvCtx *pageConverter, const char *src,
 | |
| 				  const char *dst, bool force);
 | |
| const char *linkAndUpdateFile(pageCnvCtx *pageConverter, const char *src,
 | |
| 				  const char *dst);
 | |
| 
 | |
| void		check_hard_link(void);
 | |
| 
 | |
| /* function.c */
 | |
| 
 | |
| void		install_support_functions_in_new_db(const char *db_name);
 | |
| void		uninstall_support_functions_from_new_cluster(void);
 | |
| void		get_loadable_libraries(void);
 | |
| void		check_loadable_libraries(void);
 | |
| 
 | |
| /* info.c */
 | |
| 
 | |
| FileNameMap *gen_db_file_maps(DbInfo *old_db,
 | |
| 				 DbInfo *new_db, int *nmaps, const char *old_pgdata,
 | |
| 				 const char *new_pgdata);
 | |
| void		get_db_and_rel_infos(ClusterInfo *cluster);
 | |
| void		free_db_and_rel_infos(DbInfoArr *db_arr);
 | |
| void print_maps(FileNameMap *maps, int n,
 | |
| 		   const char *db_name);
 | |
| 
 | |
| /* option.c */
 | |
| 
 | |
| void		parseCommandLine(int argc, char *argv[]);
 | |
| 
 | |
| /* relfilenode.c */
 | |
| 
 | |
| void		get_pg_database_relfilenode(ClusterInfo *cluster);
 | |
| const char *transfer_all_new_dbs(DbInfoArr *olddb_arr,
 | |
| 				   DbInfoArr *newdb_arr, char *old_pgdata, char *new_pgdata);
 | |
| 
 | |
| 
 | |
| /* tablespace.c */
 | |
| 
 | |
| void		init_tablespaces(void);
 | |
| 
 | |
| 
 | |
| /* server.c */
 | |
| 
 | |
| PGconn	   *connectToServer(ClusterInfo *cluster, const char *db_name);
 | |
| PGresult   *executeQueryOrDie(PGconn *conn, const char *fmt,...);
 | |
| 
 | |
| void		start_postmaster(ClusterInfo *cluster, bool quiet);
 | |
| void		stop_postmaster(bool fast, bool quiet);
 | |
| uint32		get_major_server_version(ClusterInfo *cluster);
 | |
| void		check_for_libpq_envvars(void);
 | |
| 
 | |
| 
 | |
| /* util.c */
 | |
| 
 | |
| char	   *quote_identifier(const char *s);
 | |
| int			get_user_info(char **user_name);
 | |
| void		check_ok(void);
 | |
| void		report_status(eLogType type, const char *fmt,...);
 | |
| void		pg_log(eLogType type, char *fmt,...);
 | |
| void		prep_status(const char *fmt,...);
 | |
| void		check_ok(void);
 | |
| char	   *pg_strdup(const char *s);
 | |
| void	   *pg_malloc(int size);
 | |
| void		pg_free(void *ptr);
 | |
| const char *getErrorText(int errNum);
 | |
| unsigned int str2uint(const char *str);
 | |
| 
 | |
| 
 | |
| /* version.c */
 | |
| 
 | |
| void new_9_0_populate_pg_largeobject_metadata(ClusterInfo *cluster,
 | |
| 										 bool check_mode);
 | |
| 
 | |
| /* version_old_8_3.c */
 | |
| 
 | |
| void		old_8_3_check_for_name_data_type_usage(ClusterInfo *cluster);
 | |
| void		old_8_3_check_for_tsquery_usage(ClusterInfo *cluster);
 | |
| void		old_8_3_rebuild_tsvector_tables(ClusterInfo *cluster, bool check_mode);
 | |
| void		old_8_3_invalidate_hash_gin_indexes(ClusterInfo *cluster, bool check_mode);
 | |
| void old_8_3_invalidate_bpchar_pattern_ops_indexes(ClusterInfo *cluster,
 | |
| 											  bool check_mode);
 | |
| char	   *old_8_3_create_sequence_script(ClusterInfo *cluster);
 |