On Mon, 5 Sep 2016 07:59:02 +0530 Mithun Cy <mithun...@enterprisedb.com> wrote:
> On Aug 31, 2016 1:44 PM, "Victor Wagner" <vi...@wagner.pp.ru> wrote: > > Thanks, I've added this to 7-th (yet unpublished here) version of my > > patch. > Hi victor, just wanted know what your plan for your patch 07. Would > you like to submit it to the community. I have just signed up as a > reviewer for your patch. As community shows an interest with this patch, and it has been included in the current commitfest, I've to submit it. This version of patch includes 1. Most important - a tap test suite (really small enough and has a room for improvements). 2. Compatibility with older versions - now readonly check is skipped if only one hostname is specified in the connect string 3. pg_host and pg_port variables now show host name and port library was connected to. 4. Your fix with compilation of ecpg tests. Patch is attached. -- Victor Wagner <vi...@wagner.pp.ru>
diff --git a/doc/src/sgml/libpq.sgml b/doc/src/sgml/libpq.sgml index f22e3da..835061d 100644 --- a/doc/src/sgml/libpq.sgml +++ b/doc/src/sgml/libpq.sgml @@ -792,7 +792,7 @@ host=localhost port=5432 dbname=mydb connect_timeout=10 <para> The general form for a connection <acronym>URI</acronym> is: <synopsis> -postgresql://[user[:password]@][netloc][:port][/dbname][?param1=value1&...] +postgresql://[user[:password]@][netloc][:port][,netloc[:port]...][/dbname][?param1=value1&...] </synopsis> </para> @@ -809,6 +809,7 @@ postgresql://localhost/mydb postgresql://user@localhost postgresql://user:secret@localhost postgresql://other@localhost/otherdb?connect_timeout=10&application_name=myapp +postgresql://node1,node2:5433,node3:4432,node4/mydb?hostorder=random&readonly=1 </programlisting> Components of the hierarchical part of the <acronym>URI</acronym> can also be given as parameters. For example: @@ -831,7 +832,9 @@ postgresql:///mydb?host=localhost&port=5433 <para> For improved compatibility with JDBC connection <acronym>URI</acronym>s, instances of parameter <literal>ssl=true</literal> are translated into - <literal>sslmode=require</literal>. + <literal>sslmode=require</literal> and + <literal>loadBalanceHosts=true</literal> into + <literal>hostorder=random</literal>. </para> <para> @@ -841,6 +844,10 @@ postgresql:///mydb?host=localhost&port=5433 postgresql://[2001:db8::1234]/database </synopsis> </para> + <para> + There can be serveral host specifications, optionally accompanied + with port, separated by comma. + </para> <para> The host component is interpreted as described for the parameter <xref @@ -881,6 +888,20 @@ postgresql://%2Fvar%2Flib%2Fpostgresql/dbname when <productname>PostgreSQL</> was built). On machines without Unix-domain sockets, the default is to connect to <literal>localhost</>. </para> + <para> + There can be more than one <literal>host</literal> parameter in + the connect string. In this case these hosts would be considered + alternate entries into same database and if connect to first one + fails, library would try to connect second etc. This can be used + for high availability cluster or for load balancing. See + <xref linkend="libpq-connect-hostorder"> parameter. + </para> + <para> + Network host name can be accompanied with port number, separated by + colon. If so, this port number is used only when connected to + this host. If there is no port number, port specified in the + <xref linkend="libpq-connect-port"> parameter would be used. + </para> </listitem> </varlistentry> @@ -942,8 +963,44 @@ postgresql://%2Fvar%2Flib%2Fpostgresql/dbname </para> </listitem> </varlistentry> - + + <varlistentry id="libpq-connect-hostorder" xreflabel="hostorder"> + <term><literal>hostorder</literal></term> + <listitem> + <para> + Specifies how to choose host from list of alternate hosts, + specified in the <xref linkend="libpq-connect-host"> parameter. + </para> + <para> + If value of this argument is <literal>sequential</literal> (the + default) library connects to the hosts in order they specified, + and tries to connect second one only if connection to the first + fails. + </para> + <para> + If value is <literal>random</literal> host to connect is randomly + picked from the list. It allows to balance load between several + cluster nodes. However, currently PostgreSQL doesn't support + multimaster clusters. So, without use of third-party products, + only read-only connections can take advantage from the + load-balancing. See <xref linkend="libpq-connect-readonly"> + </para> + </listitem> + </varlistentry> + <varlistentry id="libpq-connect-readonly" xreflabel="readonly"> + <term><literal>readonly</literal></term> + <listitem> + <para> + If this parameter is 0 (the default), upon successful connection + library checks if host is in recovery state, and if it is so, + tries next host in the connect string. If this parameter is + non-zero, connection to warm standby nodes are considered + successful. + </para> + </listitem> + </varlistentry> <varlistentry id="libpq-connect-port" xreflabel="port"> + <term><literal>port</literal></term> <listitem> <para> @@ -985,7 +1042,6 @@ postgresql://%2Fvar%2Flib%2Fpostgresql/dbname </para> </listitem> </varlistentry> - <varlistentry id="libpq-connect-connect-timeout" xreflabel="connect_timeout"> <term><literal>connect_timeout</literal></term> <listitem> @@ -996,7 +1052,27 @@ postgresql://%2Fvar%2Flib%2Fpostgresql/dbname </para> </listitem> </varlistentry> - + <varlistentry id="libpq-connect-falover-timeout" xreflabel="failover_timeout"> + <term><literal>falover_timeout</literal></term> + <listitem> + <para> + Maximum time to cycilically retry all the hosts in commect string. + (as decimal integer number of seconds). If not specified, then + hosts are tried just once. + </para> + <para> + If we have replicating cluster, and master node fails, it might + take some time to promote one of standby nodes to the new master. + So clients which notice that connect to the master fails, can + already give up attempt to reestablish a connection when new master + became available. + </para> + <para> + Setting this parameter to reasonable time makes library try to + reconnect all the host in cyclically until new master appears. + </para> + </listitem> + </varlistentry> <varlistentry id="libpq-connect-client-encoding" xreflabel="client_encoding"> <term><literal>client_encoding</literal></term> <listitem> @@ -7212,6 +7288,18 @@ user=admin An example file is provided at <filename>share/pg_service.conf.sample</filename>. </para> + <para> + If more than one <literal>host</literal> option present in the section of service file, it + is interpeted as alternate servers for failover or load-balancing. See + <xref linkend="libpq-connect-host"> option in the connect string. + </para> + <para> + For all other options first value takes precedence over later ones. + </para> + <para> + Options, specified in the connect string along with service option + have precedence over values from the service file. + </para> </sect1> diff --git a/src/interfaces/ecpg/test/Makefile.regress b/src/interfaces/ecpg/test/Makefile.regress index b3d7c1e..7bc67a0 100644 --- a/src/interfaces/ecpg/test/Makefile.regress +++ b/src/interfaces/ecpg/test/Makefile.regress @@ -20,7 +20,7 @@ ECPG_TEST_DEPENDENCIES = ../../preproc/ecpg$(X) \ $(srcdir)/../../include/sql3types.h %: %.o - $(CC) $(CFLAGS) $< $(LDFLAGS) $(LDFLAGS_EX) $(LIBS) -o $@ + $(CC) $(CFLAGS) $< $(libpq_pgport) $(LDFLAGS) $(LDFLAGS_EX) $(LIBS) -o $@ # Caution: this build rule is overridden in some child Makefiles # where it's necessary to use nondefault switches to ecpg; diff --git a/src/interfaces/libpq/Makefile b/src/interfaces/libpq/Makefile index 0b4065e..7a5682f 100644 --- a/src/interfaces/libpq/Makefile +++ b/src/interfaces/libpq/Makefile @@ -129,6 +129,9 @@ install: all installdirs install-lib installcheck: $(MAKE) -C test $@ +check: + $(prove_check) + installdirs: installdirs-lib $(MKDIR_P) '$(DESTDIR)$(includedir)' '$(DESTDIR)$(includedir_internal)' '$(DESTDIR)$(datadir)' @@ -142,6 +145,7 @@ uninstall: uninstall-lib clean distclean: clean-lib $(MAKE) -C test $@ rm -f $(OBJS) pthread.h libpq.rc + rm -rf tmp_check # Might be left over from a Win32 client-only build rm -f pg_config_paths.h rm -f inet_net_ntop.c noblock.c pgstrcasecmp.c pqsignal.c thread.c diff --git a/src/interfaces/libpq/fe-connect.c b/src/interfaces/libpq/fe-connect.c index 76b61bd..608fdc0 100644 --- a/src/interfaces/libpq/fe-connect.c +++ b/src/interfaces/libpq/fe-connect.c @@ -299,7 +299,16 @@ static const internalPQconninfoOption PQconninfoOptions[] = { {"replication", NULL, NULL, NULL, "Replication", "D", 5, offsetof(struct pg_conn, replication)}, - + /* Parameters added by failover patch */ + {"hostorder", NULL, "sequential", NULL, + "Host order", "", 10, + offsetof(struct pg_conn, hostorder)}, + {"readonly", NULL, "0", NULL, + "Read only", "", 1, + offsetof(struct pg_conn, read_only)}, + {"failover_timeout", NULL, NULL, NULL, + "Failover Timoeut", "", 10, + offsetof(struct pg_conn, failover_timeout)}, /* Terminating entry --- MUST BE LAST */ {NULL, NULL, NULL, NULL, NULL, NULL, 0} @@ -336,6 +345,7 @@ static PGconn *makeEmptyPGconn(void); static bool fillPGconn(PGconn *conn, PQconninfoOption *connOptions); static void freePGconn(PGconn *conn); static void closePGconn(PGconn *conn); +static void pqTerminateConn(PGconn *conn); static PQconninfoOption *conninfo_init(PQExpBuffer errorMessage); static PQconninfoOption *parse_connection_string(const char *conninfo, PQExpBuffer errorMessage, bool use_defaults); @@ -380,6 +390,7 @@ static bool getPgPassFilename(char *pgpassfile); static void dot_pg_pass_warning(PGconn *conn); static void default_threadlock(int acquire); +static int try_next_address(PGconn *conn); /* global variable because fe-auth.c needs to access it */ pgthreadlock_t pg_g_threadlock = default_threadlock; @@ -806,8 +817,10 @@ connectOptions2(PGconn *conn) { if (conn->pgpass) free(conn->pgpass); - conn->pgpass = PasswordFromFile(conn->pghost, conn->pgport, - conn->dbName, conn->pguser); + conn->pgpass = PasswordFromFile( + conn->actualhost? conn->actualhost:conn->pghost, + conn->actualport? conn->actualport:conn->pgport, + conn->dbName, conn->pguser); if (conn->pgpass == NULL) { conn->pgpass = strdup(DefaultPassword); @@ -1173,6 +1186,8 @@ connectFailureMessage(PGconn *conn, int errorno) if (conn->pghostaddr && conn->pghostaddr[0] != '\0') displayed_host = conn->pghostaddr; + else if (conn->actualhost && conn->actualhost[0] != '\0') + displayed_host = conn->actualhost; else if (conn->pghost && conn->pghost[0] != '\0') displayed_host = conn->pghost; else @@ -1390,11 +1405,17 @@ setKeepalivesWin32(PGconn *conn) static int connectDBStart(PGconn *conn) { + struct nodeinfo + { + char *host; + char *port; + }; int portnum; char portstr[MAXPGPATH]; struct addrinfo *addrs = NULL; struct addrinfo hint; - const char *node; + struct nodeinfo *nodes, + *node; int ret; if (!conn) @@ -1436,21 +1457,78 @@ connectDBStart(PGconn *conn) if (conn->pghostaddr != NULL && conn->pghostaddr[0] != '\0') { /* Using pghostaddr avoids a hostname lookup */ - node = conn->pghostaddr; + + nodes = calloc(sizeof(struct nodeinfo), 2); + nodes->host = strdup(conn->pghostaddr); hint.ai_family = AF_UNSPEC; hint.ai_flags = AI_NUMERICHOST; } else if (conn->pghost != NULL && conn->pghost[0] != '\0') { /* Using pghost, so we have to look-up the hostname */ - node = conn->pghost; + char *p = conn->pghost, + *q, + *r; + int nodecount = 0, + nodesallocated = 4; + /* Parse comma-separated list of host-port pairs into + function-local array of records */ + nodes = malloc(sizeof(struct nodeinfo) * 4); + while (*p) + { + q = p; + r = NULL; + /* Scan for the comma or end of string */ + while (*q != ',' && *q != 0) + { + if (*q == ':') + r = q; + if (*q == ']') + r = NULL; /* if there is IPv6, colons before close + * bracket are part of address */ + q++; + } + if (r) + { + /* Host has explicitely specified port */ + nodes[nodecount].port = malloc(q - r ); + strncpy(nodes[nodecount].port, r + 1 , q - r); + /* FIXME need to check that port is numeric */ + nodes[nodecount].port[q - r - 1] = 0; + } + else + { + r = q; + nodes[nodecount].port = NULL; + } + if ((*p) == '[' && *(r - 1) == ']') + { + /* IPv6 address found. Strip brackets */ + p++; + r--; + } + /* Fill node record */ + nodes[nodecount].host = malloc(r - p + 1); + strncpy(nodes[nodecount].host, p, r - p); + nodes[nodecount].host[r - p] = 0; + /* skip a comma */ + if (*q) + q++; + nodecount++; + if (nodecount == nodesallocated) + nodes = realloc(nodes, sizeof(struct nodeinfo) * (nodesallocated += 4)); + p = q; + } + /* Fill end-of-host list marker */ + nodes[nodecount].host = NULL; + nodes[nodecount].port = NULL; hint.ai_family = AF_UNSPEC; } else { #ifdef HAVE_UNIX_SOCKETS /* pghostaddr and pghost are NULL, so use Unix domain socket */ - node = NULL; + nodes = calloc(sizeof(struct nodeinfo), 2); hint.ai_family = AF_UNIX; UNIXSOCK_PATH(portstr, portnum, conn->pgunixsocket); if (strlen(portstr) >= UNIXSOCK_PATH_BUFLEN) @@ -1460,33 +1538,80 @@ connectDBStart(PGconn *conn) portstr, (int) (UNIXSOCK_PATH_BUFLEN - 1)); conn->options_valid = false; + free(nodes->port); + free(nodes); goto connect_errReturn; } + nodes->port = strdup(portstr); #else /* Without Unix sockets, default to localhost instead */ - node = DefaultHost; + nodes = calloc(sizeof(struct nodeinfo), 2); + nodes->host = strdup(DefaultHost); hint.ai_family = AF_UNSPEC; #endif /* HAVE_UNIX_SOCKETS */ } /* Use pg_getaddrinfo_all() to resolve the address */ - ret = pg_getaddrinfo_all(node, portstr, &hint, &addrs); - if (ret || !addrs) - { - if (node) - appendPQExpBuffer(&conn->errorMessage, - libpq_gettext("could not translate host name \"%s\" to address: %s\n"), - node, gai_strerror(ret)); + /* loop over all the host specs in the node variable */ + for (node = nodes; node->host != NULL || node->port != NULL; node++) + { + struct addrinfo *this_node_addrs; + /* Resolve each hostname into list of addrinfo structures */ + ret = pg_getaddrinfo_all(node->host, (node->port ? node->port : portstr), + &hint, &this_node_addrs); + if (ret || !this_node_addrs) + { + if (node->host) + appendPQExpBuffer(&conn->errorMessage, + libpq_gettext("could not translate host name \"%s\" to address: %s\n"), + node->host, gai_strerror(ret)); + else + appendPQExpBuffer(&conn->errorMessage, + libpq_gettext("could not translate Unix-domain socket path \"%s\" to address: %s\n"), + node->port, gai_strerror(ret)); + if (this_node_addrs) + pg_freeaddrinfo_all(hint.ai_family, this_node_addrs); + + + /* + * We shouldn't fail here unless there is no valid addrinfos left + */ + continue; + } + if (node->host) { + struct addrinfo *n; + for (n=this_node_addrs;n != NULL; n=n->ai_next) { + n->ai_canonname = strdup(node->host); + } + } + /* add this host addrs to addrs field of PGconn structure */ + if (!addrs) + { + addrs = this_node_addrs; + } else - appendPQExpBuffer(&conn->errorMessage, - libpq_gettext("could not translate Unix-domain socket path \"%s\" to address: %s\n"), - portstr, gai_strerror(ret)); - if (addrs) - pg_freeaddrinfo_all(hint.ai_family, addrs); + { + struct addrinfo *p; + + for (p = addrs; p->ai_next != NULL; p = p->ai_next); + p->ai_next = this_node_addrs; + } + } + /* Free nodes array */ + for (node = nodes; node->host != NULL || node->port != NULL; node++) + { + if (node->host) + free(node->host); + if (node->port) + free(node->port); + } + free(nodes); + /* Check if we've found at least one usable address */ + if (!addrs) + { conn->options_valid = false; goto connect_errReturn; } - #ifdef USE_SSL /* setup values based on SSL mode */ if (conn->sslmode[0] == 'd') /* "disable" */ @@ -1499,12 +1624,23 @@ connectDBStart(PGconn *conn) * Set up to try to connect, with protocol 3.0 as the first attempt. */ conn->addrlist = addrs; - conn->addr_cur = addrs; + + /* + * We cannot just assign first addrs record to addr_cur, because host + * order may be random. So, use try_next_address + */ + conn->addr_cur = NULL; + try_next_address(conn); conn->addrlist_family = hint.ai_family; conn->pversion = PG_PROTOCOL(3, 0); conn->send_appname = true; conn->status = CONNECTION_NEEDED; - + if (conn->failover_timeout) { + conn->failover_finish_time = time(NULL) + + atoi(conn->failover_timeout); + } else { + conn->failover_finish_time = (time_t)0; /* it is in past, so its ok */ + } /* * The code for processing CONNECTION_NEEDED state is in PQconnectPoll(), * so that it can easily be re-executed if needed again during the @@ -1522,6 +1658,14 @@ connect_errReturn: } +static char* get_port_from_addrinfo(struct addrinfo *ai) +{ + char port[6]; + sprintf(port,"%d", htons(((struct sockaddr_in *)ai->ai_addr)->sin_port)); + return strdup(port); +} + + /* * connectDBComplete * @@ -1603,6 +1747,98 @@ connectDBComplete(PGconn *conn) } } +/* + * Gets address of pointer to the list of addrinfo sturctures. + * If order is random, rearranges the list by moving random element to + * the beginning (and putting its addres into given pointer. + * Returns address of first list element + */ +static struct addrinfo *get_next_element(struct addrinfo **list,char + *order) { + struct addrinfo *choice = NULL, *prev, *current, *prechoice = NULL; + int count = 0; + if (*list == NULL) return NULL; + if (strcmp(order,"random") == 0) { + /* Peek random element from the list. */ + for (current = *list,prev = NULL; current != NULL; + prev=current, current=current->ai_next) + { + count++; + if ((rand()&0xffff) < 0x10000 / count) + { + choice = current; + prechoice = prev; + } + } + /* If prechoice is not NULL, selected element is not + first in the list. We have to move it to he head + */ + if (prechoice != NULL) + { + prechoice->ai_next=choice->ai_next; + choice->ai_next=*list; + *list=choice; + } + } + /* We always return first element of the list */ + return *list; +} +/* ------------- + * try_next_address + * Attempts to set next address from the list of known ones. + * Returns 1 if address is choosen and 0 if there are no more addresses + * to try + * Takes into account hostorder parameter + * ------------ + */ + +static int try_next_address(PGconn *conn) +{ + if (strcmp(conn->hostorder,"random")==0) { + /* Initialize random number generator in case if nobody have + * done it before. Use value from rand along with time in case + * random number have been initialized by application. + * Use address of conn structure to load-balance different + * connections in the same app + */ + srand((unsigned int)((long int)conn ^ (long int) time(NULL) ^ + (long int)rand())); + } + if (conn->addr_cur == NULL) + { + + conn->addr_cur = get_next_element(&(conn->addrlist), + conn->hostorder); + conn->actualhost = conn->addr_cur->ai_canonname; + + return 1; + } + else + { + conn->addr_cur = get_next_element(&(conn->addr_cur->ai_next), + conn->hostorder); + } + if (conn->addr_cur == NULL && time(NULL) < conn->failover_finish_time) { + /* If failover timeout is set, retry list of hosts from + * the beginning */ + pg_usleep(1000000); + conn->addr_cur = get_next_element(&(conn->addrlist), + conn->hostorder); + } + + if (conn->addr_cur != NULL) { + /* Clean up error message buffer. + */ + resetPQExpBuffer(&conn->errorMessage); + conn->actualhost = conn->addr_cur->ai_canonname; + + return 1; + } else { + return 0; + } + +} + /* ---------------- * PQconnectPoll * @@ -1680,7 +1916,8 @@ PQconnectPoll(PGconn *conn) case CONNECTION_SSL_STARTUP: case CONNECTION_NEEDED: break; - + case CONNECTION_CHECK_RW: + break; default: appendPQExpBufferStr(&conn->errorMessage, libpq_gettext( @@ -1718,9 +1955,8 @@ keep_going: /* We will come back to here until there is * ignore socket() failure if we have more addresses * to try */ - if (addr_cur->ai_next != NULL) + if (try_next_address(conn)) { - conn->addr_cur = addr_cur->ai_next; continue; } appendPQExpBuffer(&conn->errorMessage, @@ -1739,7 +1975,7 @@ keep_going: /* We will come back to here until there is if (!connectNoDelay(conn)) { pqDropConnection(conn, true); - conn->addr_cur = addr_cur->ai_next; + try_next_address(conn); continue; } } @@ -1749,7 +1985,7 @@ keep_going: /* We will come back to here until there is libpq_gettext("could not set socket to nonblocking mode: %s\n"), SOCK_STRERROR(SOCK_ERRNO, sebuf, sizeof(sebuf))); pqDropConnection(conn, true); - conn->addr_cur = addr_cur->ai_next; + try_next_address(conn); continue; } @@ -1760,7 +1996,7 @@ keep_going: /* We will come back to here until there is libpq_gettext("could not set socket to close-on-exec mode: %s\n"), SOCK_STRERROR(SOCK_ERRNO, sebuf, sizeof(sebuf))); pqDropConnection(conn, true); - conn->addr_cur = addr_cur->ai_next; + try_next_address(conn); continue; } #endif /* F_SETFD */ @@ -1807,7 +2043,7 @@ keep_going: /* We will come back to here until there is if (err) { pqDropConnection(conn, true); - conn->addr_cur = addr_cur->ai_next; + try_next_address(conn); continue; } } @@ -1883,6 +2119,8 @@ keep_going: /* We will come back to here until there is * go do the next stuff. */ conn->status = CONNECTION_STARTED; + /* Save the name of the current host + */ goto keep_going; } @@ -1898,7 +2136,7 @@ keep_going: /* We will come back to here until there is /* * Try the next address, if any. */ - conn->addr_cur = addr_cur->ai_next; + try_next_address(conn); } /* loop over addresses */ /* @@ -1944,9 +2182,8 @@ keep_going: /* We will come back to here until there is * If more addresses remain, keep trying, just as in the * case where connect() returned failure immediately. */ - if (conn->addr_cur->ai_next != NULL) + if (try_next_address(conn)) { - conn->addr_cur = conn->addr_cur->ai_next; conn->status = CONNECTION_NEEDED; goto keep_going; } @@ -2596,13 +2833,19 @@ keep_going: /* We will come back to here until there is conn->errorMessage.data[conn->errorMessage.len - 1] != '\n') appendPQExpBufferChar(&conn->errorMessage, '\n'); PQclear(res); + /* If we have more than one host in the connect + * string, fatal message from one of them is not + * really fatal + */ + if (try_next_address(conn)) { + /* Must drop the old connection */ + pqDropConnection(conn, true); + conn->status = CONNECTION_NEEDED; + goto keep_going; + } goto error_return; } - /* We can release the address list now. */ - pg_freeaddrinfo_all(conn->addrlist_family, conn->addrlist); - conn->addrlist = NULL; - conn->addr_cur = NULL; /* Fire up post-connection housekeeping if needed */ if (PG_PROTOCOL_MAJOR(conn->pversion) < 3) @@ -2614,8 +2857,9 @@ keep_going: /* We will come back to here until there is } /* Otherwise, we are open for business! */ - conn->status = CONNECTION_OK; - return PGRES_POLLING_OK; + conn->status = CONNECTION_CHECK_RO; + goto keep_going; + } case CONNECTION_SETENV: @@ -2645,9 +2889,120 @@ keep_going: /* We will come back to here until there is goto error_return; } - /* We are open for business! */ + /* + * check if connection is readonly if we need readwrite one + */ + conn->status = CONNECTION_CHECK_RO; + goto keep_going; + + case CONNECTION_CHECK_RO: + + /* + * consult connection options and check if RO connection is OK + * RO connection is OK if readonly connection is explicitely + * requested or if replication option is set, or just one + * host is specified in the connect string. + */ + if ((conn->pghost==NULL || strchr(conn->pghost,',')==NULL) || + ((conn->read_only && conn->read_only[0] > '0')) + ||(conn->replication && conn->replication[0]) + ) + { + + /* We can release the address list now. + * but first make a copy of name of host we are + * connected to or it would be freed with list + */ + if (conn->actualhost) { + conn->actualhost = strdup(conn->actualhost); + conn->actualport = get_port_from_addrinfo(conn->addr_cur); + } + pg_freeaddrinfo_all(conn->addrlist_family, conn->addrlist); + conn->addrlist = NULL; + conn->addr_cur = NULL; + + conn->status = CONNECTION_OK; + return PGRES_POLLING_OK; + } + /* Otherwise request result pg_is_in_recovery() */ + /* pretend that status is OK for time of sending query */ conn->status = CONNECTION_OK; - return PGRES_POLLING_OK; + PQsendQuery(conn, "SELECT pg_catalog.pg_is_in_recovery()"); + conn->status = CONNECTION_CHECK_RW; + return PGRES_POLLING_READING; + case CONNECTION_CHECK_RW: + { + char *value; + PGresult *res; + + conn->status = CONNECTION_OK; + if (!PQconsumeInput(conn)) + { + conn->status = CONNECTION_BAD; + return PGRES_POLLING_FAILED; + } + if (PQisBusy(conn)) + { + /* Result is not ready yet */ + conn->status = CONNECTION_CHECK_RW; + return PGRES_POLLING_READING; + } + res = PQgetResult(conn); + /* Call PQgetResult second time to clear connection + * state. Should return NULL, so result is ignored + */ + PQgetResult(conn); + if (!res || PQresultStatus(res) != PGRES_TUPLES_OK || + PQntuples(res)!=1) + { + /* + * Something wrong happened with this host. skip to next + * one + */ + conn->status = CONNECTION_NEEDED; + } else { + value = PQgetvalue(res, 0, 0); + if (value[0]=='t') + { + appendPQExpBuffer(&conn->errorMessage, + libpq_gettext("cannot make RW connection to hot " + "standby node %s"),conn->actualhost); + conn->status = CONNECTION_NEEDED; + } + } + if (res) PQclear(res); + if (conn->status != CONNECTION_OK) + { + ConnStatusType save_status = conn->status; + + conn->status = CONNECTION_OK; + pqTerminateConn(conn); + pqDropConnection(conn,true); + conn->sock = PGINVALID_SOCKET; + if (try_next_address(conn)) + { + conn->status = save_status; + goto keep_going; + } + else + { + conn->status = CONNECTION_BAD; + return PGRES_POLLING_FAILED; + } + + } + /* We can release the address list now. */ + if (conn->actualhost) + { + conn->actualhost = strdup(conn->actualhost); + conn->actualport = get_port_from_addrinfo(conn->addr_cur); + + } + pg_freeaddrinfo_all(conn->addrlist_family, conn->addrlist); + conn->addrlist = NULL; + conn->addr_cur = NULL; + return PGRES_POLLING_OK; + } default: appendPQExpBuffer(&conn->errorMessage, @@ -2936,21 +3291,17 @@ freePGconn(PGconn *conn) WSACleanup(); #endif } - /* - * closePGconn - * - properly close a connection to the backend + * pqTerminateConn + * + * - send terminate message to the backend, but do not free any + * transient state of PGconn object, which can be needed to reconnect * - * This should reset or release all transient state, but NOT the connection - * parameters. On exit, the PGconn should be in condition to start a fresh - * connection with the same parameters (see PQreset()). */ -static void -closePGconn(PGconn *conn) -{ - PGnotify *notify; - pgParameterStatus *pstatus; + +static void pqTerminateConn(PGconn *conn) +{ /* * Note that the protocol doesn't allow us to send Terminate messages * during the startup phase. @@ -2966,6 +3317,25 @@ closePGconn(PGconn *conn) (void) pqFlush(conn); } + +} + +/* + * closePGconn + * - properly close a connection to the backend + * + * This should reset or release all transient state, but NOT the connection + * parameters. On exit, the PGconn should be in condition to start a fresh + * connection with the same parameters (see PQreset()). + */ +static void +closePGconn(PGconn *conn) +{ + PGnotify *notify; + pgParameterStatus *pstatus; + + /* Send terminate request to backend */ + pqTerminateConn(conn); /* * Must reset the blocking status so a possible reconnect will work. * @@ -2978,11 +3348,23 @@ closePGconn(PGconn *conn) * Close the connection, reset all transient state, flush I/O buffers. */ pqDropConnection(conn, true); + conn->status = CONNECTION_BAD; /* Well, not really _bad_ - just * absent */ conn->asyncStatus = PGASYNC_IDLE; pqClearAsyncResult(conn); /* deallocate result */ resetPQExpBuffer(&conn->errorMessage); + + /* If addrlist is not freed, actualhost points in there. + Otherwice it is allocated and should be freed */ + if (conn->addrlist == NULL && conn->actualhost !=NULL) { + free(conn->actualhost); + conn->actualhost = NULL; + } + if (conn->actualport) { + free(conn->actualport); + conn->actualport = NULL; + } pg_freeaddrinfo_all(conn->addrlist_family, conn->addrlist); conn->addrlist = NULL; conn->addr_cur = NULL; @@ -3969,6 +4351,8 @@ parseServiceFile(const char *serviceFile, { int linenr = 0, i; + bool hostflag = false; /* true if we already have seen 'host' + * parameter in the service file, and */ FILE *f; char buf[MAXBUFSIZE], *line; @@ -4088,7 +4472,29 @@ parseServiceFile(const char *serviceFile, if (strcmp(options[i].keyword, key) == 0) { if (options[i].val == NULL) + { options[i].val = strdup(val); + /* Set flag that we get value of host option + from this service file, so subsequent + host lines should be appended to it, + not ignored + */ + if (strcmp(key,"host")==0) + hostflag=true; + } + else if (strcmp(key,"host") == 0 && hostflag) + { + /* Old host value is from same service file, + so append new one to it */ + char *old=options[i].val; + int oldlen=strlen(old); + options[i].val=malloc(oldlen+1+strlen(val)+1); + strncpy(options[i].val,old,oldlen); + options[i].val[oldlen]=','; + strcpy(options[i].val+oldlen+1,val); + free(old); + } + if (!options[i].val) { printfPQExpBuffer(errorMessage, @@ -4808,87 +5214,118 @@ conninfo_uri_parse_options(PQconninfoOption *options, const char *uri, */ p = start; } - - /* - * "p" has been incremented past optional URI credential information at - * this point and now points at the "netloc" part of the URI. - * - * Look for IPv6 address. - */ - if (*p == '[') - { - host = ++p; - while (*p && *p != ']') - ++p; - if (!*p) + host = p; + if (*p==':') { + int portnum; + char *portstr; + *(p++)='\0'; + portstr = p; + portnum = 0; + while (*p >= '0' && *p <= '9') { - printfPQExpBuffer(errorMessage, - libpq_gettext("end of string reached when looking for matching \"]\" in IPv6 host address in URI: \"%s\"\n"), - uri); - goto cleanup; + portnum = portnum * 10 + (*(p++) - '0'); } - if (p == host) + if (portnum > 65535 || portnum <1) { printfPQExpBuffer(errorMessage, - libpq_gettext("IPv6 host address may not be empty in URI: \"%s\"\n"), - uri); + libpq_gettext("invalid port number: \"%d\"\n"), + portnum); goto cleanup; } + prevchar = *p; + *p='\0'; + if (*portstr && + !conninfo_storeval(options, "port", portstr, + errorMessage, false, true)); - /* Cut off the bracket and advance */ - *(p++) = '\0'; - - /* - * The address may be followed by a port specifier or a slash or a - * query. - */ - if (*p && *p != ':' && *p != '/' && *p != '?') + } else { + do { - printfPQExpBuffer(errorMessage, - libpq_gettext("unexpected character \"%c\" at position %d in URI (expected \":\" or \"/\"): \"%s\"\n"), - *p, (int) (p - buf + 1), uri); - goto cleanup; - } - } - else - { - /* not an IPv6 address: DNS-named or IPv4 netloc */ - host = p; + if (*p == ',') + p++; - /* - * Look for port specifier (colon) or end of host specifier (slash), - * or query (question mark). - */ - while (*p && *p != ':' && *p != '/' && *p != '?') - ++p; - } - - /* Save the hostname terminator before we null it */ - prevchar = *p; - *p = '\0'; + /* + * "p" has been incremented past optional URI credential information + * at this point and now points at the "netloc" part of the URI. + * + * Look for IPv6 address. + */ + if (*p == '[') + { + char *ipv6start = ++p; - if (*host && - !conninfo_storeval(options, "host", host, - errorMessage, false, true)) - goto cleanup; + while (*p && *p != ']') + ++p; + if (!*p) + { + printfPQExpBuffer(errorMessage, + libpq_gettext("end of string reached when looking for matching \"]\" in IPv6 host address in URI: \"%s\"\n"), + uri); + goto cleanup; + } + if (p == ipv6start) + { + printfPQExpBuffer(errorMessage, + libpq_gettext("IPv6 host address may not be empty in URI: \"%s\"\n"), + uri); + goto cleanup; + } + p++; + /* + * The address may be followed by a port specifier, a comma or a + * slash or a query. + */ + if (*p && *p != ',' && *p != ':' && *p != '/' && *p != '?') + { + printfPQExpBuffer(errorMessage, + libpq_gettext("unexpected character \"%c\" at position %d in URI (expected \":\" or \"/\"): \"%s\"\n"), + *p, (int) (p - buf + 1), uri); + goto cleanup; + } - if (prevchar == ':') - { - const char *port = ++p; /* advance past host terminator */ + } + else + { + /* not an IPv6 address: DNS-named or IPv4 netloc */ - while (*p && *p != '/' && *p != '?') - ++p; + /* + * Look for port specifier (colon) or end of host specifier + * (slash), or query (question mark). + */ + while (*p && *p != ',' && *p != ':' && *p != '/' && *p != '?') + ++p; + } + /* Skip port specifier */ + if (*p == ':') + { + int portnum; + p++; + portnum = 0; + while (*p >= '0' && *p <= '9') + { + portnum = portnum * 10 + (*(p++) - '0'); + } + if (portnum > 65535 || portnum <1) + { + printfPQExpBuffer(errorMessage, + libpq_gettext("invalid port number: \"%d\"\n"), + portnum); + goto cleanup; + } + } + } while (*p == ','); + /* Save the hostname terminator before we null it */ prevchar = *p; *p = '\0'; - - if (*port && - !conninfo_storeval(options, "port", port, - errorMessage, false, true)) + + if (*host && + !conninfo_storeval(options, "host", host, + errorMessage, false, true)) goto cleanup; - } + } if (prevchar && prevchar != '?') { const char *dbname = ++p; /* advance past host terminator */ @@ -5018,6 +5455,16 @@ conninfo_uri_parse_params(char *params, keyword = "sslmode"; value = "require"; } + if ((strcmp(keyword, "loadBalanceHosts") == 0 || + strcmp(keyword, "load_balance_hosts") == 0) && + strcmp(value, "true") == 0) + { + free(keyword); + free(value); + malloced = false; + keyword = "hostorder"; + value = "random"; + } /* * Store the value if the corresponding option exists; ignore @@ -5232,7 +5679,22 @@ conninfo_storeval(PQconninfoOption *connOptions, } if (option->val) + { + if (strcmp(option->keyword, "host") == 0) + { + /* Accumulate multiple hosts in the single string */ + int val_len = strlen(option->val), + new_len = strlen(value); + + free(value_copy); + value_copy = malloc(val_len + 1 + new_len + 1); + strncpy(value_copy, option->val, val_len + 1); + value_copy[val_len] = ','; + strncpy(value_copy + val_len + 1, value, new_len + 1); + } free(option->val); + + } option->val = value_copy; return option; @@ -5352,7 +5814,9 @@ PQhost(const PGconn *conn) { if (!conn) return NULL; - if (conn->pghost != NULL && conn->pghost[0] != '\0') + if (conn->actualhost != NULL && conn->actualhost[0] != '\0') + return conn->actualhost; + else if (conn->pghost != NULL && conn->pghost[0] != '\0') return conn->pghost; else { @@ -5372,6 +5836,8 @@ PQport(const PGconn *conn) { if (!conn) return NULL; + if (conn->actualport) + return conn->actualport; return conn->pgport; } diff --git a/src/interfaces/libpq/libpq-fe.h b/src/interfaces/libpq/libpq-fe.h index 9ca0756..23560f4 100644 --- a/src/interfaces/libpq/libpq-fe.h +++ b/src/interfaces/libpq/libpq-fe.h @@ -62,7 +62,11 @@ typedef enum * backend startup. */ CONNECTION_SETENV, /* Negotiating environment. */ CONNECTION_SSL_STARTUP, /* Negotiating SSL. */ - CONNECTION_NEEDED /* Internal state: connect() needed */ + CONNECTION_NEEDED, /* Internal state: connect() needed */ + CONNECTION_CHECK_RO, /* Internal state: need to check is RO + * connection acceptable */ + CONNECTION_CHECK_RW, /* Internal state: waiting that server replies + * if it is in recovery */ } ConnStatusType; typedef enum diff --git a/src/interfaces/libpq/libpq-int.h b/src/interfaces/libpq/libpq-int.h index 1183323..53d8891 100644 --- a/src/interfaces/libpq/libpq-int.h +++ b/src/interfaces/libpq/libpq-int.h @@ -334,7 +334,11 @@ struct pg_conn #if defined(ENABLE_GSS) || defined(ENABLE_SSPI) char *krbsrvname; /* Kerberos service name */ #endif - + char *hostorder; /* How to handle multiple hosts */ + char *read_only; /* If true, we could work with readonly + * standby server */ + char *failover_timeout; /* If no usable server found, how long + * to wait before retry */ /* Optional file to write trace info to */ FILE *Pfdebug; @@ -382,6 +386,12 @@ struct pg_conn struct addrinfo *addrlist; /* list of possible backend addresses */ struct addrinfo *addr_cur; /* the one currently being tried */ int addrlist_family; /* needed to know how to free addrlist */ + time_t failover_finish_time; /* how long to retry host list + *waiting for new master to appear */ + char *actualhost; /* Name of host we are actually connected + * to (if there is list in the pghost) */ + char *actualport; /* Port number we are actually + * connected to */ PGSetenvStatusType setenv_state; /* for 2.0 protocol only */ const PQEnvironmentOption *next_eo; bool send_appname; /* okay to send application_name? */ @@ -467,6 +477,7 @@ struct pg_conn /* Buffer for receiving various parts of messages */ PQExpBufferData workBuffer; /* expansible string */ + }; /* PGcancel stores all data necessary to cancel a connection. A copy of this diff --git a/src/interfaces/libpq/t/001-multihost.pl b/src/interfaces/libpq/t/001-multihost.pl new file mode 100644 index 0000000..5cf237b --- /dev/null +++ b/src/interfaces/libpq/t/001-multihost.pl @@ -0,0 +1,403 @@ +# Minimal test testing streaming replication +use strict; +use warnings; +use PostgresNode; +use TestLib; +use Test::More tests => 28; +# Initialize master node + +my $node_master = get_new_node('master'); +$node_master->init(allows_streaming => 1, use_tcp =>1); +$node_master->start; +my $backup_name = 'my_backup'; + +# Take backup +$node_master->backup($backup_name); +# Create streaming standby linking to master +my $node_standby_1 = get_new_node('standby_1'); + +$node_standby_1->init_from_backup($node_master, $backup_name, + has_streaming => 1, use_tcp=>1); +$node_standby_1->start; +# Take backup of standby 1 (not mandatory, but useful to check if +# pg_basebackup works on a standby). +$node_standby_1->backup($backup_name); + +# Create second standby node linking to standby 1 +my $node_standby_2 = get_new_node('standby_2'); +$node_standby_2->init_from_backup($node_standby_1, $backup_name, + has_streaming => 1,use_tcp=>1); +$node_standby_2->start; + + +sub get_host_port { + my $node = shift; + return "$PostgresNode::test_localhost:".$node->port +} + +sub multiconnstring { + my $nodes = shift; + my $database = shift || "postgres"; + my $params=shift; + my $extra=""; + if ($params) { + my @cs; + while (my ($key,$val) = each %$params) { + push @cs, $key ."=".$val; + } + $extra = "?".join("&",@cs); + } + my $str= "postgresql://".join(",",map({get_host_port($_)} @$nodes))."/$database$extra"; + return $str; + +} + +sub connstring2 { + my $nodes = shift; + my $database = shift; + my $params = shift; + my @args=(); + for my $n (@$nodes) { + push @args, "host=".get_host_port($n); + } + push @args, "dbname=$database" if defined($database); + while (my ($key,$val) = each %$params) { + push @args,"$key=$val"; + } + return join(" ",@args); +} + +# +# Copied from PosgresNode.pm passing explicit connect-string instead of +# constructed from object +# + +sub psql +{ + # We expect dbname to be part of connstr + my ($connstr, $sql, %params) = @_; + + my $stdout = $params{stdout}; + my $stderr = $params{stderr}; + my $timeout = undef; + my $timeout_exception = 'psql timed out'; + my @psql_params = + ('psql', '-XAtq', '-d', $connstr, '-f', '-'); + + # If the caller wants an array and hasn't passed stdout/stderr + # references, allocate temporary ones to capture them so we + # can return them. Otherwise we won't redirect them at all. + if (wantarray) + { + if (!defined($stdout)) + { + my $temp_stdout = ""; + $stdout = \$temp_stdout; + } + if (!defined($stderr)) + { + my $temp_stderr = ""; + $stderr = \$temp_stderr; + } + } + + $params{on_error_stop} = 1 unless defined $params{on_error_stop}; + $params{on_error_die} = 0 unless defined $params{on_error_die}; + + push @psql_params, '-v', 'ON_ERROR_STOP=1' if $params{on_error_stop}; + push @psql_params, @{ $params{extra_params} } + if defined $params{extra_params}; + + $timeout = + IPC::Run::timeout($params{timeout}, exception => $timeout_exception) + if (defined($params{timeout})); + + ${ $params{timed_out} } = 0 if defined $params{timed_out}; + + # IPC::Run would otherwise append to existing contents: + $$stdout = "" if ref($stdout); + $$stderr = "" if ref($stderr); + + my $ret; + + # Run psql and capture any possible exceptions. If the exception is + # because of a timeout and the caller requested to handle that, just return + # and set the flag. Otherwise, and for any other exception, rethrow. + # + # For background, see + # http://search.cpan.org/~ether/Try-Tiny-0.24/lib/Try/Tiny.pm + do + { + local $@; + eval { + my @ipcrun_opts = (\@psql_params, '<', \$sql); + push @ipcrun_opts, '>', $stdout if defined $stdout; + push @ipcrun_opts, '2>', $stderr if defined $stderr; + push @ipcrun_opts, $timeout if defined $timeout; + + IPC::Run::run @ipcrun_opts; + $ret = $?; + }; + my $exc_save = $@; + if ($exc_save) + { + + # IPC::Run::run threw an exception. re-throw unless it's a + # timeout, which we'll handle by testing is_expired + die $exc_save + if (blessed($exc_save) || $exc_save ne $timeout_exception); + + $ret = undef; + + die "Got timeout exception '$exc_save' but timer not expired?!" + unless $timeout->is_expired; + + if (defined($params{timed_out})) + { + ${ $params{timed_out} } = 1; + } + else + { + die "psql timed out: stderr: '$$stderr'\n" + . "while running '@psql_params'"; + } + } + }; + + if (defined $$stdout) + { + chomp $$stdout; + $$stdout =~ s/\r//g if $TestLib::windows_os; + } + + if (defined $$stderr) + { + chomp $$stderr; + $$stderr =~ s/\r//g if $TestLib::windows_os; + } + + # See http://perldoc.perl.org/perlvar.html#%24CHILD_ERROR + # We don't use IPC::Run::Simple to limit dependencies. + # + # We always die on signal. + my $core = $ret & 128 ? " (core dumped)" : ""; + die "psql exited with signal " + . ($ret & 127) + . "$core: '$$stderr' while running '@psql_params'" + if $ret & 127; + $ret = $ret >> 8; + + if ($ret && $params{on_error_die}) + { + die "psql error: stderr: '$$stderr'\nwhile running '@psql_params'" + if $ret == 1; + die "connection error: '$$stderr'\nwhile running '@psql_params'" + if $ret == 2; + die "error running SQL: '$$stderr'\nwhile running '@psql_params'" + if $ret == 3; + die "psql returns $ret: '$$stderr'\nwhile running '@psql_params'"; + } + + if (wantarray) + { + return ($ret, $$stdout, $$stderr); + } + else + { + return $ret; + } +} + +sub psql_conninfo { + my ($connstr) = shift; + my ($timed_out); + my ($retcode,$stdout,$stderr)=psql($connstr,'\conninfo', + timed_out=>\$timed_out); + if ($retcode==0 && $stdout =~ /on host "([^"]*)" at port "([^"]*)"/s) { + return "$1:$2"; + } else { + return "STDOUT:$stdout\nSTDERR:$stderr"; + } +} + +sub psql_server_addr { + my ($connstr) = shift; + my ($timed_out); + my $sql = "select abbrev(inet_server_addr()) ||':'||inet_server_port();\n"; + my ($retcode,$stdout,$stderr)=psql($connstr,$sql, + timed_out=>\$timed_out); + if ($retcode==0 ) { + return $stdout; + } else { + return "STDOUT:$stdout\nSTDERR:$stderr"; + } +} +my $conninfo; +# Test 1.1 - all hosts available, master first, readwrite requested + $conninfo=psql_conninfo(multiconnstring([$node_master,$node_standby_1,$node_standby_2])); + is($conninfo,get_host_port($node_master),"master first, rw, conninfo"); +# Test 1.2 + $conninfo=psql_server_addr(multiconnstring([$node_master,$node_standby_1,$node_standby_2])); + is($conninfo,get_host_port($node_master),"master first, rw, server funcs"); + +# Test 2.1 - use symbolic name for master and IP for slave + $conninfo=psql_conninfo("postgresql://localhost:".$node_master->port. + ",127.0.0.1:".$node_standby_1->port."/postgres"); + + is($conninfo,"localhost:".$node_master->port, "master symbolic, rw, conninfo"); +# Test 2.2 - check server-side connect info (would return numeric IP) + $conninfo=psql_server_addr("postgresql://localhost:".$node_master->port. + ",127.0.0.1:".$node_standby_1->port."/postgres"); + is($conninfo,"127.0.0.1:".$node_master->port,'master symbolic, rw server funcs'); + +# Test 3.1 - all nodes available, master second, readwrite requested + $conninfo=psql_conninfo(multiconnstring([$node_standby_1,$node_master,$node_standby_2])); + + is($conninfo,get_host_port($node_master),"master second,rw, conninfo" ); +# Test 3.2 Check server-side connection info + $conninfo=psql_server_addr(multiconnstring([$node_standby_1,$node_master,$node_standby_2])); + + is($conninfo,get_host_port($node_master),"master second, rw, server funcs"); +# Test 4.1 - use symbolic name for slave and IP for smaster + $conninfo=psql_conninfo("postgresql://localhost:".$node_standby_1->port. + ",127.0.0.1:".$node_master->port."/postgres"); + is($conninfo,"127.0.0.1:".$node_master->port,"slave symbolic, rw,conninfo"); + +# Test 4.2 - check server-side connect info + $conninfo=psql_server_addr("postgresql://localhost:".$node_standby_1->port. + ",127.0.0.1:".$node_master->port."/postgres"); + is($conninfo,"127.0.0.1:".$node_master->port,"slave symbolic rw, server funcs"); + +# Test 5 - all nodes available, master first, readonly requested + + $conninfo=psql_conninfo(multiconnstring([$node_master,$node_standby_1,$node_standby_2],undef,{readonly=>1})); + + is($conninfo,get_host_port($node_master),"master first, ro, conninfo"); + +# Test 6 - all nodes available, master second, readonly requested + $conninfo=psql_conninfo(multiconnstring([$node_standby_1,$node_master,$node_standby_2],undef,{readonly=>1})); + + is($conninfo,get_host_port($node_standby_1),"master second, ro conninfo"); + +# Test 7.1 - all nodes available, random order, readonly. +# Expect that during six attempts any of three nodes would be collected +# at least once + +my %conncount=(); +for (my $i=0;$i<9;$i++) { + my $conn = psql_conninfo(multiconnstring([$node_master,$node_standby_1, + $node_standby_2],undef,{readonly=>1,hostorder=>'random'})); + $conncount{$conn}++; +} +is(scalar(keys(%conncount)),3,'random order, readonly connect'); +# Test 7.2 - alternate (jdbc compatible) syntax for randomized hosts + +for (my $i=0;$i<6;$i++) { + my $conn = psql_conninfo(multiconnstring([$node_master,$node_standby_1, + $node_standby_2],undef,{readonly=>1,loadBalanceHosts=>"true"})); + $conncount{$conn}++; +} +#diag(join(",",keys %conncount)); +is(scalar(keys %conncount),3,"alternate JDBC-compatible syntax for random order"); + +# Test 8 - all nodes available, random order, readwrite +# Expect all six connections go to the master + +%conncount = (); +for (my $i=0;$i<6;$i++) { + my $conn = psql_conninfo(multiconnstring([$node_master,$node_standby_1, + $node_standby_2],undef,{hostorder=>'random'})); + $conncount{$conn}++; +} + +is(length(keys %conncount),1,'random order, rw connect only one node'); +ok(exists $conncount{get_host_port($node_master)},'random order, rw connects master'); + +# Test 8.1 one host in URL, master +$conninfo=psql_conninfo(multiconnstring([$node_master])); +is($conninfo,get_host_port($node_master),"old behavoir compat - master"); + + +# Test 8.2 one host in URL, slave +$conninfo=psql_conninfo(multiconnstring([$node_standby_1])); +is($conninfo,get_host_port($node_standby_1),"old behavoir compat - slave"); + +# Test 9 - try to connect only slaves in rw mode + +$conninfo=psql_conninfo(multiconnstring([$node_standby_1,$node_standby_2])); +is($conninfo,"STDOUT:\nSTDERR:psql: cannot make RW connection to hot standby node 127.0.0.1","cannot connect just slaves in RW mode"); + + + +# Test 10 - one of slaves is not available +$node_standby_1->stop(); + +# Test 10.1 + +$conninfo= psql_conninfo(multiconnstring([$node_standby_1,$node_master,$node_standby_2])); + +is($conninfo,get_host_port($node_master),"first node is unavailable"); + +# Test 10.2 + +$conninfo = psql_conninfo(multiconnstring([$node_standby_2,$node_standby_1,$node_master])); + +is($conninfo,get_host_port($node_master),"first node standby, second unavailable"); + +# Test 10.3 + +$conninfo = psql_conninfo(multiconnstring([$node_standby_1,$node_standby_2,$node_master],undef,{readonly=>1})); +is($conninfo,get_host_port($node_standby_2),"first node unavailable, second standmby, readonly mode"); + +$node_standby_1->start(); + +$node_master->stop(); + +$conninfo= psql_conninfo(multiconnstring([$node_standby_1,$node_master,$node_standby_2])); + +is($conninfo,"STDOUT:\nSTDERR:psql: cannot make RW connection to hot standby node 127.0.0.1","master unavialble, cannot connect just slaves in RW mode"); + +$conninfo= psql_conninfo(multiconnstring([$node_master,$node_standby_1,$node_standby_2],undef,{readonly=>1})); + +is($conninfo,get_host_port($node_standby_1),"Master unavailable, read only "); + +$node_master->start(); + +# Test 11 Alternate syntax + +$conninfo = psql_conninfo(connstring2([$node_standby_1,$node_standby_2,$node_master])); + +is ($conninfo, get_host_port($node_master),"Alternate syntax, master third, rw"); + + + +$conninfo = psql_conninfo(connstring2([$node_master, $node_standby_1,$node_standby_2])); + +is ($conninfo, get_host_port($node_master),"Alternate syntax, master first, rw"); + + + +$conninfo = +psql_conninfo(connstring2([$node_standby_1,$node_standby_2,$node_master],undef, +{readonly => 1})); + +is ($conninfo, get_host_port($node_standby_1),"Alternate syntax, master third, ro"); + + + +$conninfo = psql_conninfo(connstring2([$node_master, $node_standby_1,$node_standby_2], undef,{readonly=>1})); + +is ($conninfo, get_host_port($node_master),"Alternate syntax, master first, ro"); + + +# Test 11.5 one host in URL, master +$conninfo=psql_conninfo(connstring2([$node_master])); +is($conninfo,get_host_port($node_master),"alt syntax old behavoir compat - master"); + + +# Test 11.6 one host in URL, slave +$conninfo=psql_conninfo(connstring2([$node_standby_1])); +is($conninfo,get_host_port($node_standby_1),"alt syntax old behavoir compat - slave"); + + + diff --git a/src/interfaces/libpq/test/expected.out b/src/interfaces/libpq/test/expected.out index d375e82..4832bdd 100644 --- a/src/interfaces/libpq/test/expected.out +++ b/src/interfaces/libpq/test/expected.out @@ -1,20 +1,20 @@ trying postgresql://uri-user:secret@host:12345/db -user='uri-user' password='secret' dbname='db' host='host' port='12345' (inet) +user='uri-user' password='secret' dbname='db' host='host:12345' (inet) trying postgresql://uri-user@host:12345/db -user='uri-user' dbname='db' host='host' port='12345' (inet) +user='uri-user' dbname='db' host='host:12345' (inet) trying postgresql://uri-user@host/db user='uri-user' dbname='db' host='host' (inet) trying postgresql://host:12345/db -dbname='db' host='host' port='12345' (inet) +dbname='db' host='host:12345' (inet) trying postgresql://host/db dbname='db' host='host' (inet) trying postgresql://uri-user@host:12345/ -user='uri-user' host='host' port='12345' (inet) +user='uri-user' host='host:12345' (inet) trying postgresql://uri-user@host/ user='uri-user' host='host' (inet) @@ -23,10 +23,10 @@ trying postgresql://uri-user@ user='uri-user' (local) trying postgresql://host:12345/ -host='host' port='12345' (inet) +host='host:12345' (inet) trying postgresql://host:12345 -host='host' port='12345' (inet) +host='host:12345' (inet) trying postgresql://host/db dbname='db' host='host' (inet) @@ -62,7 +62,7 @@ trying postgresql://host/db?u%7aer=someotheruser&port=12345 uri-regress: invalid URI query parameter: "uzer" trying postgresql://host:12345?user=uri-user -user='uri-user' host='host' port='12345' (inet) +user='uri-user' host='host:12345' (inet) trying postgresql://host?user=uri-user user='uri-user' host='host' (inet) @@ -71,19 +71,19 @@ trying postgresql://host? host='host' (inet) trying postgresql://[::1]:12345/db -dbname='db' host='::1' port='12345' (inet) +dbname='db' host='[::1]:12345' (inet) trying postgresql://[::1]/db -dbname='db' host='::1' (inet) +dbname='db' host='[::1]' (inet) trying postgresql://[2001:db8::1234]/ -host='2001:db8::1234' (inet) +host='[2001:db8::1234]' (inet) trying postgresql://[200z:db8::1234]/ -host='200z:db8::1234' (inet) +host='[200z:db8::1234]' (inet) trying postgresql://[::1] -host='::1' (inet) +host='[::1]' (inet) trying postgres:// (local) @@ -143,7 +143,7 @@ trying postgres://@host host='host' (inet) trying postgres://host:/ -host='host' (inet) +uri-regress: invalid port number: "0" trying postgres://:12345/ port='12345' (local) diff --git a/src/test/perl/PostgresNode.pm b/src/test/perl/PostgresNode.pm index fede1e6..5d88c9a 100644 --- a/src/test/perl/PostgresNode.pm +++ b/src/test/perl/PostgresNode.pm @@ -392,7 +392,7 @@ sub init unless defined $params{hba_permit_replication}; $params{allows_streaming} = 0 unless defined $params{allows_streaming}; $params{has_archiving} = 0 unless defined $params{has_archiving}; - + $params{use_tcp} = 0 unless defined $params{use_tcp}; mkdir $self->backup_dir; mkdir $self->archive_dir; @@ -424,7 +424,11 @@ sub init else { print $conf "unix_socket_directories = '$host'\n"; - print $conf "listen_addresses = ''\n"; + if ($params{use_tcp} ) { + print $conf "listen_addresses = '$test_localhost'\n"; + } else { + print $conf "listen_addresses = ''\n"; + } } close $conf;
-- Sent via pgsql-hackers mailing list (pgsql-hackers@postgresql.org) To make changes to your subscription: http://www.postgresql.org/mailpref/pgsql-hackers