swift.proxy.controllers.base.
ByteCountEnforcer
(file_like, nbytes)¶Bases: object
Enforces that successive calls to file_like.read() give at least <nbytes> bytes before exhaustion.
If file_like fails to do so, ShortReadError is raised.
If more than <nbytes> bytes are read, we don’t care.
read
(amt=None)¶swift.proxy.controllers.base.
Controller
(app)¶Bases: object
Base WSGI controller class for the proxy
GET
(req)¶Handler for HTTP GET requests.
Parameters: | req – The client request |
---|---|
Returns: | the response to the client |
GETorHEAD_base
(req, server_type, node_iter, partition, path, concurrency=1, client_chunk_size=None)¶Base handler for HTTP GET or HEAD requests.
Parameters: |
|
---|---|
Returns: | swob.Response object |
HEAD
(req)¶Handler for HTTP HEAD requests.
Parameters: | req – The client request |
---|---|
Returns: | the response to the client |
OPTIONS
(req)¶Base handler for OPTIONS requests
Parameters: | req – swob.Request object |
---|---|
Returns: | swob.Response object |
account_info
(account, req=None)¶Get account information, and also verify that the account exists.
Parameters: |
|
---|---|
Returns: | tuple of (account partition, account nodes, container_count) or (None, None, None) if it does not exist |
allowed_methods
¶autocreate_account
(req, account)¶Autocreate an account
Parameters: |
|
---|
best_response
(req, statuses, reasons, bodies, server_type, etag=None, headers=None, overrides=None, quorum_size=None)¶Given a list of responses from several servers, choose the best to return to the API.
Parameters: |
|
---|---|
Returns: | swob.Response object with the correct status, body, etc. set |
container_info
(account, container, req=None)¶Get container information and thusly verify container existence. This will also verify account existence.
Parameters: |
|
---|---|
Returns: | dict containing at least container partition (‘partition’), container nodes (‘containers’), container read acl (‘read_acl’), container write acl (‘write_acl’), and container sync key (‘sync_key’). Values are set to None if the container does not exist. |
generate_request_headers
(orig_req=None, additional=None, transfer=False)¶Create a list of headers to be used in backend requests
Parameters: |
|
---|---|
Returns: | a dictionary of headers |
get_name_length_limit
()¶have_quorum
(statuses, node_count, quorum=None)¶Given a list of statuses from several requests, determine if a quorum response can already be decided.
Parameters: |
|
---|---|
Returns: | True or False, depending on if quorum is established |
is_origin_allowed
(cors_info, origin)¶Is the given Origin allowed to make requests to this resource
Parameters: |
|
---|---|
Returns: | True or False |
make_requests
(req, ring, part, method, path, headers, query_string='', overrides=None, node_count=None, node_iterator=None)¶Sends an HTTP request to multiple nodes and aggregates the results. It attempts the primary nodes concurrently, then iterates over the handoff nodes as needed.
Parameters: |
|
---|---|
Returns: | a swob.Response object |
pass_through_headers
= []¶server_type
= 'Base'¶transfer_headers
(src_headers, dst_headers)¶Transfer legal headers from an original client request to dictionary that will be used as headers by the backend request
Parameters: |
|
---|
swift.proxy.controllers.base.
GetOrHeadHandler
(app, req, server_type, node_iter, partition, path, backend_headers, concurrency=1, client_chunk_size=None, newest=None, header_provider=None)¶Bases: swift.proxy.controllers.base.ResumingGetter
get_working_response
(req)¶swift.proxy.controllers.base.
NodeIter
(app, ring, partition, node_iter=None, policy=None)¶Bases: object
Yields nodes for a ring partition, skipping over error limited nodes and stopping at the configurable number of nodes. If a node yielded subsequently gets error limited, an extra node will be yielded to take its place.
Note that if you’re going to iterate over this concurrently from multiple greenthreads, you’ll want to use a swift.common.utils.GreenthreadSafeIterator to serialize access. Otherwise, you may get ValueErrors from concurrent access. (You also may not, depending on how logging is configured, the vagaries of socket IO and eventlet, and the phase of the moon.)
Parameters: |
|
---|
log_handoffs
(handoffs)¶Log handoff requests if handoff logging is enabled and the handoff was not expected.
We only log handoffs when we’ve pushed the handoff count further than we would normally have expected under normal circumstances, that is (request_node_count - num_primaries), when handoffs goes higher than that it means one of the primaries must have been skipped because of error limiting before we consumed all of our nodes_left.
next
()¶set_node_provider
(callback)¶Install a callback function that will be used during a call to next() to get an alternate node instead of returning the next node from the iterator.
Parameters: | callback – A no argument function that should return a node dict or None. |
---|
swift.proxy.controllers.base.
ResumingGetter
(app, req, server_type, node_iter, partition, path, backend_headers, concurrency=1, client_chunk_size=None, newest=None, header_provider=None)¶Bases: object
fast_forward
(num_bytes)¶Will skip num_bytes into the current ranges.
Params num_bytes: | |
---|---|
the number of bytes that have already been read on this request. This will change the Range header so that the next req will start where it left off. |
|
Raises: |
|
is_good_source
(src)¶Indicates whether or not the request made to the backend found what it was looking for.
Parameters: | src – the response from the backend |
---|---|
Returns: | True if found, False if not |
last_headers
¶last_status
¶learn_size_from_content_range
(start, end, length)¶If client_chunk_size is set, makes sure we yield things starting on chunk boundaries based on the Content-Range header in the response.
Sets our Range header’s first byterange to the value learned from the Content-Range header in the response; if we were given a fully-specified range (e.g. “bytes=123-456”), this is a no-op.
If we were given a half-specified range (e.g. “bytes=123-” or “bytes=-456”), then this changes the Range header to a semantically-equivalent one and it lets us resume on a proper boundary instead of just in the middle of a piece somewhere.
pop_range
()¶Remove the first byterange from our Range header.
This is used after a byterange has been completely sent to the client; this way, should we need to resume the download from another object server, we do not re-fetch byteranges that the client already has.
If we have no Range header, this is a no-op.
response_parts_iter
(req)¶swift.proxy.controllers.base.
bytes_to_skip
(record_size, range_start)¶Assume an object is composed of N records, where the first N-1 are all the same size and the last is at most that large, but may be smaller.
When a range request is made, it might start with a partial record. This must be discarded, lest the consumer get bad data. This is particularly true of suffix-byte-range requests, e.g. “Range: bytes=-12345” where the size of the object is unknown at the time the request is made.
This function computes the number of bytes that must be discarded to ensure only whole records are yielded. Erasure-code decoding needs this.
This function could have been inlined, but it took enough tries to get right that some targeted unit tests were desirable, hence its extraction.
swift.proxy.controllers.base.
clear_info_cache
(app, env, account, container=None)¶Clear the cached info in both memcache and env
Parameters: |
|
---|
swift.proxy.controllers.base.
close_swift_conn
(src)¶Force close the http connection to the backend.
Parameters: | src – the response from the backend |
---|
swift.proxy.controllers.base.
cors_validation
(func)¶Decorator to check if the request is a CORS request and if so, if it’s valid.
Parameters: | func – function to check |
---|
swift.proxy.controllers.base.
delay_denial
(func)¶Decorator to declare which methods should have any swift.authorize call delayed. This is so the method can load the Request object up with additional information that may be needed by the authorization system.
Parameters: | func – function for which authorization will be delayed |
---|
swift.proxy.controllers.base.
get_account_info
(env, app, swift_source=None)¶Get the info structure for an account, based on env and app. This is useful to middlewares.
Note
This call bypasses auth. Success does not imply that the request has authorization to the account.
Raises: | ValueError – when path doesn’t contain an account |
---|
swift.proxy.controllers.base.
get_cache_key
(account, container=None, obj=None)¶Get the keys for both memcache and env[‘swift.infocache’] (cache_key) where info about accounts, containers, and objects is cached
Parameters: |
|
---|---|
Returns: | a string cache_key |
swift.proxy.controllers.base.
get_container_info
(env, app, swift_source=None)¶Get the info structure for a container, based on env and app. This is useful to middlewares.
Note
This call bypasses auth. Success does not imply that the request has authorization to the container.
swift.proxy.controllers.base.
get_info
(app, env, account, container=None, swift_source=None)¶Get info about accounts or containers
Parameters: |
|
---|---|
Returns: | information about the specified entity in a dictionary. See get_account_info and get_container_info for details on what’s in the dictionary. |
swift.proxy.controllers.base.
get_object_info
(env, app, path=None, swift_source=None)¶Get the info structure for an object, based on env and app. This is useful to middlewares.
Note
This call bypasses auth. Success does not imply that the request has authorization to the object.
swift.proxy.controllers.base.
headers_to_account_info
(headers, status_int=200)¶Construct a cacheable dict of account info based on response headers.
swift.proxy.controllers.base.
headers_to_container_info
(headers, status_int=200)¶Construct a cacheable dict of container info based on response headers.
swift.proxy.controllers.base.
headers_to_object_info
(headers, status_int=200)¶Construct a cacheable dict of object info based on response headers.
swift.proxy.controllers.base.
set_info_cache
(app, env, account, container, resp)¶Cache info in both memcache and env.
Parameters: |
|
---|---|
Returns: | the info that was placed into the cache, or None if the request status was not in (404, 410, 2xx). |
swift.proxy.controllers.base.
set_object_info_cache
(app, env, account, container, obj, resp)¶Cache object info in the WSGI environment, but not in memcache. Caching in memcache would lead to cache pressure and mass evictions due to the large number of objects in a typical Swift cluster. This is a per-request cache only.
Parameters: |
|
---|---|
Returns: | the object info |
swift.proxy.controllers.base.
source_key
(resp)¶Provide the timestamp of the swift http response as a floating point value. Used as a sort key.
Parameters: | resp – bufferedhttp response object |
---|
swift.proxy.controllers.base.
update_headers
(response, headers)¶Helper function to update headers in the response.
Parameters: |
|
---|
swift.proxy.controllers.account.
AccountController
(app, account_name, **kwargs)¶Bases: swift.proxy.controllers.base.Controller
WSGI controller for account requests
DELETE
(req)¶HTTP DELETE request handler.
GETorHEAD
(req)¶Handler for HTTP GET/HEAD requests.
POST
(req)¶HTTP POST request handler.
PUT
(req)¶HTTP PUT request handler.
add_acls_from_sys_metadata
(resp)¶server_type
= 'Account'¶swift.proxy.controllers.container.
ContainerController
(app, account_name, container_name, **kwargs)¶Bases: swift.proxy.controllers.base.Controller
WSGI controller for container requests
DELETE
(**kw)¶HTTP DELETE request handler.
GET
(**kw)¶Handler for HTTP GET requests.
GETorHEAD
(req)¶Handler for HTTP GET/HEAD requests.
HEAD
(**kw)¶Handler for HTTP HEAD requests.
POST
(**kw)¶HTTP POST request handler.
PUT
(**kw)¶HTTP PUT request handler.
clean_acls
(req)¶pass_through_headers
= ['x-container-read', 'x-container-write', 'x-container-sync-key', 'x-container-sync-to', 'x-versions-location']¶server_type
= 'Container'¶swift.proxy.controllers.obj.
BaseObjectController
(app, account_name, container_name, object_name, **kwargs)¶Bases: swift.proxy.controllers.base.Controller
Base WSGI controller for object requests.
DELETE
(**kw)¶HTTP DELETE request handler.
GET
(**kw)¶Handler for HTTP GET requests.
GETorHEAD
(req)¶Handle HTTP GET or HEAD requests.
HEAD
(**kw)¶Handler for HTTP HEAD requests.
POST
(**kw)¶HTTP POST request handler.
PUT
(**kw)¶HTTP PUT request handler.
iter_nodes_local_first
(ring, partition, policy=None, local_handoffs_first=False)¶Yields nodes for a ring partition.
If the ‘write_affinity’ setting is non-empty, then this will yield N local nodes (as defined by the write_affinity setting) first, then the rest of the nodes as normal. It is a re-ordering of the nodes such that the local ones come first; no node is omitted. The effect is that the request will be serviced by local object servers first, but nonlocal ones will be employed if not enough local ones are available.
Parameters: |
|
---|
server_type
= 'Object'¶swift.proxy.controllers.obj.
ECAppIter
(path, policy, internal_parts_iters, range_specs, fa_length, obj_length, logger)¶Bases: object
WSGI iterable that decodes EC fragment archives (or portions thereof) into the original object (or portions thereof).
Parameters: |
|
---|
app_iter_range
(start, end)¶app_iter_ranges
(ranges, content_type, boundary, content_size)¶close
()¶kickoff
(req, resp)¶Start pulling data from the backends so that we can learn things like the real Content-Type that might only be in the multipart/byteranges response body. Update our response accordingly.
Also, this is the first point at which we can learn the MIME boundary that our response has in the headers. We grab that so we can also use it in the body.
Returns: | None |
---|---|
Raises: | HTTPException – on error |
swift.proxy.controllers.obj.
ECGetResponseBucket
(policy, timestamp_str)¶Bases: object
A helper class to encapsulate the properties of buckets in which fragment getters and alternate nodes are collected.
add_alternate_nodes
(node, frag_indexes)¶add_response
(getter, parts_iter)¶get_responses
()¶Return a list of all useful sources. Where there are multiple sources associated with the same frag_index then only one is included.
Returns: | a list of sources, each source being a tuple of form (ResumingGetter, iter) |
---|
set_durable
()¶shortfall
¶shortfall_with_alts
¶swift.proxy.controllers.obj.
ECGetResponseCollection
(policy)¶Bases: object
Manages all successful EC GET responses gathered by ResumingGetters.
A response comprises a tuple of (<getter instance>, <parts iterator>). All responses having the same data timestamp are placed in an ECGetResponseBucket for that timestamp. The buckets are stored in the ‘buckets’ dict which maps timestamp-> bucket.
This class encapsulates logic for selecting the best bucket from the collection, and for choosing alternate nodes.
add_response
(get, parts_iter)¶Add a response to the collection.
Parameters: |
|
---|---|
Raises: | ValueError – if the response etag or status code values do not match any values previously received for the same timestamp |
best_bucket
¶Return the best bucket in the collection.
The “best” bucket is the newest timestamp with sufficient getters, or the closest to having sufficient getters, unless it is bettered by a bucket with potential alternate nodes.
Returns: | An instance of ECGetResponseBucket or None if there
are no buckets in the collection. |
---|
get_extra_headers
()¶has_alternate_node
()¶provide_alternate_node
()¶Callback function that is installed in a NodeIter. Called on every call to NodeIter.next(), which means we can track the number of nodes to which GET requests have been made and selectively inject an alternate node, if we have one.
Returns: | A dict describing a node to which the next GET request should be made. |
---|
swift.proxy.controllers.obj.
ECObjectController
(app, account_name, container_name, object_name, **kwargs)¶Bases: swift.proxy.controllers.obj.BaseObjectController
policy_type
= 'erasure_coding'¶swift.proxy.controllers.obj.
MIMEPutter
(conn, node, resp, req, connect_duration, logger, mime_boundary, multiphase=False)¶Bases: swift.proxy.controllers.obj.Putter
Putter for backend PUT requests that use MIME.
This is here mostly to wrap up the fact that all multipart PUTs are chunked because of the mime boundary footer trick and the first half of the two-phase PUT conversation handling.
An HTTP PUT request that supports streaming.
connect
(node, part, req, headers, conn_timeout, node_timeout, logger=None, need_multiphase=True, **kwargs)¶Connect to a backend node and send the headers.
Override superclass method to notify object of need for support for multipart body with footers and optionally multiphase commit, and verify object server’s capabilities.
Parameters: | need_multiphase – if True then multiphase support is required of the object server |
---|---|
Raises: |
|
end_of_object_data
(footer_metadata=None)¶Call when there is no more data to send.
Overrides superclass implementation to send any footer metadata after object data.
Parameters: | footer_metadata – dictionary of metadata items to be sent as footers. |
---|
send_commit_confirmation
()¶Call when there are > quorum 2XX responses received. Send commit confirmations to all object nodes to finalize the PUT.
swift.proxy.controllers.obj.
ObjectControllerRouter
¶Bases: object
policy_type_to_controller_map
= {'erasure_coding': <class 'swift.proxy.controllers.obj.ECObjectController'>, 'replication': <class 'swift.proxy.controllers.obj.ReplicatedObjectController'>}¶register
(policy_type)¶Decorator for Storage Policy implementations to register their ObjectController implementations.
This also fills in a policy_type attribute on the class.
swift.proxy.controllers.obj.
Putter
(conn, node, resp, path, connect_duration, logger, chunked=False)¶Bases: object
Putter for backend PUT requests.
Encapsulates all the actions required to establish a connection with a storage node and stream data to that node.
Parameters: |
|
---|
await_response
(timeout, informational=False)¶Get 100-continue response indicating the end of 1st phase of a 2-phase commit or the final response, i.e. the one with status >= 200.
Might or might not actually wait for anything. If we said Expect: 100-continue but got back a non-100 response, that’ll be the thing returned, and we won’t do any network IO to get it. OTOH, if we got a 100 Continue response and sent up the PUT request’s body, then we’ll actually read the 2xx-5xx response off the network here.
Parameters: |
|
---|---|
Returns: | HTTPResponse |
Raises: | Timeout – if the response took too long |
close
()¶connect
(node, part, path, headers, conn_timeout, node_timeout, logger=None, chunked=False, **kwargs)¶Connect to a backend node and send the headers.
Returns: | Putter instance |
---|---|
Raises: |
|
end_of_object_data
(**kwargs)¶Call when there is no more data to send.
send_chunk
(chunk)¶spawn_sender_greenthread
(pool, queue_depth, write_timeout, exception_handler)¶Call before sending the first chunk of request body
wait
()¶swift.proxy.controllers.obj.
ReplicatedObjectController
(app, account_name, container_name, object_name, **kwargs)¶Bases: swift.proxy.controllers.obj.BaseObjectController
policy_type
= 'replication'¶swift.proxy.controllers.obj.
check_content_type
(req)¶swift.proxy.controllers.obj.
chunk_transformer
(policy)¶A generator to transform a source chunk to erasure coded chunks for each send call. The number of erasure coded chunks is as policy.ec_n_unique_fragments.
swift.proxy.controllers.obj.
client_range_to_segment_range
(client_start, client_end, segment_size)¶Takes a byterange from the client and converts it into a byterange spanning the necessary segments.
Handles prefix, suffix, and fully-specified byte ranges.
Parameters: |
|
---|---|
Returns: | a 2-tuple (seg_start, seg_end) where
|
swift.proxy.controllers.obj.
num_container_updates
(container_replicas, container_quorum, object_replicas, object_quorum)¶We need to send container updates via enough object servers such that, if the object PUT succeeds, then the container update is durable (either it’s synchronously updated or written to async pendings).
A durable container update is one that’s made it to at least Qc nodes. To always be durable, we have to send enough container updates so that, if only Qo object PUTs succeed, and all the failed object PUTs had container updates, at least Qc updates remain. Since (Ro - Qo) object PUTs may fail, we must have at least Qc + Ro - Qo container updates to ensure that Qc of them remain.
Also, each container replica is named in at least one object PUT request so that, when all requests succeed, no work is generated for the container replicator. Thus, at least Rc updates are necessary.
Parameters: |
|
---|
swift.proxy.controllers.obj.
segment_range_to_fragment_range
(segment_start, segment_end, segment_size, fragment_size)¶Takes a byterange spanning some segments and converts that into a byterange spanning the corresponding fragments within their fragment archives.
Handles prefix, suffix, and fully-specified byte ranges.
Parameters: |
|
---|---|
Returns: | a 2-tuple (frag_start, frag_end) where
|
swift.proxy.controllers.obj.
trailing_metadata
(policy, client_obj_hasher, bytes_transferred_from_client, fragment_archive_index)¶swift.proxy.server.
Application
(conf, memcache=None, logger=None, account_ring=None, container_ring=None)¶Bases: object
WSGI application for the proxy server.
check_config
()¶Check the configuration for possible errors
error_limit
(node, msg)¶Mark a node as error limited. This immediately pretends the
node received enough errors to trigger error suppression. Use
this for errors like Insufficient Storage. For other errors
use error_occurred()
.
Parameters: |
|
---|
error_limited
(node)¶Check if the node is currently error limited.
Parameters: | node – dictionary of node to check |
---|---|
Returns: | True if error limited, False otherwise |
error_occurred
(node, msg)¶Handle logging, and handling of errors.
Parameters: |
|
---|
exception_occurred
(node, typ, additional_info, **kwargs)¶Handle logging of generic exceptions.
Parameters: |
|
---|
get_controller
(req)¶Get the controller to handle a request.
Parameters: | req – the request |
---|---|
Returns: | tuple of (controller class, path dictionary) |
Raises: | ValueError – (thrown by split_path) if given invalid path |
get_object_ring
(policy_idx)¶Get the ring object to use to handle a request based on its policy.
Parameters: | policy_idx – policy index as defined in swift.conf |
---|---|
Returns: | appropriate ring object |
get_policy_options
(policy)¶Return policy specific options.
Parameters: | policy – an instance of BaseStoragePolicy or None |
---|---|
Returns: | an instance of ProxyOverrideOptions |
handle_request
(req)¶Entry point for proxy server. Should return a WSGI-style callable (such as swob.Response).
Parameters: | req – swob.Request object |
---|
iter_nodes
(ring, partition, node_iter=None, policy=None)¶modify_wsgi_pipeline
(pipe)¶Called during WSGI pipeline creation. Modifies the WSGI pipeline context to ensure that mandatory middleware is present in the pipeline.
Parameters: | pipe – A PipelineWrapper object |
---|
set_node_timing
(node, timing)¶sort_nodes
(nodes, policy=None)¶Sorts nodes in-place (and returns the sorted list) according to the configured strategy. The default “sorting” is to randomly shuffle the nodes. If the “timing” strategy is chosen, the nodes are sorted according to the stored timing data.
Parameters: |
|
---|
update_request
(req)¶swift.proxy.server.
ProxyOverrideOptions
(base_conf, override_conf)¶Bases: object
Encapsulates proxy server options that may be overridden e.g. for policy specific configurations.
Parameters: |
|
---|
swift.proxy.server.
app_factory
(global_conf, **local_conf)¶paste.deploy app factory for creating WSGI proxy apps.
swift.proxy.server.
parse_per_policy_config
(conf)¶Search the config file for any per-policy config sections and load those sections to a dict mapping policy reference (name or index) to policy options.
Parameters: | conf – the proxy server conf dict |
---|---|
Returns: | a dict mapping policy reference -> dict of policy options |
Raises: | ValueError – if a policy config section has an invalid name |
Except where otherwise noted, this document is licensed under Creative Commons Attribution 3.0 License. See all OpenStack Legal Documents.