Implement Patroni’s REST API.
Exposes a REST API of patroni operations functions, such as status, performance and management to web clients.
Much of what can be achieved with the command line tool patronictl can be done via the API. Patroni CLI and daemon utilises the API to perform these functions.
- class patroni.api. RestApiHandler ( request : Any , client_address : Any , server : RestApiServer | HTTPServer ) View on GitHub
-
Bases:
BaseHTTPRequestHandler
Define how to handle each of the requests that are made against the REST API server.
- __init__ ( request : Any , client_address : Any , server : RestApiServer | HTTPServer ) None View on GitHub
-
Create a
RestApiHandler
instance.Note
Currently not different from its superclass
__init__()
, and only used sopyright
can understand the type ofserver
attribute.- Parameters :
-
-
request – client request to be processed.
-
client_address – address of the client connection.
-
server – HTTP server that received the request.
-
- _read_json_content ( body_is_optional : bool = False ) Dict [ Any , Any ] | None View on GitHub
-
Read JSON from HTTP request body.
Note
Retrieves the request body based on
content-length
HTTP header. The body is expected to be a JSON string with that length.If request body is expected but
content-length
HTTP header is absent, then write an HTTP response with HTTP status411
.If request body is expected but contains nothing, or if an exception is faced, then write an HTTP response with HTTP status
400
.- Parameters :
-
body_is_optional – if
False
then the request must contain a body. IfTrue
, then the request may or may not contain a body. - Returns :
-
deserialized JSON string from request body, if present. If body is absent, but body_is_optional is
True
, then return an empty dictionary. ReturnsNone
otherwise.
- _write_json_response ( status_code : int , response : Any ) None View on GitHub
-
Write an HTTP response with a JSON content type.
Call
write_response()
withcontent_type
asapplication/json
.- Parameters :
-
-
status_code – response HTTP status code.
-
response – value to be dumped as a JSON string and to be used as the response body.
-
- _write_status_code_only ( status_code : int ) None View on GitHub
-
Write a response that is composed only of the HTTP status.
The response is written with these values separated by space:
-
HTTP protocol version;
-
status_code ;
-
description of status_code .
Note
This is usually useful for replying to requests from software like HAProxy.
- Parameters :
-
status_code – HTTP status code.
- Example :
-
-
_write_status_code_only(200)
would write a response likeHTTP/1.0 200 OK
.
-
-
- _write_status_response ( status_code : int , response : Dict [ str , Any ] ) None View on GitHub
-
Write an HTTP response with Patroni/Postgres status in JSON format.
Modifies response before sending it to the client. Defines the
patroni
key, which is a dictionary that contains the mandatory keys:-
version
: Patroni version, e.g.3.0.2
; -
scope
: value ofscope
setting from Patroni configuration.
May also add the following optional keys, depending on the status of this Patroni/PostgreSQL node:
-
tags
: tags that were set through Patroni configuration merged with dynamically applied tags; -
database_system_identifier
:Database system identifier
frompg_controldata
output; -
pending_restart
:True
if PostgreSQL is pending to be restarted; -
-
pending_restart_reason
: dictionary where each key is the parameter that caused "pending restart" flag -
to be set and the value is a dictionary with the old and the new value.
-
-
-
scheduled_restart
: a dictionary with a single keyschedule
, which is the timestamp for the -
scheduled restart;
-
-
watchdog_failed
:True
if watchdog device is unhealthy; -
logger_queue_size
: log queue length if it is longer than expected; -
logger_records_lost
: number of log records that have been lost while the log queue was full.
- Parameters :
-
-
status_code – response HTTP status code.
-
response – represents the status of the PostgreSQL node, and is used as a basis for the HTTP response. This dictionary is built through
get_postgresql_status()
.
-
-
- do_DELETE_restart ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_DELETE_switchover ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_GET ( write_status_code_only : bool = False ) None View on GitHub
-
Process all GET requests which can not be routed to other methods.
Is used for handling all health-checks requests. E.g. "GET /(primary|replica|sync|async|etc…)".
The (optional) query parameters and the HTTP response status depend on the requested path:
-
/
,primary
, orread-write
:-
HTTP status
200
: if a primary with the leader lock.
-
-
/standby-leader
:-
HTTP status
200
: if holds the leader lock in a standby cluster.
-
-
/leader
:-
HTTP status
200
: if holds the leader lock.
-
-
/replica
:-
Query parameters:
-
HTTP status
200
: if up and running as a standby and withoutnoloadbalance
tag.
-
-
/read-only
:-
HTTP status
200
: if up and running and withoutnoloadbalance
tag.
-
-
/synchronous
or/sync
:-
HTTP status
200
: if up and running as a synchronous standby.
-
-
/read-only-sync
:-
HTTP status
200
: if up and running as a synchronous standby or primary.
-
-
/asynchronous
: -
/health
:-
HTTP status
200
: if up and running.
-
Note
If not able to honor the query parameter, or not able to match the condition described for HTTP status
200
in each path above, then HTTP status will be503
.Note
Independently of the requested path, if write_status_code_only is
False
, then it always write an HTTP response through_write_status_response()
, with the node status.- Parameters :
-
write_status_code_only – indicates that instead of a normal HTTP response we should send only the HTTP Status Code and close the connection. Useful when health-checks are executed by HAProxy.
-
- do_GET_cluster ( ) None View on GitHub
-
Handle a
GET
request to/cluster
path.Write an HTTP response with JSON content based on the output of
cluster_as_json()
, with HTTP status200
and the JSON representation of the cluster topology.
- do_GET_config ( ) None View on GitHub
-
Handle a
GET
request to/config
path.Write an HTTP response with a JSON content representing the Patroni configuration that is stored in the DCS, with HTTP status
200
.If the cluster information is not available in the DCS, then it will respond with no body and HTTP status
502
instead.
- do_GET_failsafe ( ) None View on GitHub
-
Handle a
GET
request to/failsafe
path.Writes a response with a JSON string body containing all nodes that are known to Patroni at a given point in time, with HTTP status
200
. The JSON contains a dictionary, each key is the name of the Patroni node, and the corresponding value is the URI to access/patroni
path of its REST API.Note
If
failsafe_mode
is not enabled, then write a response with HTTP status502
.
- do_GET_history ( ) None View on GitHub
-
Handle a
GET
request to/history
path.Write an HTTP response with a JSON content representing the history of events in the cluster, with HTTP status
200
.The response contains a
list
of failover/switchover events. Each item is alist
with the following items:-
Timeline when the event occurred (class:
int
); -
LSN at which the event occurred (class:
int
); -
The reason for the event (class:
str
); -
Timestamp when the new timeline was created (class:
str
); -
Name of the involved Patroni node (class:
str
).
-
- do_GET_liveness ( ) None View on GitHub
-
Handle a
GET
request to/liveness
path.Write a simple HTTP response with HTTP status:
-
200
:-
If the cluster is in maintenance mode; or
-
If Patroni heartbeat loop is properly running;
-
-
503
:-
-
if Patroni heartbeat loop last run was more than
ttl
setting ago on the primary (or twice the -
value of
ttl
on a replica).
-
if Patroni heartbeat loop last run was more than
-
-
- do_GET_metrics ( ) None View on GitHub
-
Handle a
GET
request to/metrics
path.Write an HTTP response with plain text content in the format used by Prometheus, with HTTP status
200
.The response contains the following items:
-
patroni_version
: Patroni version without periods, e.g.030002
for Patroni3.0.2
; -
patroni_postgres_running
:1
if PostgreSQL is running, else0
; -
patroni_postmaster_start_time
: epoch timestamp since Postmaster was started; -
patroni_master
:1
if this node holds the leader lock, else0
; -
patroni_primary
: same aspatroni_master
; -
patroni_xlog_location
:pg_wal_lsn_diff(pg_current_wal_flush_lsn(), '0/0')
if leader, else0
; -
patroni_standby_leader
:1
if standby leader node, else0
; -
patroni_replica
:1
if a replica, else0
; -
patroni_sync_standby
:1
if a sync replica, else0
; -
patroni_xlog_received_location
:pg_wal_lsn_diff(pg_last_wal_receive_lsn(), '0/0')
; -
patroni_xlog_replayed_location
:pg_wal_lsn_diff(pg_last_wal_replay_lsn(), '0/0)
; -
patroni_xlog_replayed_timestamp
:pg_last_xact_replay_timestamp
; -
patroni_xlog_paused
:pg_is_wal_replay_paused()
; -
patroni_postgres_server_version
: Postgres version without periods, e.g.150002
for Postgres15.2
; -
patroni_cluster_unlocked
:1
if no one holds the leader lock, else0
; -
patroni_failsafe_mode_is_active
:1
iffailsafe_mode
is currently active, else0
; -
patroni_postgres_timeline
: PostgreSQL timeline based on current WAL file name; -
patroni_dcs_last_seen
: epoch timestamp when DCS was last contacted successfully; -
patroni_pending_restart
:1
if this PostgreSQL node is pending a restart, else0
; -
patroni_is_paused
:1
if Patroni is in maintenance node, else0
.
For PostgreSQL v9.6+ the response will also have the following:
-
patroni_postgres_streaming
: 1 if Postgres is streaming from another node, else0
; -
patroni_postgres_in_archive_recovery
:1
if Postgres isn’t streaming and there isrestore_command
available, else0
.
-
- do_GET_patroni ( ) None View on GitHub
-
Handle a
GET
request to/patroni
path.Write an HTTP response through
_write_status_response()
, with HTTP status200
and the status of Postgres.
- do_GET_readiness ( ) None View on GitHub
-
Handle a
GET
request to/readiness
path.Write a simple HTTP response which HTTP status can be:
-
200
:-
If this Patroni node holds the DCS leader lock; or
-
If this PostgreSQL instance is up and running;
-
-
503
: if none of the previous conditions apply.
-
- do_HEAD ( ) None View on GitHub
-
Handle a
HEAD
request.Write a simple HTTP response that represents the current PostgreSQL status. Send only
200 OK
or503 Service Unavailable
as a response and nothing more, particularly no headers.
- do_OPTIONS ( ) None View on GitHub
-
Handle an
OPTIONS
request.Write a simple HTTP response that represents the current PostgreSQL status. Send only
200 OK
or503 Service Unavailable
as a response and nothing more, particularly no headers.
- do_PATCH_config ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_POST_citus ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_POST_failover ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_POST_failsafe ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_POST_mpp ( ) None View on GitHub
-
Handle a
POST
request to/mpp
path.Call
handle_event()
to handle the request, then write a response with HTTP status code200
.Note
If unable to parse the request body, then the request is silently discarded.
- do_POST_reinitialize ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_POST_reload ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_POST_restart ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_POST_sigterm ( * args : Any , ** kwargs : Any ) Any View on GitHub
- do_POST_switchover ( ) None View on GitHub
-
Handle a
POST
request to/switchover
path.Calls
do_POST_failover()
withswitchover
option.
- do_PUT_config ( * args : Any , ** kwargs : Any ) Any View on GitHub
- get_postgresql_status ( retry : bool = False ) Dict [ str , Any ] View on GitHub
-
Builds an object representing a status of "postgres".
Some of the values are collected by executing a query and other are taken from the state stored in memory.
- Parameters :
-
retry – whether the query should be retried if failed or give up immediately
- Returns :
-
a dict with the status of Postgres/Patroni. The keys are:
-
state
: Postgres state amongstopping
,stopped
,stop failed
,crashed
,running
,starting
,start failed
,restarting
,restart failed
,initializing new cluster
,initdb failed
,running custom bootstrap script
,custom bootstrap failed
,creating replica
, orunknown
; -
postmaster_start_time
:pg_postmaster_start_time()
; -
role
:replica
ormaster
based onpg_is_in_recovery()
output; -
server_version
: Postgres version without periods, e.g.150002
for Postgres15.2
; -
xlog
: dictionary. Its structure depends onrole
:-
If
master
:-
location
:pg_current_wal_flush_lsn()
-
-
If
replica
:-
received_location
:pg_wal_lsn_diff(pg_last_wal_receive_lsn(), '0/0')
; -
replayed_location
:pg_wal_lsn_diff(pg_last_wal_replay_lsn(), '0/0)
; -
replayed_timestamp
:pg_last_xact_replay_timestamp
; -
paused
:pg_is_wal_replay_paused()
;
-
-
-
sync_standby
:True
if replication mode is synchronous and this is a sync standby; -
timeline
: PostgreSQL primary node timeline; -
-
replication
:list
ofdict
entries, one for each replication connection. Each entry -
contains the following keys:
-
application_name
:pg_stat_activity.application_name
; -
client_addr
:pg_stat_activity.client_addr
; -
state
:pg_stat_replication.state
; -
sync_priority
:pg_stat_replication.sync_priority
; -
sync_state
:pg_stat_replication.sync_state
; -
usename
:pg_stat_activity.usename
.
-
-
-
pause
:True
if cluster is in maintenance mode; -
cluster_unlocked
:True
if cluster has no node holding the leader lock; -
failsafe_mode_is_active
:True
if DCS failsafe mode is currently active; -
dcs_last_seen
: epoch timestamp DCS was last reached by Patroni.
-
- handle_one_request ( ) None View on GitHub
-
Parse and dispatch a request to the appropriate
do_*
method.Note
This is only used to keep track of latency when logging messages through
log_message()
.
- is_failover_possible ( cluster : Cluster , leader : str | None , candidate : str | None , action : str ) str | None View on GitHub
-
Checks whether there are nodes that could take over after demoting the primary.
- Parameters :
-
-
cluster – the Patroni cluster.
-
leader – name of the current Patroni leader.
-
candidate – name of the Patroni node to be promoted.
-
action – the action to be performed (
switchover
orfailover
).
-
- Returns :
-
a string with the error message or
None
if good nodes are found.
- log_message ( format : str , * args : Any ) None View on GitHub
-
Log a custom
debug
message.Additionally, to format , the log entry contains the client IP address and the current latency of the request.
- Parameters :
-
-
format – printf-style format string message to be logged.
-
args – arguments to be applied as inputs to format .
-
- parse_request ( ) bool View on GitHub
-
Override
parse_request()
to enrich basic functionality ofBaseHTTPRequestHandler
.Original class can only invoke
do_GET()
,do_POST()
,do_PUT()
, etc method implementations if they are defined.But we would like to have at least some simple routing mechanism, i.e.:
-
GET /uri1/part2
request should invokedo_GET_uri1()
-
POST /other
should invokedo_POST_other()
If the
do_
method does not exist we’ll fall back to original behavior.- Returns :
-
True
for success,False
for failure; on failure, any relevant error response has already been sent back.
-
- static parse_schedule ( schedule : str , action : str ) Tuple [ int | None , str | None , datetime | None ] View on GitHub
-
Parse the given schedule and validate it.
- Parameters :
-
-
schedule – a string representing a timestamp, e.g.
2023-04-14T20:27:00+00:00
. -
action – the action to be scheduled (
restart
,switchover
, orfailover
).
-
- Returns :
-
a tuple composed of 3 items:
-
Suggested HTTP status code for a response:
-
None
: if no issue was faced while parsing, leaving it up to the caller to decide the status; or -
400
: if no timezone information could be found in schedule ; or -
422
: if schedule is invalid – in the past or not parsable.
-
-
An error message, if any error is faced, otherwise
None
; -
Parsed schedule , if able to parse, otherwise
None
.
-
- poll_failover_result ( leader : str | None , candidate : str | None , action : str ) Tuple [ int , str ] View on GitHub
-
Poll failover/switchover operation until it finishes or times out.
- Parameters :
-
-
leader – name of the current Patroni leader.
-
candidate – name of the Patroni node to be promoted.
-
action – the action that is ongoing (
switchover
orfailover
).
-
- Returns :
-
a tuple composed of 2 items:
-
Response HTTP status codes:
-
200
: if the operation succeeded; or -
503
: if the operation failed or timed out.
-
-
A status message about the operation.
-
- query ( sql : str , * params : Any , retry : bool = False ) List [ Tuple [ Any , ... ] ] View on GitHub
-
Execute sql query with params and optionally return results.
- Parameters :
-
-
sql – the SQL statement to be run.
-
params – positional arguments to call
RestApiServer.query()
with. -
retry – whether the query should be retried upon failure or given up immediately.
-
- Returns :
-
a list of rows that were fetched from the database.
- write_response ( status_code : int , body : str , content_type : str = 'text/html' , headers : Dict [ str , str ] | None = None ) None View on GitHub
-
Write an HTTP response.
Note
Besides
Content-Type
header, and the HTTP headers passed through headers , this function will also write the HTTP headers defined throughrestapi.http_extra_headers
andrestapi.https_extra_headers
from Patroni configuration.- Parameters :
-
-
status_code – response HTTP status code.
-
body – response body.
-
content_type – value for
Content-Type
HTTP header. -
headers – dictionary of additional HTTP headers to set for the response. Each key is the header name, and the corresponding value is the value for the header in the response.
-
- class patroni.api. RestApiServer ( patroni : Patroni , config : Dict [ str , Any ] ) View on GitHub
-
Bases:
ThreadingMixIn
,HTTPServer
,Thread
Patroni REST API server.
An asynchronous thread-based HTTP server.
- __has_dual_stack ( ) bool
-
Check if the system has support for dual stack sockets.
- Returns :
-
True
if it has support for dual stack sockets.
- __httpserver_init ( host : str , port : int ) None
-
Start REST API HTTP server.
Note
If system has no support for dual stack sockets, then IPv4 is preferred over IPv6.
- Parameters :
-
-
host – host to bind REST API to.
-
port – port to bind REST API to.
-
- __init__ ( patroni : Patroni , config : Dict [ str , Any ] ) None View on GitHub
-
Establish patroni configuration for the REST API daemon.
Create a
RestApiServer
instance.- Parameters :
-
-
patroni – Patroni daemon process.
-
config –
restapi
section of Patroni configuration.
-
- __initialize ( listen : str , ssl_options : Dict [ str , Any ] ) None
-
Configure and start REST API HTTP server.
Note
This method can be called upon first initialization, and also when reloading Patroni. When reloading Patroni, it restarts the HTTP server thread.
- Parameters :
-
-
listen – IP and port to bind REST API to. It should be a string in the format
host:port
, wherehost
can be a hostname or IP address. It is the value ofrestapi.listen
setting. -
ssl_options –
dictionary that may contain the following keys, depending on what has been configured in
restapi
section:-
certfile
: path to PEM certificate. If given, will start in HTTPS mode; -
keyfile
: path to key ofcertfile
; -
keyfile_password
: password for decryptingkeyfile
; -
cafile
: path to CA file to validate client certificates; -
ciphers
: permitted cipher suites; -
verify_client
: value can be one among:-
none
: do not check client certificates; -
optional
: check client certificate only for unsafe REST API endpoints; -
required
: check client certificate for all REST API endpoints.
-
-
-
- Raises :
-
ValueError
: if any issue is faced while parsing listen .
- __members_ips ( ) Iterator [ IPv4Network | IPv6Network ]
-
Resolve each Patroni node
restapi.connect_address
to IP networks.Note
Only yields object if
restapi.allowlist_include_members
setting is enabled.- Yields :
-
each node
restapi.connect_address
resolved to an IP network.
- __resolve_ips ( port : int ) Iterator [ IPv4Network | IPv6Network ]
-
Resolve host + port to one or more IP networks.
- Parameters :
-
-
host – hostname to be checked.
-
port – port to be checked.
-
- Yields :
-
host + port resolved to IP networks.
- _build_allowlist ( value : List [ str ] | None ) Iterator [ IPv4Network | IPv6Network ] View on GitHub
-
Resolve each entry in value to an IP network object.
- Parameters :
-
value – list of IPs and/or networks contained in
restapi.allowlist
setting. Each item can be a host, an IP, or a network in CIDR format. - Yields :
-
host + port resolved to IP networks.
- static _set_fd_cloexec ( fd : socket ) None View on GitHub
-
Set
FD_CLOEXEC
for fd .It is used to avoid inheriting the REST API port when forking its process.
Note
Only takes effect on non-Windows environments.
- Parameters :
-
fd – socket file descriptor.
- check_access ( rh : RestApiHandler , allowlist_check_members : bool = True ) bool | None View on GitHub
-
Ensure client has enough privileges to perform a given request.
Write a response back to the client if any issue is observed, and the HTTP status may be:
-
401
: ifAuthorization
header is missing or contain an invalid password; -
403
: if:-
restapi.allowlist
was configured, but client IP is not in the allowed list; or -
restapi.allowlist_include_members
is enabled, but client IP is not in the members list; or -
a client certificate is expected by the server, but is missing in the request.
-
- Parameters :
-
-
rh – the request which access should be checked.
-
allowlist_check_members – whether we should check the source ip against existing cluster members.
-
- Returns :
-
True
if client access verification succeeded, otherwiseNone
.
-
- check_auth_header ( auth_header : str | None ) str | None View on GitHub
-
Validate HTTP Basic authorization header, if present.
- Parameters :
-
auth_header – value of
Authorization
HTTP header, if present, elseNone
. - Returns :
-
an error message if any issue is found,
None
otherwise.
- check_basic_auth_key ( key : str ) bool View on GitHub
-
Check if key matches the password configured for the REST API.
- Parameters :
-
key – the password received through the Basic authorization header of an HTTP request.
- Returns :
-
True
if key matches the password configured for the REST API.
- daemon_threads = True
- get_certificate_serial_number ( ) str | None View on GitHub
-
Get serial number of the certificate used by the REST API.
- Returns :
-
serial number of the certificate configured through
restapi.certfile
setting.
- handle_error ( request : socket | Tuple [ bytes , socket ] , client_address : Tuple [ str , int ] ) None View on GitHub
-
Handle any exception that is thrown while handling a request to the REST API.
Logs
WARNING
messages with the client information, and the stack trace of the faced exception.- Parameters :
-
-
request – the request that faced an exception.
-
client_address – a tuple composed of the IP and port of the client connection.
-
- process_request_thread ( request : socket | Tuple [ bytes , socket ] , client_address : Tuple [ str , int ] ) None View on GitHub
-
Process a request to the REST API.
Wrapper for
process_request_thread()
that additionally:-
Enable TCP keepalive
-
Perform SSL handshake (if an SSL socket).
- Parameters :
-
-
request – socket to handle the client request.
-
client_address – tuple containing the client IP and port.
-
-
- query ( sql : str , * params : Any ) List [ Tuple [ Any , ... ] ] View on GitHub
-
Execute sql query with params and optionally return results.
Note
Prefer to use own connection to postgres and fallback to
heartbeat
when own isn’t available.- Parameters :
-
-
sql – the SQL statement to be run.
-
params – positional arguments to be used as parameters for sql .
-
- Returns :
-
a list of rows that were fetched from the database.
- Raises :
-
psycopg.Error
: if had issues while executing sql .PostgresConnectionException
: if had issues while connecting to the database.
- reload_config ( config : Dict [ str , Any ] ) None View on GitHub
-
Reload REST API configuration.
- Parameters :
-
config – dictionary representing values under the
restapi
configuration section. - Raises :
-
ValueError
: iflisten
key is not present in config .
- reload_local_certificate ( ) bool | None View on GitHub
-
Reload the SSL certificate used by the REST API.
- Returns :
-
True
if a different certificate has been configured throughrestapi.certfile` setting, ``None
otherwise.
- shutdown_request ( request : socket | Tuple [ bytes , socket ] ) None View on GitHub
-
Shut down a request to the REST API.
Wrapper for
http.server.HTTPServer.shutdown_request()
that additionally:-
Perform SSL shutdown handshake (if a SSL socket).
- Parameters :
-
request – socket to handle the client request.
-
- patroni.api. check_access ( * args : Any , ** kwargs : Any ) Callable [ [ ... ] , Any ] View on GitHub
-
Check the source ip, authorization header, or client certificates.
Note
The actual logic to check access is implemented through
RestApiServer.check_access()
.Optionally it is possible to skip source ip check by specifying
allowlist_check_members=False
.- Returns :
-
a decorator that executes func only if
RestApiServer.check_access()
returnsTrue
. - Example :
-
>>> class FooServer: ... def check_access(self, *args, **kwargs): ... print(f'In FooServer: {args[0].__class__.__name__}') ... return True ...
>>> class Foo: ... server = FooServer() ... @check_access ... def do_PUT_foo(self): ... print('In do_PUT_foo') ... @check_access(allowlist_check_members=False) ... def do_POST_bar(self): ... print('In do_POST_bar')
>>> f = Foo() >>> f.do_PUT_foo() In FooServer: Foo In do_PUT_foo