1fromdataclassesimportdataclass
+ 2fromdatetimeimporttimedelta
+ 3fromtypingimportAny,Dict,Iterator,Mapping,Optional,Union,List
+ 4
+ 5importfauna
+ 6fromfauna.client.headersimport_DriverEnvironment,_Header,_Auth,Header
+ 7fromfauna.client.retryableimportRetryable
+ 8fromfauna.client.utilsimport_Environment,LastTxnTs
+ 9fromfauna.encodingimportFaunaEncoder,FaunaDecoder
+ 10fromfauna.encodingimportQuerySuccess,QueryTags,QueryStats
+ 11fromfauna.errorsimportFaunaError,ClientError,ProtocolError, \
+ 12RetryableFaunaException,NetworkError
+ 13fromfauna.http.http_clientimportHTTPClient
+ 14fromfauna.queryimportQuery,Page,fql
+ 15fromfauna.query.modelsimportStreamToken
+ 16
+ 17DefaultHttpConnectTimeout=timedelta(seconds=5)
+ 18DefaultHttpReadTimeout:Optional[timedelta]=None
+ 19DefaultHttpWriteTimeout=timedelta(seconds=5)
+ 20DefaultHttpPoolTimeout=timedelta(seconds=5)
+ 21DefaultIdleConnectionTimeout=timedelta(seconds=5)
+ 22DefaultQueryTimeout=timedelta(seconds=5)
+ 23DefaultClientBufferTimeout=timedelta(seconds=5)
+ 24DefaultMaxConnections=20
+ 25DefaultMaxIdleConnections=20
+ 26
+ 27
+ 28@dataclass
+ 29classQueryOptions:
+ 30"""
+ 31 A dataclass representing options available for a query.
+ 32
+ 33 * linearized - If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
+ 34 * max_contention_retries - The max number of times to retry the query if contention is encountered.
+ 35 * query_timeout - Controls the maximum amount of time Fauna will execute your query before marking it failed.
+ 36 * query_tags - Tags to associate with the query. See `logging <https://docs.fauna.com/fauna/current/build/logs/query_log/>`_
+ 37 * traceparent - A traceparent to associate with the query. See `logging <https://docs.fauna.com/fauna/current/build/logs/query_log/>`_ Must match format: https://www.w3.org/TR/trace-context/#traceparent-header
+ 38 * typecheck - Enable or disable typechecking of the query before evaluation. If not set, the value configured on the Client will be used. If neither is set, Fauna will use the value of the "typechecked" flag on the database configuration.
+ 39 * additional_headers - Add/update HTTP request headers for the query. In general, this should not be necessary.
+ 40 """
+ 41
+ 42linearized:Optional[bool]=None
+ 43max_contention_retries:Optional[int]=None
+ 44query_timeout:Optional[timedelta]=DefaultQueryTimeout
+ 45query_tags:Optional[Mapping[str,str]]=None
+ 46traceparent:Optional[str]=None
+ 47typecheck:Optional[bool]=None
+ 48additional_headers:Optional[Dict[str,str]]=None
+ 49
+ 50
+ 51@dataclass
+ 52classStreamOptions:
+ 53"""
+ 54 A dataclass representing options available for a stream.
+ 55
+ 56 * max_attempts - The maximum number of times to attempt a stream query when a retryable exception is thrown.
+ 57 * max_backoff - The maximum backoff in seconds for an individual retry.
+ 58 * start_ts - The starting timestamp of the stream, exclusive. If set, Fauna will return events starting after
+ 59 the timestamp.
+ 60 * cursor - The starting event cursor, exclusive. If set, Fauna will return events starting after the cursor.
+ 61 * status_events - Indicates if stream should include status events. Status events are periodic events that
+ 62 update the client with the latest valid timestamp (in the event of a dropped connection) as well as metrics
+ 63 about the cost of maintaining the stream other than the cost of the received events.
+ 64 """
+ 65
+ 66max_attempts:Optional[int]=None
+ 67max_backoff:Optional[int]=None
+ 68start_ts:Optional[int]=None
+ 69cursor:Optional[str]=None
+ 70status_events:bool=False
+ 71
+ 72
+ 73@dataclass
+ 74classChangeFeedOptions:
+ 75"""
+ 76 A dataclass representing options available for a change feed.
+ 77
+ 78 * max_attempts - The maximum number of times to attempt a change feed query when a retryable exception is thrown.
+ 79 * max_backoff - The maximum backoff in seconds for an individual retry.
+ 80 * query_timeout - Controls the maximum amount of time Fauna will execute a query before returning a page of events.
+ 81 * start_ts - The starting timestamp of the change feed, exclusive. If set, Fauna will return events starting after
+ 82 the timestamp.
+ 83 * cursor - The starting event cursor, exclusive. If set, Fauna will return events starting after the cursor.
+ 84 * page_size - The desired number of events per page.
+ 85 """
+ 86max_attempts:Optional[int]=None
+ 87max_backoff:Optional[int]=None
+ 88query_timeout:Optional[timedelta]=None
+ 89page_size:Optional[int]=None
+ 90start_ts:Optional[int]=None
+ 91cursor:Optional[str]=None
+ 92
+ 93
+ 94classClient:
+ 95
+ 96def__init__(
+ 97self,
+ 98endpoint:Optional[str]=None,
+ 99secret:Optional[str]=None,
+100http_client:Optional[HTTPClient]=None,
+101query_tags:Optional[Mapping[str,str]]=None,
+102linearized:Optional[bool]=None,
+103max_contention_retries:Optional[int]=None,
+104typecheck:Optional[bool]=None,
+105additional_headers:Optional[Dict[str,str]]=None,
+106query_timeout:Optional[timedelta]=DefaultQueryTimeout,
+107client_buffer_timeout:Optional[timedelta]=DefaultClientBufferTimeout,
+108http_read_timeout:Optional[timedelta]=DefaultHttpReadTimeout,
+109http_write_timeout:Optional[timedelta]=DefaultHttpWriteTimeout,
+110http_connect_timeout:Optional[timedelta]=DefaultHttpConnectTimeout,
+111http_pool_timeout:Optional[timedelta]=DefaultHttpPoolTimeout,
+112http_idle_timeout:Optional[timedelta]=DefaultIdleConnectionTimeout,
+113max_attempts:int=3,
+114max_backoff:int=20,
+115):
+116"""Initializes a Client.
+117
+118 :param endpoint: The Fauna Endpoint to use. Defaults to https://db.fauna.com, or the `FAUNA_ENDPOINT` env variable.
+119 :param secret: The Fauna Secret to use. Defaults to empty, or the `FAUNA_SECRET` env variable.
+120 :param http_client: An :class:`HTTPClient` implementation. Defaults to a global :class:`HTTPXClient`.
+121 :param query_tags: Tags to associate with the query. See `logging <https://docs.fauna.com/fauna/current/build/logs/query_log/>`_
+122 :param linearized: If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
+123 :param max_contention_retries: The max number of times to retry the query if contention is encountered.
+124 :param typecheck: Enable or disable typechecking of the query before evaluation. If not set, Fauna will use the value of the "typechecked" flag on the database configuration.
+125 :param additional_headers: Add/update HTTP request headers for the query. In general, this should not be necessary.
+126 :param query_timeout: Controls the maximum amount of time Fauna will execute your query before marking it failed, default is :py:data:`DefaultQueryTimeout`.
+127 :param client_buffer_timeout: Time in milliseconds beyond query_timeout at which the client will abort a request if it has not received a response. The default is :py:data:`DefaultClientBufferTimeout`, which should account for network latency for most clients. The value must be greater than zero. The closer to zero the value is, the more likely the client is to abort the request before the server can report a legitimate response or error.
+128 :param http_read_timeout: Set HTTP Read timeout, default is :py:data:`DefaultHttpReadTimeout`.
+129 :param http_write_timeout: Set HTTP Write timeout, default is :py:data:`DefaultHttpWriteTimeout`.
+130 :param http_connect_timeout: Set HTTP Connect timeout, default is :py:data:`DefaultHttpConnectTimeout`.
+131 :param http_pool_timeout: Set HTTP Pool timeout, default is :py:data:`DefaultHttpPoolTimeout`.
+132 :param http_idle_timeout: Set HTTP Idle timeout, default is :py:data:`DefaultIdleConnectionTimeout`.
+133 :param max_attempts: The maximum number of times to attempt a query when a retryable exception is thrown. Defaults to 3.
+134 :param max_backoff: The maximum backoff in seconds for an individual retry. Defaults to 20.
+135 """
+136
+137self._set_endpoint(endpoint)
+138self._max_attempts=max_attempts
+139self._max_backoff=max_backoff
+140
+141ifsecretisNone:
+142self._auth=_Auth(_Environment.EnvFaunaSecret())
+143else:
+144self._auth=_Auth(secret)
+145
+146self._last_txn_ts=LastTxnTs()
+147
+148self._query_tags={}
+149ifquery_tagsisnotNone:
+150self._query_tags.update(query_tags)
+151
+152ifquery_timeoutisnotNone:
+153self._query_timeout_ms=int(query_timeout.total_seconds()*1000)
+154else:
+155self._query_timeout_ms=None
+156
+157self._headers:Dict[str,str]={
+158_Header.AcceptEncoding:"gzip",
+159_Header.ContentType:"application/json;charset=utf-8",
+160_Header.Driver:"python",
+161_Header.DriverEnv:str(_DriverEnvironment()),
+162}
+163
+164iftypecheckisnotNone:
+165self._headers[Header.Typecheck]=str(typecheck).lower()
+166
+167iflinearizedisnotNone:
+168self._headers[Header.Linearized]=str(linearized).lower()
+169
+170ifmax_contention_retriesisnotNoneandmax_contention_retries>0:
+171self._headers[Header.MaxContentionRetries]= \
+172f"{max_contention_retries}"
+173
+174ifadditional_headersisnotNone:
+175self._headers={
+176**self._headers,
+177**additional_headers,
+178}
+179
+180self._session:HTTPClient
+181
+182ifhttp_clientisnotNone:
+183self._session=http_client
+184else:
+185iffauna.global_http_clientisNone:
+186timeout_s:Optional[float]=None
+187ifquery_timeoutisnotNoneandclient_buffer_timeoutisnotNone:
+188timeout_s=(query_timeout+client_buffer_timeout).total_seconds()
+189read_timeout_s:Optional[float]=None
+190ifhttp_read_timeoutisnotNone:
+191read_timeout_s=http_read_timeout.total_seconds()
+192
+193write_timeout_s:Optional[float]=http_write_timeout.total_seconds(
+194)ifhttp_write_timeoutisnotNoneelseNone
+195connect_timeout_s:Optional[float]=http_connect_timeout.total_seconds(
+196)ifhttp_connect_timeoutisnotNoneelseNone
+197pool_timeout_s:Optional[float]=http_pool_timeout.total_seconds(
+198)ifhttp_pool_timeoutisnotNoneelseNone
+199idle_timeout_s:Optional[float]=http_idle_timeout.total_seconds(
+200)ifhttp_idle_timeoutisnotNoneelseNone
+201
+202importhttpx
+203fromfauna.http.httpx_clientimportHTTPXClient
+204c=HTTPXClient(
+205httpx.Client(
+206http1=True,
+207http2=False,
+208timeout=httpx.Timeout(
+209timeout=timeout_s,
+210connect=connect_timeout_s,
+211read=read_timeout_s,
+212write=write_timeout_s,
+213pool=pool_timeout_s,
+214),
+215limits=httpx.Limits(
+216max_connections=DefaultMaxConnections,
+217max_keepalive_connections=DefaultMaxIdleConnections,
+218keepalive_expiry=idle_timeout_s,
+219),
+220))
+221fauna.global_http_client=c
+222
+223self._session=fauna.global_http_client
+224
+225defclose(self):
+226self._session.close()
+227ifself._session==fauna.global_http_client:
+228fauna.global_http_client=None
+229
+230defset_last_txn_ts(self,txn_ts:int):
+231"""
+232 Set the last timestamp seen by this client.
+233 This has no effect if earlier than stored timestamp.
+234
+235 .. WARNING:: This should be used only when coordinating timestamps across
+236 multiple clients. Moving the timestamp arbitrarily forward into
+237 the future will cause transactions to stall.
+238
+239 :param txn_ts: the new transaction time.
+240 """
+241self._last_txn_ts.update_txn_time(txn_ts)
+242
+243defget_last_txn_ts(self)->Optional[int]:
+244"""
+245 Get the last timestamp seen by this client.
+246 :return:
+247 """
+248returnself._last_txn_ts.time
+249
+250defget_query_timeout(self)->Optional[timedelta]:
+251"""
+252 Get the query timeout for all queries.
+253 """
+254ifself._query_timeout_msisnotNone:
+255returntimedelta(milliseconds=self._query_timeout_ms)
+256else:
+257returnNone
+258
+259defpaginate(
+260self,
+261fql:Query,
+262opts:Optional[QueryOptions]=None,
+263)->"QueryIterator":
+264"""
+265 Run a query on Fauna and returning an iterator of results. If the query
+266 returns a Page, the iterator will fetch additional Pages until the
+267 after token is null. Each call for a page will be retried with exponential
+268 backoff up to the max_attempts set in the client's retry policy in the
+269 event of a 429 or 502.
+270
+271 :param fql: A Query
+272 :param opts: (Optional) Query Options
+273
+274 :return: a :class:`QueryResponse`
+275
+276 :raises NetworkError: HTTP Request failed in transit
+277 :raises ProtocolError: HTTP error not from Fauna
+278 :raises ServiceError: Fauna returned an error
+279 :raises ValueError: Encoding and decoding errors
+280 :raises TypeError: Invalid param types
+281 """
+282
+283ifnotisinstance(fql,Query):
+284err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+285f"Query by calling fauna.fql()"
+286raiseTypeError(err_msg)
+287
+288returnQueryIterator(self,fql,opts)
+289
+290defquery(
+291self,
+292fql:Query,
+293opts:Optional[QueryOptions]=None,
+294)->QuerySuccess:
+295"""
+296 Run a query on Fauna. A query will be retried max_attempt times with exponential backoff
+297 up to the max_backoff in the event of a 429.
+298
+299 :param fql: A Query
+300 :param opts: (Optional) Query Options
+301
+302 :return: a :class:`QueryResponse`
+303
+304 :raises NetworkError: HTTP Request failed in transit
+305 :raises ProtocolError: HTTP error not from Fauna
+306 :raises ServiceError: Fauna returned an error
+307 :raises ValueError: Encoding and decoding errors
+308 :raises TypeError: Invalid param types
+309 """
+310
+311ifnotisinstance(fql,Query):
+312err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+313f"Query by calling fauna.fql()"
+314raiseTypeError(err_msg)
+315
+316try:
+317encoded_query:Mapping[str,Any]=FaunaEncoder.encode(fql)
+318exceptExceptionase:
+319raiseClientError("Failed to encode Query")frome
+320
+321retryable=Retryable[QuerySuccess](
+322self._max_attempts,
+323self._max_backoff,
+324self._query,
+325"/query/1",
+326fql=encoded_query,
+327opts=opts,
+328)
+329
+330r=retryable.run()
+331r.response.stats.attempts=r.attempts
+332returnr.response
+333
+334def_query(
+335self,
+336path:str,
+337fql:Mapping[str,Any],
+338arguments:Optional[Mapping[str,Any]]=None,
+339opts:Optional[QueryOptions]=None,
+340)->QuerySuccess:
+341
+342headers=self._headers.copy()
+343headers[_Header.Format]="tagged"
+344headers[_Header.Authorization]=self._auth.bearer()
+345
+346ifself._query_timeout_msisnotNone:
+347headers[Header.QueryTimeoutMs]=str(self._query_timeout_ms)
+348
+349headers.update(self._last_txn_ts.request_header)
+350
+351query_tags={}
+352ifself._query_tagsisnotNone:
+353query_tags.update(self._query_tags)
+354
+355ifoptsisnotNone:
+356ifopts.linearizedisnotNone:
+357headers[Header.Linearized]=str(opts.linearized).lower()
+358ifopts.max_contention_retriesisnotNone:
+359headers[Header.MaxContentionRetries]= \
+360f"{opts.max_contention_retries}"
+361ifopts.traceparentisnotNone:
+362headers[Header.Traceparent]=opts.traceparent
+363ifopts.query_timeoutisnotNone:
+364timeout_ms=f"{int(opts.query_timeout.total_seconds()*1000)}"
+365headers[Header.QueryTimeoutMs]=timeout_ms
+366ifopts.query_tagsisnotNone:
+367query_tags.update(opts.query_tags)
+368ifopts.typecheckisnotNone:
+369headers[Header.Typecheck]=str(opts.typecheck).lower()
+370ifopts.additional_headersisnotNone:
+371headers.update(opts.additional_headers)
+372
+373iflen(query_tags)>0:
+374headers[Header.Tags]=QueryTags.encode(query_tags)
+375
+376data:dict[str,Any]={
+377"query":fql,
+378"arguments":argumentsor{},
+379}
+380
+381withself._session.request(
+382method="POST",
+383url=self._endpoint+path,
+384headers=headers,
+385data=data,
+386)asresponse:
+387status_code=response.status_code()
+388response_json=response.json()
+389headers=response.headers()
+390
+391self._check_protocol(response_json,status_code)
+392
+393dec:Any=FaunaDecoder.decode(response_json)
+394
+395ifstatus_code>399:
+396FaunaError.parse_error_and_throw(dec,status_code)
+397
+398if"txn_ts"indec:
+399self.set_last_txn_ts(int(response_json["txn_ts"]))
+400
+401stats=QueryStats(dec["stats"])if"stats"indecelseNone
+402summary=dec["summary"]if"summary"indecelseNone
+403query_tags=QueryTags.decode(
+404dec["query_tags"])if"query_tags"indecelseNone
+405txn_ts=dec["txn_ts"]if"txn_ts"indecelseNone
+406schema_version=dec["schema_version"]if"schema_version"indecelseNone
+407traceparent=headers.get("traceparent",None)
+408static_type=dec["static_type"]if"static_type"indecelseNone
+409
+410returnQuerySuccess(
+411data=dec["data"],
+412query_tags=query_tags,
+413static_type=static_type,
+414stats=stats,
+415summary=summary,
+416traceparent=traceparent,
+417txn_ts=txn_ts,
+418schema_version=schema_version,
+419)
+420
+421defstream(
+422self,
+423fql:Union[StreamToken,Query],
+424opts:StreamOptions=StreamOptions()
+425)->"StreamIterator":
+426"""
+427 Opens a Stream in Fauna and returns an iterator that consume Fauna events.
+428
+429 :param fql: A Query that returns a StreamToken or a StreamToken.
+430 :param opts: (Optional) Stream Options.
+431
+432 :return: a :class:`StreamIterator`
+433
+434 :raises ClientError: Invalid options provided
+435 :raises NetworkError: HTTP Request failed in transit
+436 :raises ProtocolError: HTTP error not from Fauna
+437 :raises ServiceError: Fauna returned an error
+438 :raises ValueError: Encoding and decoding errors
+439 :raises TypeError: Invalid param types
+440 """
+441
+442ifisinstance(fql,Query):
+443ifopts.cursorisnotNone:
+444raiseClientError(
+445"The 'cursor' configuration can only be used with a stream token.")
+446
+447token=self.query(fql).data
+448else:
+449token=fql
+450
+451ifnotisinstance(token,StreamToken):
+452err_msg=f"'fql' must be a StreamToken, or a Query that returns a StreamToken but was a {type(token)}."
+453raiseTypeError(err_msg)
+454
+455headers=self._headers.copy()
+456headers[_Header.Format]="tagged"
+457headers[_Header.Authorization]=self._auth.bearer()
+458
+459returnStreamIterator(self._session,headers,self._endpoint+"/stream/1",
+460self._max_attempts,self._max_backoff,opts,token)
+461
+462defchange_feed(
+463self,
+464fql:Union[StreamToken,Query],
+465opts:ChangeFeedOptions=ChangeFeedOptions()
+466)->"ChangeFeedIterator":
+467"""
+468 Opens a change feed in Fauna and returns an iterator that consume Fauna events.
+469
+470 :param fql: A Query that returns a StreamToken or a StreamToken.
+471 :param opts: (Optional) Change feed options.
+472
+473 :return: a :class:`ChangeFeedIterator`
+474
+475 :raises ClientError: Invalid options provided
+476 :raises NetworkError: HTTP Request failed in transit
+477 :raises ProtocolError: HTTP error not from Fauna
+478 :raises ServiceError: Fauna returned an error
+479 :raises ValueError: Encoding and decoding errors
+480 :raises TypeError: Invalid param types
+481 """
+482
+483ifisinstance(fql,Query):
+484token=self.query(fql).data
+485else:
+486token=fql
+487
+488ifnotisinstance(token,StreamToken):
+489err_msg=f"'fql' must be a StreamToken, or a Query that returns a StreamToken but was a {type(token)}."
+490raiseTypeError(err_msg)
+491
+492headers=self._headers.copy()
+493headers[_Header.Format]="tagged"
+494headers[_Header.Authorization]=self._auth.bearer()
+495
+496ifopts.query_timeoutisnotNone:
+497query_timeout_ms=int(opts.query_timeout.total_seconds()*1000)
+498headers[Header.QueryTimeoutMs]=str(query_timeout_ms)
+499elifself._query_timeout_msisnotNone:
+500headers[Header.QueryTimeoutMs]=str(self._query_timeout_ms)
+501
+502returnChangeFeedIterator(self._session,headers,
+503self._endpoint+"/changefeed/1",
+504self._max_attempts,self._max_backoff,opts,
+505token)
+506
+507def_check_protocol(self,response_json:Any,status_code):
+508# TODO: Logic to validate wire protocol belongs elsewhere.
+509should_raise=False
+510
+511# check for QuerySuccess
+512ifstatus_code<=399and"data"notinresponse_json:
+513should_raise=True
+514
+515# check for QueryFailure
+516ifstatus_code>399:
+517if"error"notinresponse_json:
+518should_raise=True
+519else:
+520e=response_json["error"]
+521if"code"notineor"message"notine:
+522should_raise=True
+523
+524ifshould_raise:
+525raiseProtocolError(
+526status_code,
+527f"Response is in an unknown format: \n{response_json}",
+528)
+529
+530def_set_endpoint(self,endpoint):
+531ifendpointisNone:
+532endpoint=_Environment.EnvFaunaEndpoint()
+533
+534ifendpoint[-1:]=="/":
+535endpoint=endpoint[:-1]
+536
+537self._endpoint=endpoint
+538
+539
+540classStreamIterator:
+541"""A class that mixes a ContextManager and an Iterator so we can detected retryable errors."""
+542
+543def__init__(self,http_client:HTTPClient,headers:Dict[str,str],
+544endpoint:str,max_attempts:int,max_backoff:int,
+545opts:StreamOptions,token:StreamToken):
+546self._http_client=http_client
+547self._headers=headers
+548self._endpoint=endpoint
+549self._max_attempts=max_attempts
+550self._max_backoff=max_backoff
+551self._opts=opts
+552self._token=token
+553self._stream=None
+554self.last_ts=None
+555self.last_cursor=None
+556self._ctx=self._create_stream()
+557
+558def__enter__(self):
+559returnself
+560
+561def__exit__(self,exc_type,exc_value,exc_traceback):
+562ifself._streamisnotNone:
+563self._stream.close()
+564
+565self._ctx.__exit__(exc_type,exc_value,exc_traceback)
+566returnFalse
+567
+568def__iter__(self):
+569returnself
+570
+571def__next__(self):
+572ifself._opts.max_attemptsisnotNone:
+573max_attempts=self._opts.max_attempts
+574else:
+575max_attempts=self._max_attempts
+576
+577ifself._opts.max_backoffisnotNone:
+578max_backoff=self._opts.max_backoff
+579else:
+580max_backoff=self._max_backoff
+581
+582retryable=Retryable[Any](max_attempts,max_backoff,self._next_element)
+583returnretryable.run().response
+584
+585def_next_element(self):
+586try:
+587ifself._streamisNone:
+588try:
+589self._stream=self._ctx.__enter__()
+590exceptException:
+591self._retry_stream()
+592
+593ifself._streamisnotNone:
+594event:Any=FaunaDecoder.decode(next(self._stream))
+595
+596ifevent["type"]=="error":
+597FaunaError.parse_error_and_throw(event,400)
+598
+599self.last_ts=event["txn_ts"]
+600self.last_cursor=event.get('cursor')
+601
+602ifevent["type"]=="start":
+603returnself._next_element()
+604
+605ifnotself._opts.status_eventsandevent["type"]=="status":
+606returnself._next_element()
+607
+608returnevent
+609
+610raiseStopIteration
+611exceptNetworkError:
+612self._retry_stream()
+613
+614def_retry_stream(self):
+615ifself._streamisnotNone:
+616self._stream.close()
+617
+618self._stream=None
+619
+620try:
+621self._ctx=self._create_stream()
+622exceptException:
+623pass
+624raiseRetryableFaunaException
+625
+626def_create_stream(self):
+627data:Dict[str,Any]={"token":self._token.token}
+628ifself.last_cursorisnotNone:
+629data["cursor"]=self.last_cursor
+630elifself._opts.cursorisnotNone:
+631data["cursor"]=self._opts.cursor
+632elifself._opts.start_tsisnotNone:
+633data["start_ts"]=self._opts.start_ts
+634
+635returnself._http_client.stream(
+636url=self._endpoint,headers=self._headers,data=data)
+637
+638defclose(self):
+639ifself._streamisnotNone:
+640self._stream.close()
+641
+642
+643classChangeFeedPage:
+644
+645def__init__(self,events:List[Any],cursor:str,stats:QueryStats):
+646self._events=events
+647self.cursor=cursor
+648self.stats=stats
+649
+650def__len__(self):
+651returnlen(self._events)
+652
+653def__iter__(self)->Iterator[Any]:
+654foreventinself._events:
+655ifevent["type"]=="error":
+656FaunaError.parse_error_and_throw(event,400)
+657yieldevent
+658
+659
+660classChangeFeedIterator:
+661"""A class to provide an iterator on top of change feed pages."""
+662
+663def__init__(self,http:HTTPClient,headers:Dict[str,str],endpoint:str,
+664max_attempts:int,max_backoff:int,opts:ChangeFeedOptions,
+665token:StreamToken):
+666self._http=http
+667self._headers=headers
+668self._endpoint=endpoint
+669self._max_attempts=opts.max_attemptsormax_attempts
+670self._max_backoff=opts.max_backofformax_backoff
+671self._request:Dict[str,Any]={"token":token.token}
+672self._is_done=False
+673
+674ifopts.page_sizeisnotNone:
+675self._request["page_size"]=opts.page_size
+676
+677ifopts.cursorisnotNone:
+678self._request["cursor"]=opts.cursor
+679elifopts.start_tsisnotNone:
+680self._request["start_ts"]=opts.start_ts
+681
+682def__iter__(self)->Iterator[ChangeFeedPage]:
+683self._is_done=False
+684returnself
+685
+686def__next__(self)->ChangeFeedPage:
+687ifself._is_done:
+688raiseStopIteration
+689
+690retryable=Retryable[Any](self._max_attempts,self._max_backoff,
+691self._next_page)
+692returnretryable.run().response
+693
+694def_next_page(self)->ChangeFeedPage:
+695withself._http.request(
+696method="POST",
+697url=self._endpoint,
+698headers=self._headers,
+699data=self._request,
+700)asresponse:
+701status_code=response.status_code()
+702decoded:Any=FaunaDecoder.decode(response.json())
+703
+704ifstatus_code>399:
+705FaunaError.parse_error_and_throw(decoded,status_code)
+706
+707self._is_done=notdecoded["has_next"]
+708self._request["cursor"]=decoded["cursor"]
+709
+710if"start_ts"inself._request:
+711delself._request["start_ts"]
+712
+713returnChangeFeedPage(decoded["events"],decoded["cursor"],
+714QueryStats(decoded["stats"]))
+715
+716defflatten(self)->Iterator:
+717"""A generator that yields events instead of pages of events."""
+718forpageinself:
+719foreventinpage:
+720yieldevent
+721
+722
+723classQueryIterator:
+724"""A class to provider an iterator on top of Fauna queries."""
+725
+726def__init__(self,
+727client:Client,
+728fql:Query,
+729opts:Optional[QueryOptions]=None):
+730"""Initializes the QueryIterator
+731
+732 :param fql: A Query
+733 :param opts: (Optional) Query Options
+734
+735 :raises TypeError: Invalid param types
+736 """
+737ifnotisinstance(client,Client):
+738err_msg=f"'client' must be a Client but was a {type(client)}. You can build a " \
+739f"Client by calling fauna.client.Client()"
+740raiseTypeError(err_msg)
+741
+742ifnotisinstance(fql,Query):
+743err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+744f"Query by calling fauna.fql()"
+745raiseTypeError(err_msg)
+746
+747self.client=client
+748self.fql=fql
+749self.opts=opts
+750
+751def__iter__(self)->Iterator:
+752returnself.iter()
+753
+754defiter(self)->Iterator:
+755"""
+756 A generator function that immediately fetches and yields the results of
+757 the stored query. Yields additional pages on subsequent iterations if
+758 they exist
+759 """
+760
+761cursor=None
+762initial_response=self.client.query(self.fql,self.opts)
+763
+764ifisinstance(initial_response.data,Page):
+765cursor=initial_response.data.after
+766yieldinitial_response.data.data
+767
+768whilecursorisnotNone:
+769next_response=self.client.query(
+770fql("Set.paginate(${after})",after=cursor),self.opts)
+771# TODO: `Set.paginate` does not yet return a `@set` tagged value
+772# so we will get back a plain object that might not have
+773# an after property.
+774cursor=next_response.data.get("after")
+775yieldnext_response.data.get("data")
+776
+777else:
+778yield[initial_response.data]
+779
+780defflatten(self)->Iterator:
+781"""
+782 A generator function that immediately fetches and yields the results of
+783 the stored query. Yields each item individually, rather than a whole
+784 Page at a time. Fetches additional pages as required if they exist.
+785 """
+786
+787forpageinself.iter():
+788foriteminpage:
+789yielditem
+
29@dataclass
+30classQueryOptions:
+31"""
+32 A dataclass representing options available for a query.
+33
+34 * linearized - If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
+35 * max_contention_retries - The max number of times to retry the query if contention is encountered.
+36 * query_timeout - Controls the maximum amount of time Fauna will execute your query before marking it failed.
+37 * query_tags - Tags to associate with the query. See `logging <https://docs.fauna.com/fauna/current/build/logs/query_log/>`_
+38 * traceparent - A traceparent to associate with the query. See `logging <https://docs.fauna.com/fauna/current/build/logs/query_log/>`_ Must match format: https://www.w3.org/TR/trace-context/#traceparent-header
+39 * typecheck - Enable or disable typechecking of the query before evaluation. If not set, the value configured on the Client will be used. If neither is set, Fauna will use the value of the "typechecked" flag on the database configuration.
+40 * additional_headers - Add/update HTTP request headers for the query. In general, this should not be necessary.
+41 """
+42
+43linearized:Optional[bool]=None
+44max_contention_retries:Optional[int]=None
+45query_timeout:Optional[timedelta]=DefaultQueryTimeout
+46query_tags:Optional[Mapping[str,str]]=None
+47traceparent:Optional[str]=None
+48typecheck:Optional[bool]=None
+49additional_headers:Optional[Dict[str,str]]=None
+
+
+
+
A dataclass representing options available for a query.
+
+
+
linearized - If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
+
max_contention_retries - The max number of times to retry the query if contention is encountered.
+
query_timeout - Controls the maximum amount of time Fauna will execute your query before marking it failed.
+
query_tags - Tags to associate with the query. See logging
typecheck - Enable or disable typechecking of the query before evaluation. If not set, the value configured on the Client will be used. If neither is set, Fauna will use the value of the "typechecked" flag on the database configuration.
+
additional_headers - Add/update HTTP request headers for the query. In general, this should not be necessary.
52@dataclass
+53classStreamOptions:
+54"""
+55 A dataclass representing options available for a stream.
+56
+57 * max_attempts - The maximum number of times to attempt a stream query when a retryable exception is thrown.
+58 * max_backoff - The maximum backoff in seconds for an individual retry.
+59 * start_ts - The starting timestamp of the stream, exclusive. If set, Fauna will return events starting after
+60 the timestamp.
+61 * cursor - The starting event cursor, exclusive. If set, Fauna will return events starting after the cursor.
+62 * status_events - Indicates if stream should include status events. Status events are periodic events that
+63 update the client with the latest valid timestamp (in the event of a dropped connection) as well as metrics
+64 about the cost of maintaining the stream other than the cost of the received events.
+65 """
+66
+67max_attempts:Optional[int]=None
+68max_backoff:Optional[int]=None
+69start_ts:Optional[int]=None
+70cursor:Optional[str]=None
+71status_events:bool=False
+
+
+
+
A dataclass representing options available for a stream.
+
+
+
max_attempts - The maximum number of times to attempt a stream query when a retryable exception is thrown.
+
max_backoff - The maximum backoff in seconds for an individual retry.
+
start_ts - The starting timestamp of the stream, exclusive. If set, Fauna will return events starting after
+the timestamp.
+
cursor - The starting event cursor, exclusive. If set, Fauna will return events starting after the cursor.
+
status_events - Indicates if stream should include status events. Status events are periodic events that
+update the client with the latest valid timestamp (in the event of a dropped connection) as well as metrics
+about the cost of maintaining the stream other than the cost of the received events.
74@dataclass
+75classChangeFeedOptions:
+76"""
+77 A dataclass representing options available for a change feed.
+78
+79 * max_attempts - The maximum number of times to attempt a change feed query when a retryable exception is thrown.
+80 * max_backoff - The maximum backoff in seconds for an individual retry.
+81 * query_timeout - Controls the maximum amount of time Fauna will execute a query before returning a page of events.
+82 * start_ts - The starting timestamp of the change feed, exclusive. If set, Fauna will return events starting after
+83 the timestamp.
+84 * cursor - The starting event cursor, exclusive. If set, Fauna will return events starting after the cursor.
+85 * page_size - The desired number of events per page.
+86 """
+87max_attempts:Optional[int]=None
+88max_backoff:Optional[int]=None
+89query_timeout:Optional[timedelta]=None
+90page_size:Optional[int]=None
+91start_ts:Optional[int]=None
+92cursor:Optional[str]=None
+
+
+
+
A dataclass representing options available for a change feed.
+
+
+
max_attempts - The maximum number of times to attempt a change feed query when a retryable exception is thrown.
+
max_backoff - The maximum backoff in seconds for an individual retry.
+
query_timeout - Controls the maximum amount of time Fauna will execute a query before returning a page of events.
+
start_ts - The starting timestamp of the change feed, exclusive. If set, Fauna will return events starting after
+the timestamp.
+
cursor - The starting event cursor, exclusive. If set, Fauna will return events starting after the cursor.
+
page_size - The desired number of events per page.
95classClient:
+ 96
+ 97def__init__(
+ 98self,
+ 99endpoint:Optional[str]=None,
+100secret:Optional[str]=None,
+101http_client:Optional[HTTPClient]=None,
+102query_tags:Optional[Mapping[str,str]]=None,
+103linearized:Optional[bool]=None,
+104max_contention_retries:Optional[int]=None,
+105typecheck:Optional[bool]=None,
+106additional_headers:Optional[Dict[str,str]]=None,
+107query_timeout:Optional[timedelta]=DefaultQueryTimeout,
+108client_buffer_timeout:Optional[timedelta]=DefaultClientBufferTimeout,
+109http_read_timeout:Optional[timedelta]=DefaultHttpReadTimeout,
+110http_write_timeout:Optional[timedelta]=DefaultHttpWriteTimeout,
+111http_connect_timeout:Optional[timedelta]=DefaultHttpConnectTimeout,
+112http_pool_timeout:Optional[timedelta]=DefaultHttpPoolTimeout,
+113http_idle_timeout:Optional[timedelta]=DefaultIdleConnectionTimeout,
+114max_attempts:int=3,
+115max_backoff:int=20,
+116):
+117"""Initializes a Client.
+118
+119 :param endpoint: The Fauna Endpoint to use. Defaults to https://db.fauna.com, or the `FAUNA_ENDPOINT` env variable.
+120 :param secret: The Fauna Secret to use. Defaults to empty, or the `FAUNA_SECRET` env variable.
+121 :param http_client: An :class:`HTTPClient` implementation. Defaults to a global :class:`HTTPXClient`.
+122 :param query_tags: Tags to associate with the query. See `logging <https://docs.fauna.com/fauna/current/build/logs/query_log/>`_
+123 :param linearized: If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
+124 :param max_contention_retries: The max number of times to retry the query if contention is encountered.
+125 :param typecheck: Enable or disable typechecking of the query before evaluation. If not set, Fauna will use the value of the "typechecked" flag on the database configuration.
+126 :param additional_headers: Add/update HTTP request headers for the query. In general, this should not be necessary.
+127 :param query_timeout: Controls the maximum amount of time Fauna will execute your query before marking it failed, default is :py:data:`DefaultQueryTimeout`.
+128 :param client_buffer_timeout: Time in milliseconds beyond query_timeout at which the client will abort a request if it has not received a response. The default is :py:data:`DefaultClientBufferTimeout`, which should account for network latency for most clients. The value must be greater than zero. The closer to zero the value is, the more likely the client is to abort the request before the server can report a legitimate response or error.
+129 :param http_read_timeout: Set HTTP Read timeout, default is :py:data:`DefaultHttpReadTimeout`.
+130 :param http_write_timeout: Set HTTP Write timeout, default is :py:data:`DefaultHttpWriteTimeout`.
+131 :param http_connect_timeout: Set HTTP Connect timeout, default is :py:data:`DefaultHttpConnectTimeout`.
+132 :param http_pool_timeout: Set HTTP Pool timeout, default is :py:data:`DefaultHttpPoolTimeout`.
+133 :param http_idle_timeout: Set HTTP Idle timeout, default is :py:data:`DefaultIdleConnectionTimeout`.
+134 :param max_attempts: The maximum number of times to attempt a query when a retryable exception is thrown. Defaults to 3.
+135 :param max_backoff: The maximum backoff in seconds for an individual retry. Defaults to 20.
+136 """
+137
+138self._set_endpoint(endpoint)
+139self._max_attempts=max_attempts
+140self._max_backoff=max_backoff
+141
+142ifsecretisNone:
+143self._auth=_Auth(_Environment.EnvFaunaSecret())
+144else:
+145self._auth=_Auth(secret)
+146
+147self._last_txn_ts=LastTxnTs()
+148
+149self._query_tags={}
+150ifquery_tagsisnotNone:
+151self._query_tags.update(query_tags)
+152
+153ifquery_timeoutisnotNone:
+154self._query_timeout_ms=int(query_timeout.total_seconds()*1000)
+155else:
+156self._query_timeout_ms=None
+157
+158self._headers:Dict[str,str]={
+159_Header.AcceptEncoding:"gzip",
+160_Header.ContentType:"application/json;charset=utf-8",
+161_Header.Driver:"python",
+162_Header.DriverEnv:str(_DriverEnvironment()),
+163}
+164
+165iftypecheckisnotNone:
+166self._headers[Header.Typecheck]=str(typecheck).lower()
+167
+168iflinearizedisnotNone:
+169self._headers[Header.Linearized]=str(linearized).lower()
+170
+171ifmax_contention_retriesisnotNoneandmax_contention_retries>0:
+172self._headers[Header.MaxContentionRetries]= \
+173f"{max_contention_retries}"
+174
+175ifadditional_headersisnotNone:
+176self._headers={
+177**self._headers,
+178**additional_headers,
+179}
+180
+181self._session:HTTPClient
+182
+183ifhttp_clientisnotNone:
+184self._session=http_client
+185else:
+186iffauna.global_http_clientisNone:
+187timeout_s:Optional[float]=None
+188ifquery_timeoutisnotNoneandclient_buffer_timeoutisnotNone:
+189timeout_s=(query_timeout+client_buffer_timeout).total_seconds()
+190read_timeout_s:Optional[float]=None
+191ifhttp_read_timeoutisnotNone:
+192read_timeout_s=http_read_timeout.total_seconds()
+193
+194write_timeout_s:Optional[float]=http_write_timeout.total_seconds(
+195)ifhttp_write_timeoutisnotNoneelseNone
+196connect_timeout_s:Optional[float]=http_connect_timeout.total_seconds(
+197)ifhttp_connect_timeoutisnotNoneelseNone
+198pool_timeout_s:Optional[float]=http_pool_timeout.total_seconds(
+199)ifhttp_pool_timeoutisnotNoneelseNone
+200idle_timeout_s:Optional[float]=http_idle_timeout.total_seconds(
+201)ifhttp_idle_timeoutisnotNoneelseNone
+202
+203importhttpx
+204fromfauna.http.httpx_clientimportHTTPXClient
+205c=HTTPXClient(
+206httpx.Client(
+207http1=True,
+208http2=False,
+209timeout=httpx.Timeout(
+210timeout=timeout_s,
+211connect=connect_timeout_s,
+212read=read_timeout_s,
+213write=write_timeout_s,
+214pool=pool_timeout_s,
+215),
+216limits=httpx.Limits(
+217max_connections=DefaultMaxConnections,
+218max_keepalive_connections=DefaultMaxIdleConnections,
+219keepalive_expiry=idle_timeout_s,
+220),
+221))
+222fauna.global_http_client=c
+223
+224self._session=fauna.global_http_client
+225
+226defclose(self):
+227self._session.close()
+228ifself._session==fauna.global_http_client:
+229fauna.global_http_client=None
+230
+231defset_last_txn_ts(self,txn_ts:int):
+232"""
+233 Set the last timestamp seen by this client.
+234 This has no effect if earlier than stored timestamp.
+235
+236 .. WARNING:: This should be used only when coordinating timestamps across
+237 multiple clients. Moving the timestamp arbitrarily forward into
+238 the future will cause transactions to stall.
+239
+240 :param txn_ts: the new transaction time.
+241 """
+242self._last_txn_ts.update_txn_time(txn_ts)
+243
+244defget_last_txn_ts(self)->Optional[int]:
+245"""
+246 Get the last timestamp seen by this client.
+247 :return:
+248 """
+249returnself._last_txn_ts.time
+250
+251defget_query_timeout(self)->Optional[timedelta]:
+252"""
+253 Get the query timeout for all queries.
+254 """
+255ifself._query_timeout_msisnotNone:
+256returntimedelta(milliseconds=self._query_timeout_ms)
+257else:
+258returnNone
+259
+260defpaginate(
+261self,
+262fql:Query,
+263opts:Optional[QueryOptions]=None,
+264)->"QueryIterator":
+265"""
+266 Run a query on Fauna and returning an iterator of results. If the query
+267 returns a Page, the iterator will fetch additional Pages until the
+268 after token is null. Each call for a page will be retried with exponential
+269 backoff up to the max_attempts set in the client's retry policy in the
+270 event of a 429 or 502.
+271
+272 :param fql: A Query
+273 :param opts: (Optional) Query Options
+274
+275 :return: a :class:`QueryResponse`
+276
+277 :raises NetworkError: HTTP Request failed in transit
+278 :raises ProtocolError: HTTP error not from Fauna
+279 :raises ServiceError: Fauna returned an error
+280 :raises ValueError: Encoding and decoding errors
+281 :raises TypeError: Invalid param types
+282 """
+283
+284ifnotisinstance(fql,Query):
+285err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+286f"Query by calling fauna.fql()"
+287raiseTypeError(err_msg)
+288
+289returnQueryIterator(self,fql,opts)
+290
+291defquery(
+292self,
+293fql:Query,
+294opts:Optional[QueryOptions]=None,
+295)->QuerySuccess:
+296"""
+297 Run a query on Fauna. A query will be retried max_attempt times with exponential backoff
+298 up to the max_backoff in the event of a 429.
+299
+300 :param fql: A Query
+301 :param opts: (Optional) Query Options
+302
+303 :return: a :class:`QueryResponse`
+304
+305 :raises NetworkError: HTTP Request failed in transit
+306 :raises ProtocolError: HTTP error not from Fauna
+307 :raises ServiceError: Fauna returned an error
+308 :raises ValueError: Encoding and decoding errors
+309 :raises TypeError: Invalid param types
+310 """
+311
+312ifnotisinstance(fql,Query):
+313err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+314f"Query by calling fauna.fql()"
+315raiseTypeError(err_msg)
+316
+317try:
+318encoded_query:Mapping[str,Any]=FaunaEncoder.encode(fql)
+319exceptExceptionase:
+320raiseClientError("Failed to encode Query")frome
+321
+322retryable=Retryable[QuerySuccess](
+323self._max_attempts,
+324self._max_backoff,
+325self._query,
+326"/query/1",
+327fql=encoded_query,
+328opts=opts,
+329)
+330
+331r=retryable.run()
+332r.response.stats.attempts=r.attempts
+333returnr.response
+334
+335def_query(
+336self,
+337path:str,
+338fql:Mapping[str,Any],
+339arguments:Optional[Mapping[str,Any]]=None,
+340opts:Optional[QueryOptions]=None,
+341)->QuerySuccess:
+342
+343headers=self._headers.copy()
+344headers[_Header.Format]="tagged"
+345headers[_Header.Authorization]=self._auth.bearer()
+346
+347ifself._query_timeout_msisnotNone:
+348headers[Header.QueryTimeoutMs]=str(self._query_timeout_ms)
+349
+350headers.update(self._last_txn_ts.request_header)
+351
+352query_tags={}
+353ifself._query_tagsisnotNone:
+354query_tags.update(self._query_tags)
+355
+356ifoptsisnotNone:
+357ifopts.linearizedisnotNone:
+358headers[Header.Linearized]=str(opts.linearized).lower()
+359ifopts.max_contention_retriesisnotNone:
+360headers[Header.MaxContentionRetries]= \
+361f"{opts.max_contention_retries}"
+362ifopts.traceparentisnotNone:
+363headers[Header.Traceparent]=opts.traceparent
+364ifopts.query_timeoutisnotNone:
+365timeout_ms=f"{int(opts.query_timeout.total_seconds()*1000)}"
+366headers[Header.QueryTimeoutMs]=timeout_ms
+367ifopts.query_tagsisnotNone:
+368query_tags.update(opts.query_tags)
+369ifopts.typecheckisnotNone:
+370headers[Header.Typecheck]=str(opts.typecheck).lower()
+371ifopts.additional_headersisnotNone:
+372headers.update(opts.additional_headers)
+373
+374iflen(query_tags)>0:
+375headers[Header.Tags]=QueryTags.encode(query_tags)
+376
+377data:dict[str,Any]={
+378"query":fql,
+379"arguments":argumentsor{},
+380}
+381
+382withself._session.request(
+383method="POST",
+384url=self._endpoint+path,
+385headers=headers,
+386data=data,
+387)asresponse:
+388status_code=response.status_code()
+389response_json=response.json()
+390headers=response.headers()
+391
+392self._check_protocol(response_json,status_code)
+393
+394dec:Any=FaunaDecoder.decode(response_json)
+395
+396ifstatus_code>399:
+397FaunaError.parse_error_and_throw(dec,status_code)
+398
+399if"txn_ts"indec:
+400self.set_last_txn_ts(int(response_json["txn_ts"]))
+401
+402stats=QueryStats(dec["stats"])if"stats"indecelseNone
+403summary=dec["summary"]if"summary"indecelseNone
+404query_tags=QueryTags.decode(
+405dec["query_tags"])if"query_tags"indecelseNone
+406txn_ts=dec["txn_ts"]if"txn_ts"indecelseNone
+407schema_version=dec["schema_version"]if"schema_version"indecelseNone
+408traceparent=headers.get("traceparent",None)
+409static_type=dec["static_type"]if"static_type"indecelseNone
+410
+411returnQuerySuccess(
+412data=dec["data"],
+413query_tags=query_tags,
+414static_type=static_type,
+415stats=stats,
+416summary=summary,
+417traceparent=traceparent,
+418txn_ts=txn_ts,
+419schema_version=schema_version,
+420)
+421
+422defstream(
+423self,
+424fql:Union[StreamToken,Query],
+425opts:StreamOptions=StreamOptions()
+426)->"StreamIterator":
+427"""
+428 Opens a Stream in Fauna and returns an iterator that consume Fauna events.
+429
+430 :param fql: A Query that returns a StreamToken or a StreamToken.
+431 :param opts: (Optional) Stream Options.
+432
+433 :return: a :class:`StreamIterator`
+434
+435 :raises ClientError: Invalid options provided
+436 :raises NetworkError: HTTP Request failed in transit
+437 :raises ProtocolError: HTTP error not from Fauna
+438 :raises ServiceError: Fauna returned an error
+439 :raises ValueError: Encoding and decoding errors
+440 :raises TypeError: Invalid param types
+441 """
+442
+443ifisinstance(fql,Query):
+444ifopts.cursorisnotNone:
+445raiseClientError(
+446"The 'cursor' configuration can only be used with a stream token.")
+447
+448token=self.query(fql).data
+449else:
+450token=fql
+451
+452ifnotisinstance(token,StreamToken):
+453err_msg=f"'fql' must be a StreamToken, or a Query that returns a StreamToken but was a {type(token)}."
+454raiseTypeError(err_msg)
+455
+456headers=self._headers.copy()
+457headers[_Header.Format]="tagged"
+458headers[_Header.Authorization]=self._auth.bearer()
+459
+460returnStreamIterator(self._session,headers,self._endpoint+"/stream/1",
+461self._max_attempts,self._max_backoff,opts,token)
+462
+463defchange_feed(
+464self,
+465fql:Union[StreamToken,Query],
+466opts:ChangeFeedOptions=ChangeFeedOptions()
+467)->"ChangeFeedIterator":
+468"""
+469 Opens a change feed in Fauna and returns an iterator that consume Fauna events.
+470
+471 :param fql: A Query that returns a StreamToken or a StreamToken.
+472 :param opts: (Optional) Change feed options.
+473
+474 :return: a :class:`ChangeFeedIterator`
+475
+476 :raises ClientError: Invalid options provided
+477 :raises NetworkError: HTTP Request failed in transit
+478 :raises ProtocolError: HTTP error not from Fauna
+479 :raises ServiceError: Fauna returned an error
+480 :raises ValueError: Encoding and decoding errors
+481 :raises TypeError: Invalid param types
+482 """
+483
+484ifisinstance(fql,Query):
+485token=self.query(fql).data
+486else:
+487token=fql
+488
+489ifnotisinstance(token,StreamToken):
+490err_msg=f"'fql' must be a StreamToken, or a Query that returns a StreamToken but was a {type(token)}."
+491raiseTypeError(err_msg)
+492
+493headers=self._headers.copy()
+494headers[_Header.Format]="tagged"
+495headers[_Header.Authorization]=self._auth.bearer()
+496
+497ifopts.query_timeoutisnotNone:
+498query_timeout_ms=int(opts.query_timeout.total_seconds()*1000)
+499headers[Header.QueryTimeoutMs]=str(query_timeout_ms)
+500elifself._query_timeout_msisnotNone:
+501headers[Header.QueryTimeoutMs]=str(self._query_timeout_ms)
+502
+503returnChangeFeedIterator(self._session,headers,
+504self._endpoint+"/changefeed/1",
+505self._max_attempts,self._max_backoff,opts,
+506token)
+507
+508def_check_protocol(self,response_json:Any,status_code):
+509# TODO: Logic to validate wire protocol belongs elsewhere.
+510should_raise=False
+511
+512# check for QuerySuccess
+513ifstatus_code<=399and"data"notinresponse_json:
+514should_raise=True
+515
+516# check for QueryFailure
+517ifstatus_code>399:
+518if"error"notinresponse_json:
+519should_raise=True
+520else:
+521e=response_json["error"]
+522if"code"notineor"message"notine:
+523should_raise=True
+524
+525ifshould_raise:
+526raiseProtocolError(
+527status_code,
+528f"Response is in an unknown format: \n{response_json}",
+529)
+530
+531def_set_endpoint(self,endpoint):
+532ifendpointisNone:
+533endpoint=_Environment.EnvFaunaEndpoint()
+534
+535ifendpoint[-1:]=="/":
+536endpoint=endpoint[:-1]
+537
+538self._endpoint=endpoint
+
97def__init__(
+ 98self,
+ 99endpoint:Optional[str]=None,
+100secret:Optional[str]=None,
+101http_client:Optional[HTTPClient]=None,
+102query_tags:Optional[Mapping[str,str]]=None,
+103linearized:Optional[bool]=None,
+104max_contention_retries:Optional[int]=None,
+105typecheck:Optional[bool]=None,
+106additional_headers:Optional[Dict[str,str]]=None,
+107query_timeout:Optional[timedelta]=DefaultQueryTimeout,
+108client_buffer_timeout:Optional[timedelta]=DefaultClientBufferTimeout,
+109http_read_timeout:Optional[timedelta]=DefaultHttpReadTimeout,
+110http_write_timeout:Optional[timedelta]=DefaultHttpWriteTimeout,
+111http_connect_timeout:Optional[timedelta]=DefaultHttpConnectTimeout,
+112http_pool_timeout:Optional[timedelta]=DefaultHttpPoolTimeout,
+113http_idle_timeout:Optional[timedelta]=DefaultIdleConnectionTimeout,
+114max_attempts:int=3,
+115max_backoff:int=20,
+116):
+117"""Initializes a Client.
+118
+119 :param endpoint: The Fauna Endpoint to use. Defaults to https://db.fauna.com, or the `FAUNA_ENDPOINT` env variable.
+120 :param secret: The Fauna Secret to use. Defaults to empty, or the `FAUNA_SECRET` env variable.
+121 :param http_client: An :class:`HTTPClient` implementation. Defaults to a global :class:`HTTPXClient`.
+122 :param query_tags: Tags to associate with the query. See `logging <https://docs.fauna.com/fauna/current/build/logs/query_log/>`_
+123 :param linearized: If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
+124 :param max_contention_retries: The max number of times to retry the query if contention is encountered.
+125 :param typecheck: Enable or disable typechecking of the query before evaluation. If not set, Fauna will use the value of the "typechecked" flag on the database configuration.
+126 :param additional_headers: Add/update HTTP request headers for the query. In general, this should not be necessary.
+127 :param query_timeout: Controls the maximum amount of time Fauna will execute your query before marking it failed, default is :py:data:`DefaultQueryTimeout`.
+128 :param client_buffer_timeout: Time in milliseconds beyond query_timeout at which the client will abort a request if it has not received a response. The default is :py:data:`DefaultClientBufferTimeout`, which should account for network latency for most clients. The value must be greater than zero. The closer to zero the value is, the more likely the client is to abort the request before the server can report a legitimate response or error.
+129 :param http_read_timeout: Set HTTP Read timeout, default is :py:data:`DefaultHttpReadTimeout`.
+130 :param http_write_timeout: Set HTTP Write timeout, default is :py:data:`DefaultHttpWriteTimeout`.
+131 :param http_connect_timeout: Set HTTP Connect timeout, default is :py:data:`DefaultHttpConnectTimeout`.
+132 :param http_pool_timeout: Set HTTP Pool timeout, default is :py:data:`DefaultHttpPoolTimeout`.
+133 :param http_idle_timeout: Set HTTP Idle timeout, default is :py:data:`DefaultIdleConnectionTimeout`.
+134 :param max_attempts: The maximum number of times to attempt a query when a retryable exception is thrown. Defaults to 3.
+135 :param max_backoff: The maximum backoff in seconds for an individual retry. Defaults to 20.
+136 """
+137
+138self._set_endpoint(endpoint)
+139self._max_attempts=max_attempts
+140self._max_backoff=max_backoff
+141
+142ifsecretisNone:
+143self._auth=_Auth(_Environment.EnvFaunaSecret())
+144else:
+145self._auth=_Auth(secret)
+146
+147self._last_txn_ts=LastTxnTs()
+148
+149self._query_tags={}
+150ifquery_tagsisnotNone:
+151self._query_tags.update(query_tags)
+152
+153ifquery_timeoutisnotNone:
+154self._query_timeout_ms=int(query_timeout.total_seconds()*1000)
+155else:
+156self._query_timeout_ms=None
+157
+158self._headers:Dict[str,str]={
+159_Header.AcceptEncoding:"gzip",
+160_Header.ContentType:"application/json;charset=utf-8",
+161_Header.Driver:"python",
+162_Header.DriverEnv:str(_DriverEnvironment()),
+163}
+164
+165iftypecheckisnotNone:
+166self._headers[Header.Typecheck]=str(typecheck).lower()
+167
+168iflinearizedisnotNone:
+169self._headers[Header.Linearized]=str(linearized).lower()
+170
+171ifmax_contention_retriesisnotNoneandmax_contention_retries>0:
+172self._headers[Header.MaxContentionRetries]= \
+173f"{max_contention_retries}"
+174
+175ifadditional_headersisnotNone:
+176self._headers={
+177**self._headers,
+178**additional_headers,
+179}
+180
+181self._session:HTTPClient
+182
+183ifhttp_clientisnotNone:
+184self._session=http_client
+185else:
+186iffauna.global_http_clientisNone:
+187timeout_s:Optional[float]=None
+188ifquery_timeoutisnotNoneandclient_buffer_timeoutisnotNone:
+189timeout_s=(query_timeout+client_buffer_timeout).total_seconds()
+190read_timeout_s:Optional[float]=None
+191ifhttp_read_timeoutisnotNone:
+192read_timeout_s=http_read_timeout.total_seconds()
+193
+194write_timeout_s:Optional[float]=http_write_timeout.total_seconds(
+195)ifhttp_write_timeoutisnotNoneelseNone
+196connect_timeout_s:Optional[float]=http_connect_timeout.total_seconds(
+197)ifhttp_connect_timeoutisnotNoneelseNone
+198pool_timeout_s:Optional[float]=http_pool_timeout.total_seconds(
+199)ifhttp_pool_timeoutisnotNoneelseNone
+200idle_timeout_s:Optional[float]=http_idle_timeout.total_seconds(
+201)ifhttp_idle_timeoutisnotNoneelseNone
+202
+203importhttpx
+204fromfauna.http.httpx_clientimportHTTPXClient
+205c=HTTPXClient(
+206httpx.Client(
+207http1=True,
+208http2=False,
+209timeout=httpx.Timeout(
+210timeout=timeout_s,
+211connect=connect_timeout_s,
+212read=read_timeout_s,
+213write=write_timeout_s,
+214pool=pool_timeout_s,
+215),
+216limits=httpx.Limits(
+217max_connections=DefaultMaxConnections,
+218max_keepalive_connections=DefaultMaxIdleConnections,
+219keepalive_expiry=idle_timeout_s,
+220),
+221))
+222fauna.global_http_client=c
+223
+224self._session=fauna.global_http_client
+
+
+
+
Initializes a Client.
+
+
Parameters
+
+
+
endpoint: The Fauna Endpoint to use. Defaults to https: //db.fauna.com, or the FAUNA_ENDPOINT env variable.
+
secret: The Fauna Secret to use. Defaults to empty, or the FAUNA_SECRET env variable.
+
http_client: An HTTPClient implementation. Defaults to a global HTTPXClient.
+
**query_tags: Tags to associate with the query. See logging
+
linearized: If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
+
max_contention_retries: The max number of times to retry the query if contention is encountered.
+
typecheck: Enable or disable typechecking of the query before evaluation. If not set, Fauna will use the value of the "typechecked" flag on the database configuration.
+
additional_headers: Add/update HTTP request headers for the query. In general, this should not be necessary.
+
query_timeout: Controls the maximum amount of time Fauna will execute your query before marking it failed, default is DefaultQueryTimeout.
+
client_buffer_timeout: Time in milliseconds beyond query_timeout at which the client will abort a request if it has not received a response. The default is DefaultClientBufferTimeout, which should account for network latency for most clients. The value must be greater than zero. The closer to zero the value is, the more likely the client is to abort the request before the server can report a legitimate response or error.
231defset_last_txn_ts(self,txn_ts:int):
+232"""
+233 Set the last timestamp seen by this client.
+234 This has no effect if earlier than stored timestamp.
+235
+236 .. WARNING:: This should be used only when coordinating timestamps across
+237 multiple clients. Moving the timestamp arbitrarily forward into
+238 the future will cause transactions to stall.
+239
+240 :param txn_ts: the new transaction time.
+241 """
+242self._last_txn_ts.update_txn_time(txn_ts)
+
+
+
+
Set the last timestamp seen by this client.
+This has no effect if earlier than stored timestamp.
+
+
.. WARNING:: This should be used only when coordinating timestamps across
+multiple clients. Moving the timestamp arbitrarily forward into
+the future will cause transactions to stall.
244defget_last_txn_ts(self)->Optional[int]:
+245"""
+246 Get the last timestamp seen by this client.
+247 :return:
+248 """
+249returnself._last_txn_ts.time
+
251defget_query_timeout(self)->Optional[timedelta]:
+252"""
+253 Get the query timeout for all queries.
+254 """
+255ifself._query_timeout_msisnotNone:
+256returntimedelta(milliseconds=self._query_timeout_ms)
+257else:
+258returnNone
+
260defpaginate(
+261self,
+262fql:Query,
+263opts:Optional[QueryOptions]=None,
+264)->"QueryIterator":
+265"""
+266 Run a query on Fauna and returning an iterator of results. If the query
+267 returns a Page, the iterator will fetch additional Pages until the
+268 after token is null. Each call for a page will be retried with exponential
+269 backoff up to the max_attempts set in the client's retry policy in the
+270 event of a 429 or 502.
+271
+272 :param fql: A Query
+273 :param opts: (Optional) Query Options
+274
+275 :return: a :class:`QueryResponse`
+276
+277 :raises NetworkError: HTTP Request failed in transit
+278 :raises ProtocolError: HTTP error not from Fauna
+279 :raises ServiceError: Fauna returned an error
+280 :raises ValueError: Encoding and decoding errors
+281 :raises TypeError: Invalid param types
+282 """
+283
+284ifnotisinstance(fql,Query):
+285err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+286f"Query by calling fauna.fql()"
+287raiseTypeError(err_msg)
+288
+289returnQueryIterator(self,fql,opts)
+
+
+
+
Run a query on Fauna and returning an iterator of results. If the query
+returns a Page, the iterator will fetch additional Pages until the
+after token is null. Each call for a page will be retried with exponential
+backoff up to the max_attempts set in the client's retry policy in the
+event of a 429 or 502.
291defquery(
+292self,
+293fql:Query,
+294opts:Optional[QueryOptions]=None,
+295)->QuerySuccess:
+296"""
+297 Run a query on Fauna. A query will be retried max_attempt times with exponential backoff
+298 up to the max_backoff in the event of a 429.
+299
+300 :param fql: A Query
+301 :param opts: (Optional) Query Options
+302
+303 :return: a :class:`QueryResponse`
+304
+305 :raises NetworkError: HTTP Request failed in transit
+306 :raises ProtocolError: HTTP error not from Fauna
+307 :raises ServiceError: Fauna returned an error
+308 :raises ValueError: Encoding and decoding errors
+309 :raises TypeError: Invalid param types
+310 """
+311
+312ifnotisinstance(fql,Query):
+313err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+314f"Query by calling fauna.fql()"
+315raiseTypeError(err_msg)
+316
+317try:
+318encoded_query:Mapping[str,Any]=FaunaEncoder.encode(fql)
+319exceptExceptionase:
+320raiseClientError("Failed to encode Query")frome
+321
+322retryable=Retryable[QuerySuccess](
+323self._max_attempts,
+324self._max_backoff,
+325self._query,
+326"/query/1",
+327fql=encoded_query,
+328opts=opts,
+329)
+330
+331r=retryable.run()
+332r.response.stats.attempts=r.attempts
+333returnr.response
+
+
+
+
Run a query on Fauna. A query will be retried max_attempt times with exponential backoff
+up to the max_backoff in the event of a 429.
422defstream(
+423self,
+424fql:Union[StreamToken,Query],
+425opts:StreamOptions=StreamOptions()
+426)->"StreamIterator":
+427"""
+428 Opens a Stream in Fauna and returns an iterator that consume Fauna events.
+429
+430 :param fql: A Query that returns a StreamToken or a StreamToken.
+431 :param opts: (Optional) Stream Options.
+432
+433 :return: a :class:`StreamIterator`
+434
+435 :raises ClientError: Invalid options provided
+436 :raises NetworkError: HTTP Request failed in transit
+437 :raises ProtocolError: HTTP error not from Fauna
+438 :raises ServiceError: Fauna returned an error
+439 :raises ValueError: Encoding and decoding errors
+440 :raises TypeError: Invalid param types
+441 """
+442
+443ifisinstance(fql,Query):
+444ifopts.cursorisnotNone:
+445raiseClientError(
+446"The 'cursor' configuration can only be used with a stream token.")
+447
+448token=self.query(fql).data
+449else:
+450token=fql
+451
+452ifnotisinstance(token,StreamToken):
+453err_msg=f"'fql' must be a StreamToken, or a Query that returns a StreamToken but was a {type(token)}."
+454raiseTypeError(err_msg)
+455
+456headers=self._headers.copy()
+457headers[_Header.Format]="tagged"
+458headers[_Header.Authorization]=self._auth.bearer()
+459
+460returnStreamIterator(self._session,headers,self._endpoint+"/stream/1",
+461self._max_attempts,self._max_backoff,opts,token)
+
+
+
+
Opens a Stream in Fauna and returns an iterator that consume Fauna events.
+
+
Parameters
+
+
+
fql: A Query that returns a StreamToken or a StreamToken.
463defchange_feed(
+464self,
+465fql:Union[StreamToken,Query],
+466opts:ChangeFeedOptions=ChangeFeedOptions()
+467)->"ChangeFeedIterator":
+468"""
+469 Opens a change feed in Fauna and returns an iterator that consume Fauna events.
+470
+471 :param fql: A Query that returns a StreamToken or a StreamToken.
+472 :param opts: (Optional) Change feed options.
+473
+474 :return: a :class:`ChangeFeedIterator`
+475
+476 :raises ClientError: Invalid options provided
+477 :raises NetworkError: HTTP Request failed in transit
+478 :raises ProtocolError: HTTP error not from Fauna
+479 :raises ServiceError: Fauna returned an error
+480 :raises ValueError: Encoding and decoding errors
+481 :raises TypeError: Invalid param types
+482 """
+483
+484ifisinstance(fql,Query):
+485token=self.query(fql).data
+486else:
+487token=fql
+488
+489ifnotisinstance(token,StreamToken):
+490err_msg=f"'fql' must be a StreamToken, or a Query that returns a StreamToken but was a {type(token)}."
+491raiseTypeError(err_msg)
+492
+493headers=self._headers.copy()
+494headers[_Header.Format]="tagged"
+495headers[_Header.Authorization]=self._auth.bearer()
+496
+497ifopts.query_timeoutisnotNone:
+498query_timeout_ms=int(opts.query_timeout.total_seconds()*1000)
+499headers[Header.QueryTimeoutMs]=str(query_timeout_ms)
+500elifself._query_timeout_msisnotNone:
+501headers[Header.QueryTimeoutMs]=str(self._query_timeout_ms)
+502
+503returnChangeFeedIterator(self._session,headers,
+504self._endpoint+"/changefeed/1",
+505self._max_attempts,self._max_backoff,opts,
+506token)
+
+
+
+
Opens a change feed in Fauna and returns an iterator that consume Fauna events.
+
+
Parameters
+
+
+
fql: A Query that returns a StreamToken or a StreamToken.
717defflatten(self)->Iterator:
+718"""A generator that yields events instead of pages of events."""
+719forpageinself:
+720foreventinpage:
+721yieldevent
+
+
+
+
A generator that yields events instead of pages of events.
+
+
+
+
+
+
+
+
+
+ class
+ QueryIterator:
+
+
+
+
+
+
724classQueryIterator:
+725"""A class to provider an iterator on top of Fauna queries."""
+726
+727def__init__(self,
+728client:Client,
+729fql:Query,
+730opts:Optional[QueryOptions]=None):
+731"""Initializes the QueryIterator
+732
+733 :param fql: A Query
+734 :param opts: (Optional) Query Options
+735
+736 :raises TypeError: Invalid param types
+737 """
+738ifnotisinstance(client,Client):
+739err_msg=f"'client' must be a Client but was a {type(client)}. You can build a " \
+740f"Client by calling fauna.client.Client()"
+741raiseTypeError(err_msg)
+742
+743ifnotisinstance(fql,Query):
+744err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+745f"Query by calling fauna.fql()"
+746raiseTypeError(err_msg)
+747
+748self.client=client
+749self.fql=fql
+750self.opts=opts
+751
+752def__iter__(self)->Iterator:
+753returnself.iter()
+754
+755defiter(self)->Iterator:
+756"""
+757 A generator function that immediately fetches and yields the results of
+758 the stored query. Yields additional pages on subsequent iterations if
+759 they exist
+760 """
+761
+762cursor=None
+763initial_response=self.client.query(self.fql,self.opts)
+764
+765ifisinstance(initial_response.data,Page):
+766cursor=initial_response.data.after
+767yieldinitial_response.data.data
+768
+769whilecursorisnotNone:
+770next_response=self.client.query(
+771fql("Set.paginate(${after})",after=cursor),self.opts)
+772# TODO: `Set.paginate` does not yet return a `@set` tagged value
+773# so we will get back a plain object that might not have
+774# an after property.
+775cursor=next_response.data.get("after")
+776yieldnext_response.data.get("data")
+777
+778else:
+779yield[initial_response.data]
+780
+781defflatten(self)->Iterator:
+782"""
+783 A generator function that immediately fetches and yields the results of
+784 the stored query. Yields each item individually, rather than a whole
+785 Page at a time. Fetches additional pages as required if they exist.
+786 """
+787
+788forpageinself.iter():
+789foriteminpage:
+790yielditem
+
+
+
+
A class to provider an iterator on top of Fauna queries.
727def__init__(self,
+728client:Client,
+729fql:Query,
+730opts:Optional[QueryOptions]=None):
+731"""Initializes the QueryIterator
+732
+733 :param fql: A Query
+734 :param opts: (Optional) Query Options
+735
+736 :raises TypeError: Invalid param types
+737 """
+738ifnotisinstance(client,Client):
+739err_msg=f"'client' must be a Client but was a {type(client)}. You can build a " \
+740f"Client by calling fauna.client.Client()"
+741raiseTypeError(err_msg)
+742
+743ifnotisinstance(fql,Query):
+744err_msg=f"'fql' must be a Query but was a {type(fql)}. You can build a " \
+745f"Query by calling fauna.fql()"
+746raiseTypeError(err_msg)
+747
+748self.client=client
+749self.fql=fql
+750self.opts=opts
+
+
+
+
Initializes the QueryIterator
+
+
Parameters
+
+
+
fql: A Query
+
opts: (Optional) Query Options
+
+
+
Raises
+
+
+
TypeError: Invalid param types
+
+
+
+
+
+
+
+ client
+
+
+
+
+
+
+
+
+
+
+ fql
+
+
+
+
+
+
+
+
+
+
+ opts
+
+
+
+
+
+
+
+
+
+
+
+
+ def
+ iter(self) -> Iterator:
+
+
+
+
+
+
755defiter(self)->Iterator:
+756"""
+757 A generator function that immediately fetches and yields the results of
+758 the stored query. Yields additional pages on subsequent iterations if
+759 they exist
+760 """
+761
+762cursor=None
+763initial_response=self.client.query(self.fql,self.opts)
+764
+765ifisinstance(initial_response.data,Page):
+766cursor=initial_response.data.after
+767yieldinitial_response.data.data
+768
+769whilecursorisnotNone:
+770next_response=self.client.query(
+771fql("Set.paginate(${after})",after=cursor),self.opts)
+772# TODO: `Set.paginate` does not yet return a `@set` tagged value
+773# so we will get back a plain object that might not have
+774# an after property.
+775cursor=next_response.data.get("after")
+776yieldnext_response.data.get("data")
+777
+778else:
+779yield[initial_response.data]
+
+
+
+
A generator function that immediately fetches and yields the results of
+the stored query. Yields additional pages on subsequent iterations if
+they exist
+
+
+
+
+
+
+
+
+ def
+ flatten(self) -> Iterator:
+
+
+
+
+
+
781defflatten(self)->Iterator:
+782"""
+783 A generator function that immediately fetches and yields the results of
+784 the stored query. Yields each item individually, rather than a whole
+785 Page at a time. Fetches additional pages as required if they exist.
+786 """
+787
+788forpageinself.iter():
+789foriteminpage:
+790yielditem
+
+
+
+
A generator function that immediately fetches and yields the results of
+the stored query. Yields each item individually, rather than a whole
+Page at a time. Fetches additional pages as required if they exist.
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/2.2.0/api/fauna/client/endpoints.html b/2.2.0/api/fauna/client/endpoints.html
new file mode 100644
index 00000000..9012d33e
--- /dev/null
+++ b/2.2.0/api/fauna/client/endpoints.html
@@ -0,0 +1,300 @@
+
+
+
+
+
+
+ fauna.client.endpoints API documentation
+
+
+
+
+
+
+
+
+
+
+
+
1importabc
+ 2fromdataclassesimportdataclass
+ 3fromrandomimportrandom
+ 4fromtimeimportsleep
+ 5fromtypingimportCallable,Optional,TypeVar,Generic
+ 6
+ 7fromfauna.errorsimportRetryableFaunaException
+ 8
+ 9
+10classRetryStrategy:
+11
+12@abc.abstractmethod
+13defwait(self)->float:
+14pass
+15
+16
+17classExponentialBackoffStrategy(RetryStrategy):
+18
+19def__init__(self,max_backoff:int):
+20self._max_backoff=float(max_backoff)
+21self._i=0.0
+22
+23defwait(self)->float:
+24"""Returns the number of seconds to wait for the next call."""
+25backoff=random()*(2.0**self._i)
+26self._i+=1.0
+27returnmin(backoff,self._max_backoff)
+28
+29
+30T=TypeVar('T')
+31
+32
+33@dataclass
+34classRetryableResponse(Generic[T]):
+35attempts:int
+36response:T
+37
+38
+39classRetryable(Generic[T]):
+40"""
+41 Retryable is a wrapper class that acts on a Callable that returns a T type.
+42 """
+43_strategy:RetryStrategy
+44_error:Optional[Exception]
+45
+46def__init__(
+47self,
+48max_attempts:int,
+49max_backoff:int,
+50func:Callable[...,T],
+51*args,
+52**kwargs,
+53):
+54self._max_attempts=max_attempts
+55self._strategy=ExponentialBackoffStrategy(max_backoff)
+56self._func=func
+57self._args=args
+58self._kwargs=kwargs
+59self._error=None
+60
+61defrun(self)->RetryableResponse[T]:
+62"""Runs the wrapped function. Retries up to max_attempts if the function throws a RetryableFaunaException. It propagates
+63 the thrown exception if max_attempts is reached or if a non-retryable is thrown.
+64
+65 Returns the number of attempts and the response
+66 """
+67attempt=0
+68whileTrue:
+69sleep_time=0.0ifattempt==0elseself._strategy.wait()
+70sleep(sleep_time)
+71
+72try:
+73attempt+=1
+74qs=self._func(*self._args,**self._kwargs)
+75returnRetryableResponse[T](attempt,qs)
+76exceptRetryableFaunaExceptionase:
+77ifattempt>=self._max_attempts:
+78raisee
+
+
+ class
+ ExponentialBackoffStrategy(RetryStrategy):
+
+
+
+
+
+
18classExponentialBackoffStrategy(RetryStrategy):
+19
+20def__init__(self,max_backoff:int):
+21self._max_backoff=float(max_backoff)
+22self._i=0.0
+23
+24defwait(self)->float:
+25"""Returns the number of seconds to wait for the next call."""
+26backoff=random()*(2.0**self._i)
+27self._i+=1.0
+28returnmin(backoff,self._max_backoff)
+
24defwait(self)->float:
+25"""Returns the number of seconds to wait for the next call."""
+26backoff=random()*(2.0**self._i)
+27self._i+=1.0
+28returnmin(backoff,self._max_backoff)
+
+
+
+
Returns the number of seconds to wait for the next call.
+
+
+
+
+
+
+
+
+
@dataclass
+
+ class
+ RetryableResponse(typing.Generic[~T]):
+
+
+
+
+
+ class
+ Retryable(typing.Generic[~T]):
+
+
+
+
+
+
40classRetryable(Generic[T]):
+41"""
+42 Retryable is a wrapper class that acts on a Callable that returns a T type.
+43 """
+44_strategy:RetryStrategy
+45_error:Optional[Exception]
+46
+47def__init__(
+48self,
+49max_attempts:int,
+50max_backoff:int,
+51func:Callable[...,T],
+52*args,
+53**kwargs,
+54):
+55self._max_attempts=max_attempts
+56self._strategy=ExponentialBackoffStrategy(max_backoff)
+57self._func=func
+58self._args=args
+59self._kwargs=kwargs
+60self._error=None
+61
+62defrun(self)->RetryableResponse[T]:
+63"""Runs the wrapped function. Retries up to max_attempts if the function throws a RetryableFaunaException. It propagates
+64 the thrown exception if max_attempts is reached or if a non-retryable is thrown.
+65
+66 Returns the number of attempts and the response
+67 """
+68attempt=0
+69whileTrue:
+70sleep_time=0.0ifattempt==0elseself._strategy.wait()
+71sleep(sleep_time)
+72
+73try:
+74attempt+=1
+75qs=self._func(*self._args,**self._kwargs)
+76returnRetryableResponse[T](attempt,qs)
+77exceptRetryableFaunaExceptionase:
+78ifattempt>=self._max_attempts:
+79raisee
+
+
+
+
Retryable is a wrapper class that acts on a Callable that returns a T type.
62defrun(self)->RetryableResponse[T]:
+63"""Runs the wrapped function. Retries up to max_attempts if the function throws a RetryableFaunaException. It propagates
+64 the thrown exception if max_attempts is reached or if a non-retryable is thrown.
+65
+66 Returns the number of attempts and the response
+67 """
+68attempt=0
+69whileTrue:
+70sleep_time=0.0ifattempt==0elseself._strategy.wait()
+71sleep(sleep_time)
+72
+73try:
+74attempt+=1
+75qs=self._func(*self._args,**self._kwargs)
+76returnRetryableResponse[T](attempt,qs)
+77exceptRetryableFaunaExceptionase:
+78ifattempt>=self._max_attempts:
+79raisee
+
+
+
+
Runs the wrapped function. Retries up to max_attempts if the function throws a RetryableFaunaException. It propagates
+the thrown exception if max_attempts is reached or if a non-retryable is thrown.
+
+
Returns the number of attempts and the response
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/2.2.0/api/fauna/client/utils.html b/2.2.0/api/fauna/client/utils.html
new file mode 100644
index 00000000..7c2c921e
--- /dev/null
+++ b/2.2.0/api/fauna/client/utils.html
@@ -0,0 +1,488 @@
+
+
+
+
+
+
+ fauna.client.utils API documentation
+
+
+
+
+
+
+
+
+
+
+
+
1importos
+ 2importthreading
+ 3fromtypingimportGeneric,Callable,TypeVar,Optional
+ 4
+ 5fromfauna.client.endpointsimportEndpoints
+ 6fromfauna.client.headersimportHeader
+ 7
+ 8
+ 9def_fancy_bool_from_str(val:str)->bool:
+10returnval.lower()in["1","true","yes","y"]
+11
+12
+13classLastTxnTs(object):
+14"""Wraps tracking the last transaction time supplied from the database."""
+15
+16def__init__(
+17self,
+18time:Optional[int]=None,
+19):
+20self._lock:threading.Lock=threading.Lock()
+21self._time:Optional[int]=time
+22
+23@property
+24deftime(self):
+25"""Produces the last transaction time, or, None if not yet updated."""
+26withself._lock:
+27returnself._time
+28
+29@property
+30defrequest_header(self):
+31"""Produces a dictionary with a non-zero `X-Last-Seen-Txn` header; or,
+32 if one has not yet been set, the empty header dictionary."""
+33t=self._time
+34iftisNone:
+35return{}
+36return{Header.LastTxnTs:str(t)}
+37
+38defupdate_txn_time(self,new_txn_time:int):
+39"""Updates the internal transaction time.
+40 In order to maintain a monotonically-increasing value, `newTxnTime`
+41 is discarded if it is behind the current timestamp."""
+42withself._lock:
+43self._time=max(self._timeor0,new_txn_time)
+44
+45
+46T=TypeVar('T')
+47
+48
+49class_SettingFromEnviron(Generic[T]):
+50
+51def__init__(
+52self,
+53var_name:str,
+54default_value:str,
+55adapt_from_str:Callable[[str],T],
+56):
+57self.__var_name=var_name
+58self.__default_value=default_value
+59self.__adapt_from_str=adapt_from_str
+60
+61def__call__(self)->T:
+62returnself.__adapt_from_str(
+63os.environ.get(
+64self.__var_name,
+65default=self.__default_value,
+66))
+67
+68
+69class_Environment:
+70EnvFaunaEndpoint=_SettingFromEnviron(
+71"FAUNA_ENDPOINT",
+72Endpoints.Default,
+73str,
+74)
+75"""environment variable for Fauna Client HTTP endpoint"""
+76
+77EnvFaunaSecret=_SettingFromEnviron(
+78"FAUNA_SECRET",
+79"",
+80str,
+81)
+82"""environment variable for Fauna Client authentication"""
+
+
+
+
+
+
+
+
+ class
+ LastTxnTs:
+
+
+
+
+
+
14classLastTxnTs(object):
+15"""Wraps tracking the last transaction time supplied from the database."""
+16
+17def__init__(
+18self,
+19time:Optional[int]=None,
+20):
+21self._lock:threading.Lock=threading.Lock()
+22self._time:Optional[int]=time
+23
+24@property
+25deftime(self):
+26"""Produces the last transaction time, or, None if not yet updated."""
+27withself._lock:
+28returnself._time
+29
+30@property
+31defrequest_header(self):
+32"""Produces a dictionary with a non-zero `X-Last-Seen-Txn` header; or,
+33 if one has not yet been set, the empty header dictionary."""
+34t=self._time
+35iftisNone:
+36return{}
+37return{Header.LastTxnTs:str(t)}
+38
+39defupdate_txn_time(self,new_txn_time:int):
+40"""Updates the internal transaction time.
+41 In order to maintain a monotonically-increasing value, `newTxnTime`
+42 is discarded if it is behind the current timestamp."""
+43withself._lock:
+44self._time=max(self._timeor0,new_txn_time)
+
+
+
+
Wraps tracking the last transaction time supplied from the database.
24@property
+25deftime(self):
+26"""Produces the last transaction time, or, None if not yet updated."""
+27withself._lock:
+28returnself._time
+
+
+
+
Produces the last transaction time, or, None if not yet updated.
+
+
+
+
+
+
+
+ request_header
+
+
+
+
+
+
30@property
+31defrequest_header(self):
+32"""Produces a dictionary with a non-zero `X-Last-Seen-Txn` header; or,
+33 if one has not yet been set, the empty header dictionary."""
+34t=self._time
+35iftisNone:
+36return{}
+37return{Header.LastTxnTs:str(t)}
+
+
+
+
Produces a dictionary with a non-zero X-Last-Seen-Txn header; or,
+if one has not yet been set, the empty header dictionary.
39defupdate_txn_time(self,new_txn_time:int):
+40"""Updates the internal transaction time.
+41 In order to maintain a monotonically-increasing value, `newTxnTime`
+42 is discarded if it is behind the current timestamp."""
+43withself._lock:
+44self._time=max(self._timeor0,new_txn_time)
+
+
+
+
Updates the internal transaction time.
+In order to maintain a monotonically-increasing value, newTxnTime
+is discarded if it is behind the current timestamp.
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/2.2.0/api/fauna/encoding.html b/2.2.0/api/fauna/encoding.html
new file mode 100644
index 00000000..22e7c62a
--- /dev/null
+++ b/2.2.0/api/fauna/encoding.html
@@ -0,0 +1,246 @@
+
+
+
+
+
+
+ fauna.encoding API documentation
+
+
+
+
+
+
+
+
+
+
+
+
54@staticmethod
+55defdecode(obj:Any):
+56"""Decodes supported objects from the tagged typed into untagged.
+57
+58 Examples:
+59 - { "@int": "100" } decodes to 100 of type int
+60 - { "@double": "100" } decodes to 100.0 of type float
+61 - { "@long": "100" } decodes to 100 of type int
+62 - { "@time": "..." } decodes to a datetime
+63 - { "@date": "..." } decodes to a date
+64 - { "@doc": ... } decodes to a Document or NamedDocument
+65 - { "@ref": ... } decodes to a DocumentReference or NamedDocumentReference
+66 - { "@mod": ... } decodes to a Module
+67 - { "@set": ... } decodes to a Page
+68 - { "@stream": ... } decodes to a StreamToken
+69 - { "@bytes": ... } decodes to a bytearray
+70
+71 :param obj: the object to decode
+72 """
+73returnFaunaDecoder._decode(obj)
+
+
+
+
Decodes supported objects from the tagged typed into untagged.
+
+
Examples:
+ - { "@int": "100" } decodes to 100 of type int
+ - { "@double": "100" } decodes to 100.0 of type float
+ - { "@long": "100" } decodes to 100 of type int
+ - { "@time": "..." } decodes to a datetime
+ - { "@date": "..." } decodes to a date
+ - { "@doc": ... } decodes to a Document or NamedDocument
+ - { "@ref": ... } decodes to a DocumentReference or NamedDocumentReference
+ - { "@mod": ... } decodes to a Module
+ - { "@set": ... } decodes to a Page
+ - { "@stream": ... } decodes to a StreamToken
+ - { "@bytes": ... } decodes to a bytearray
+
+
Parameters
+
+
+
obj: the object to decode
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/2.2.0/api/fauna/encoding/encoder.html b/2.2.0/api/fauna/encoding/encoder.html
new file mode 100644
index 00000000..f858912b
--- /dev/null
+++ b/2.2.0/api/fauna/encoding/encoder.html
@@ -0,0 +1,1222 @@
+
+
+
+
+
+
+ fauna.encoding.encoder API documentation
+
+
+
+
+
+
+
+
+
+
+
+
71@staticmethod
+72defencode(obj:Any)->Any:
+73"""Encodes supported objects into the tagged format.
+74
+75 Examples:
+76 - Up to 32-bit ints encode to { "@int": "..." }
+77 - Up to 64-bit ints encode to { "@long": "..." }
+78 - Floats encode to { "@double": "..." }
+79 - datetime encodes to { "@time": "..." }
+80 - date encodes to { "@date": "..." }
+81 - DocumentReference encodes to { "@doc": "..." }
+82 - Module encodes to { "@mod": "..." }
+83 - Query encodes to { "fql": [...] }
+84 - ValueFragment encodes to { "value": <encoded_val> }
+85 - LiteralFragment encodes to a string
+86 - StreamToken encodes to a string
+87
+88 :raises ValueError: If value cannot be encoded, cannot be encoded safely, or there's a circular reference.
+89 :param obj: the object to decode
+90 """
+91returnFaunaEncoder._encode(obj)
+
+
+
+
Encodes supported objects into the tagged format.
+
+
Examples:
+ - Up to 32-bit ints encode to { "@int": "..." }
+ - Up to 64-bit ints encode to { "@long": "..." }
+ - Floats encode to { "@double": "..." }
+ - datetime encodes to { "@time": "..." }
+ - date encodes to { "@date": "..." }
+ - DocumentReference encodes to { "@doc": "..." }
+ - Module encodes to { "@mod": "..." }
+ - Query encodes to { "fql": [...] }
+ - ValueFragment encodes to { "value": }
+ - LiteralFragment encodes to a string
+ - StreamToken encodes to a string
+
+
Raises
+
+
+
ValueError: If value cannot be encoded, cannot be encoded safely, or there's a circular reference.
+
+
+
Parameters
+
+
+
obj: the object to decode
+
+
+
+
+
+
+
+
+
@staticmethod
+
+ def
+ from_int(obj:int):
+
+
+
+
+
+
93@staticmethod
+ 94deffrom_int(obj:int):
+ 95if-2**31<=obj<=2**31-1:
+ 96return{"@int":repr(obj)}
+ 97elif-2**63<=obj<=2**63-1:
+ 98return{"@long":repr(obj)}
+ 99else:
+100raiseValueError("Precision loss when converting int to Fauna type")
+
114@staticmethod
+115deffrom_datetime(obj:datetime):
+116ifobj.utcoffset()isNone:
+117raiseValueError("datetimes must be timezone-aware")
+118
+119return{"@time":obj.isoformat(sep="T")}
+
1fromdataclassesimportdataclass
+ 2fromtypingimportOptional,Mapping,Any,List
+ 3
+ 4
+ 5classQueryStats:
+ 6"""Query stats"""
+ 7
+ 8@property
+ 9defcompute_ops(self)->int:
+ 10"""The amount of Transactional Compute Ops consumed by the query."""
+ 11returnself._compute_ops
+ 12
+ 13@property
+ 14defread_ops(self)->int:
+ 15"""The amount of Transactional Read Ops consumed by the query."""
+ 16returnself._read_ops
+ 17
+ 18@property
+ 19defwrite_ops(self)->int:
+ 20"""The amount of Transactional Write Ops consumed by the query."""
+ 21returnself._write_ops
+ 22
+ 23@property
+ 24defquery_time_ms(self)->int:
+ 25"""The query run time in milliseconds."""
+ 26returnself._query_time_ms
+ 27
+ 28@property
+ 29defstorage_bytes_read(self)->int:
+ 30"""The amount of data read from storage, in bytes."""
+ 31returnself._storage_bytes_read
+ 32
+ 33@property
+ 34defstorage_bytes_write(self)->int:
+ 35"""The amount of data written to storage, in bytes."""
+ 36returnself._storage_bytes_write
+ 37
+ 38@property
+ 39defcontention_retries(self)->int:
+ 40"""The number of times the transaction was retried due to write contention."""
+ 41returnself._contention_retries
+ 42
+ 43@property
+ 44defattempts(self)->int:
+ 45"""The number of attempts made by the client to run the query."""
+ 46returnself._attempts
+ 47
+ 48@attempts.setter
+ 49defattempts(self,value):
+ 50self._attempts=value
+ 51
+ 52def__init__(self,stats:Mapping[str,Any]):
+ 53self._compute_ops=stats.get("compute_ops",0)
+ 54self._read_ops=stats.get("read_ops",0)
+ 55self._write_ops=stats.get("write_ops",0)
+ 56self._query_time_ms=stats.get("query_time_ms",0)
+ 57self._storage_bytes_read=stats.get("storage_bytes_read",0)
+ 58self._storage_bytes_write=stats.get("storage_bytes_write",0)
+ 59self._contention_retries=stats.get("contention_retries",0)
+ 60self._attempts=0
+ 61
+ 62def__repr__(self):
+ 63stats={
+ 64"compute_ops":self._compute_ops,
+ 65"read_ops":self._read_ops,
+ 66"write_ops":self._write_ops,
+ 67"query_time_ms":self._query_time_ms,
+ 68"storage_bytes_read":self._storage_bytes_read,
+ 69"storage_bytes_write":self._storage_bytes_write,
+ 70"contention_retries":self._contention_retries,
+ 71"attempts":self._attempts,
+ 72}
+ 73
+ 74returnf"{self.__class__.__name__}(stats={repr(stats)})"
+ 75
+ 76def__eq__(self,other):
+ 77returntype(self)==type(other) \
+ 78andself.compute_ops==other.compute_ops \
+ 79andself.read_ops==other.read_ops \
+ 80andself.write_ops==other.write_ops \
+ 81andself.query_time_ms==other.query_time_ms \
+ 82andself.storage_bytes_read==other.storage_bytes_read \
+ 83andself.storage_bytes_write==other.storage_bytes_write \
+ 84andself.contention_retries==other.contention_retries \
+ 85andself.attempts==other.attempts
+ 86
+ 87def__ne__(self,other):
+ 88returnnotself.__eq__(other)
+ 89
+ 90
+ 91classQueryInfo:
+ 92
+ 93@property
+ 94defquery_tags(self)->Mapping[str,Any]:
+ 95"""The tags associated with the query."""
+ 96returnself._query_tags
+ 97
+ 98@property
+ 99defsummary(self)->str:
+100"""A comprehensive, human readable summary of any errors, warnings and/or logs returned from the query."""
+101returnself._summary
+102
+103@property
+104defstats(self)->QueryStats:
+105"""Query stats associated with the query."""
+106returnself._stats
+107
+108@property
+109deftxn_ts(self)->int:
+110"""The last transaction timestamp of the query. A Unix epoch in microseconds."""
+111returnself._txn_ts
+112
+113@property
+114defschema_version(self)->int:
+115"""The schema version that was used for the query execution."""
+116returnself._schema_version
+117
+118def__init__(
+119self,
+120query_tags:Optional[Mapping[str,str]]=None,
+121stats:Optional[QueryStats]=None,
+122summary:Optional[str]=None,
+123txn_ts:Optional[int]=None,
+124schema_version:Optional[int]=None,
+125):
+126self._query_tags=query_tagsor{}
+127self._stats=statsorQueryStats({})
+128self._summary=summaryor""
+129self._txn_ts=txn_tsor0
+130self._schema_version=schema_versionor0
+131
+132def__repr__(self):
+133returnf"{self.__class__.__name__}(" \
+134f"query_tags={repr(self.query_tags)}," \
+135f"stats={repr(self.stats)}," \
+136f"summary={repr(self.summary)}," \
+137f"txn_ts={repr(self.txn_ts)}," \
+138f"schema_version={repr(self.schema_version)})"
+139
+140
+141classQuerySuccess(QueryInfo):
+142"""The result of the query."""
+143
+144@property
+145defdata(self)->Any:
+146"""The data returned by the query. This is the result of the FQL query."""
+147returnself._data
+148
+149@property
+150defstatic_type(self)->Optional[str]:
+151"""If typechecked, the query's inferred static result type, if the query was typechecked."""
+152returnself._static_type
+153
+154@property
+155deftraceparent(self)->Optional[str]:
+156"""The traceparent for the query."""
+157returnself._traceparent
+158
+159def__init__(
+160self,
+161data:Any,
+162query_tags:Optional[Mapping[str,str]],
+163static_type:Optional[str],
+164stats:Optional[QueryStats],
+165summary:Optional[str],
+166traceparent:Optional[str],
+167txn_ts:Optional[int],
+168schema_version:Optional[int],
+169):
+170
+171super().__init__(
+172query_tags=query_tags,
+173stats=stats,
+174summary=summary,
+175txn_ts=txn_ts,
+176schema_version=schema_version,
+177)
+178
+179self._traceparent=traceparent
+180self._static_type=static_type
+181self._data=data
+182
+183def__repr__(self):
+184returnf"{self.__class__.__name__}(" \
+185f"query_tags={repr(self.query_tags)}," \
+186f"static_type={repr(self.static_type)}," \
+187f"stats={repr(self.stats)}," \
+188f"summary={repr(self.summary)}," \
+189f"traceparent={repr(self.traceparent)}," \
+190f"txn_ts={repr(self.txn_ts)}," \
+191f"schema_version={repr(self.schema_version)}," \
+192f"data={repr(self.data)})"
+193
+194
+195@dataclass
+196classConstraintFailure:
+197message:str
+198name:Optional[str]=None
+199paths:Optional[List[Any]]=None
+200
+201
+202classQueryTags:
+203
+204@staticmethod
+205defencode(tags:Mapping[str,str])->str:
+206return",".join([f"{k}={v}"fork,vintags.items()])
+207
+208@staticmethod
+209defdecode(tag_str:str)->Mapping[str,str]:
+210res:dict[str,str]={}
+211forpairintag_str.split(","):
+212kv=pair.split("=")
+213res[kv[0]]=kv[1]
+214returnres
+
+
+
+
+
+
+
+
+ class
+ QueryStats:
+
+
+
+
+
+
6classQueryStats:
+ 7"""Query stats"""
+ 8
+ 9@property
+10defcompute_ops(self)->int:
+11"""The amount of Transactional Compute Ops consumed by the query."""
+12returnself._compute_ops
+13
+14@property
+15defread_ops(self)->int:
+16"""The amount of Transactional Read Ops consumed by the query."""
+17returnself._read_ops
+18
+19@property
+20defwrite_ops(self)->int:
+21"""The amount of Transactional Write Ops consumed by the query."""
+22returnself._write_ops
+23
+24@property
+25defquery_time_ms(self)->int:
+26"""The query run time in milliseconds."""
+27returnself._query_time_ms
+28
+29@property
+30defstorage_bytes_read(self)->int:
+31"""The amount of data read from storage, in bytes."""
+32returnself._storage_bytes_read
+33
+34@property
+35defstorage_bytes_write(self)->int:
+36"""The amount of data written to storage, in bytes."""
+37returnself._storage_bytes_write
+38
+39@property
+40defcontention_retries(self)->int:
+41"""The number of times the transaction was retried due to write contention."""
+42returnself._contention_retries
+43
+44@property
+45defattempts(self)->int:
+46"""The number of attempts made by the client to run the query."""
+47returnself._attempts
+48
+49@attempts.setter
+50defattempts(self,value):
+51self._attempts=value
+52
+53def__init__(self,stats:Mapping[str,Any]):
+54self._compute_ops=stats.get("compute_ops",0)
+55self._read_ops=stats.get("read_ops",0)
+56self._write_ops=stats.get("write_ops",0)
+57self._query_time_ms=stats.get("query_time_ms",0)
+58self._storage_bytes_read=stats.get("storage_bytes_read",0)
+59self._storage_bytes_write=stats.get("storage_bytes_write",0)
+60self._contention_retries=stats.get("contention_retries",0)
+61self._attempts=0
+62
+63def__repr__(self):
+64stats={
+65"compute_ops":self._compute_ops,
+66"read_ops":self._read_ops,
+67"write_ops":self._write_ops,
+68"query_time_ms":self._query_time_ms,
+69"storage_bytes_read":self._storage_bytes_read,
+70"storage_bytes_write":self._storage_bytes_write,
+71"contention_retries":self._contention_retries,
+72"attempts":self._attempts,
+73}
+74
+75returnf"{self.__class__.__name__}(stats={repr(stats)})"
+76
+77def__eq__(self,other):
+78returntype(self)==type(other) \
+79andself.compute_ops==other.compute_ops \
+80andself.read_ops==other.read_ops \
+81andself.write_ops==other.write_ops \
+82andself.query_time_ms==other.query_time_ms \
+83andself.storage_bytes_read==other.storage_bytes_read \
+84andself.storage_bytes_write==other.storage_bytes_write \
+85andself.contention_retries==other.contention_retries \
+86andself.attempts==other.attempts
+87
+88def__ne__(self,other):
+89returnnotself.__eq__(other)
+
9@property
+10defcompute_ops(self)->int:
+11"""The amount of Transactional Compute Ops consumed by the query."""
+12returnself._compute_ops
+
+
+
+
The amount of Transactional Compute Ops consumed by the query.
+
+
+
+
+
+
+
+ read_ops: int
+
+
+
+
+
+
14@property
+15defread_ops(self)->int:
+16"""The amount of Transactional Read Ops consumed by the query."""
+17returnself._read_ops
+
+
+
+
The amount of Transactional Read Ops consumed by the query.
+
+
+
+
+
+
+
+ write_ops: int
+
+
+
+
+
+
19@property
+20defwrite_ops(self)->int:
+21"""The amount of Transactional Write Ops consumed by the query."""
+22returnself._write_ops
+
+
+
+
The amount of Transactional Write Ops consumed by the query.
+
+
+
+
+
+
+
+ query_time_ms: int
+
+
+
+
+
+
24@property
+25defquery_time_ms(self)->int:
+26"""The query run time in milliseconds."""
+27returnself._query_time_ms
+
+
+
+
The query run time in milliseconds.
+
+
+
+
+
+
+
+ storage_bytes_read: int
+
+
+
+
+
+
29@property
+30defstorage_bytes_read(self)->int:
+31"""The amount of data read from storage, in bytes."""
+32returnself._storage_bytes_read
+
+
+
+
The amount of data read from storage, in bytes.
+
+
+
+
+
+
+
+ storage_bytes_write: int
+
+
+
+
+
+
34@property
+35defstorage_bytes_write(self)->int:
+36"""The amount of data written to storage, in bytes."""
+37returnself._storage_bytes_write
+
+
+
+
The amount of data written to storage, in bytes.
+
+
+
+
+
+
+
+ contention_retries: int
+
+
+
+
+
+
39@property
+40defcontention_retries(self)->int:
+41"""The number of times the transaction was retried due to write contention."""
+42returnself._contention_retries
+
+
+
+
The number of times the transaction was retried due to write contention.
+
+
+
+
+
+
+
+ attempts: int
+
+
+
+
+
+
44@property
+45defattempts(self)->int:
+46"""The number of attempts made by the client to run the query."""
+47returnself._attempts
+
+
+
+
The number of attempts made by the client to run the query.
+
+
+
+
+
+
+
+
+
+ class
+ QueryInfo:
+
+
+
+
+
+
92classQueryInfo:
+ 93
+ 94@property
+ 95defquery_tags(self)->Mapping[str,Any]:
+ 96"""The tags associated with the query."""
+ 97returnself._query_tags
+ 98
+ 99@property
+100defsummary(self)->str:
+101"""A comprehensive, human readable summary of any errors, warnings and/or logs returned from the query."""
+102returnself._summary
+103
+104@property
+105defstats(self)->QueryStats:
+106"""Query stats associated with the query."""
+107returnself._stats
+108
+109@property
+110deftxn_ts(self)->int:
+111"""The last transaction timestamp of the query. A Unix epoch in microseconds."""
+112returnself._txn_ts
+113
+114@property
+115defschema_version(self)->int:
+116"""The schema version that was used for the query execution."""
+117returnself._schema_version
+118
+119def__init__(
+120self,
+121query_tags:Optional[Mapping[str,str]]=None,
+122stats:Optional[QueryStats]=None,
+123summary:Optional[str]=None,
+124txn_ts:Optional[int]=None,
+125schema_version:Optional[int]=None,
+126):
+127self._query_tags=query_tagsor{}
+128self._stats=statsorQueryStats({})
+129self._summary=summaryor""
+130self._txn_ts=txn_tsor0
+131self._schema_version=schema_versionor0
+132
+133def__repr__(self):
+134returnf"{self.__class__.__name__}(" \
+135f"query_tags={repr(self.query_tags)}," \
+136f"stats={repr(self.stats)}," \
+137f"summary={repr(self.summary)}," \
+138f"txn_ts={repr(self.txn_ts)}," \
+139f"schema_version={repr(self.schema_version)})"
+
94@property
+95defquery_tags(self)->Mapping[str,Any]:
+96"""The tags associated with the query."""
+97returnself._query_tags
+
+
+
+
The tags associated with the query.
+
+
+
+
+
+
+
+ summary: str
+
+
+
+
+
+
99@property
+100defsummary(self)->str:
+101"""A comprehensive, human readable summary of any errors, warnings and/or logs returned from the query."""
+102returnself._summary
+
+
+
+
A comprehensive, human readable summary of any errors, warnings and/or logs returned from the query.
142classQuerySuccess(QueryInfo):
+143"""The result of the query."""
+144
+145@property
+146defdata(self)->Any:
+147"""The data returned by the query. This is the result of the FQL query."""
+148returnself._data
+149
+150@property
+151defstatic_type(self)->Optional[str]:
+152"""If typechecked, the query's inferred static result type, if the query was typechecked."""
+153returnself._static_type
+154
+155@property
+156deftraceparent(self)->Optional[str]:
+157"""The traceparent for the query."""
+158returnself._traceparent
+159
+160def__init__(
+161self,
+162data:Any,
+163query_tags:Optional[Mapping[str,str]],
+164static_type:Optional[str],
+165stats:Optional[QueryStats],
+166summary:Optional[str],
+167traceparent:Optional[str],
+168txn_ts:Optional[int],
+169schema_version:Optional[int],
+170):
+171
+172super().__init__(
+173query_tags=query_tags,
+174stats=stats,
+175summary=summary,
+176txn_ts=txn_ts,
+177schema_version=schema_version,
+178)
+179
+180self._traceparent=traceparent
+181self._static_type=static_type
+182self._data=data
+183
+184def__repr__(self):
+185returnf"{self.__class__.__name__}(" \
+186f"query_tags={repr(self.query_tags)}," \
+187f"static_type={repr(self.static_type)}," \
+188f"stats={repr(self.stats)}," \
+189f"summary={repr(self.summary)}," \
+190f"traceparent={repr(self.traceparent)}," \
+191f"txn_ts={repr(self.txn_ts)}," \
+192f"schema_version={repr(self.schema_version)}," \
+193f"data={repr(self.data)})"
+
145@property
+146defdata(self)->Any:
+147"""The data returned by the query. This is the result of the FQL query."""
+148returnself._data
+
+
+
+
The data returned by the query. This is the result of the FQL query.
+
+
+
+
+
+
+
+ static_type: Optional[str]
+
+
+
+
+
+
150@property
+151defstatic_type(self)->Optional[str]:
+152"""If typechecked, the query's inferred static result type, if the query was typechecked."""
+153returnself._static_type
+
+
+
+
If typechecked, the query's inferred static result type, if the query was typechecked.
+
+
+
+
+
+
+
+ traceparent: Optional[str]
+
+
+
+
+
+
155@property
+156deftraceparent(self)->Optional[str]:
+157"""The traceparent for the query."""
+158returnself._traceparent
+
16classClientError(FaunaException):
+17"""An error representing a failure internal to the client, itself.
+18 This indicates Fauna was never called - the client failed internally
+19 prior to sending the request."""
+20pass
+
+
+
+
An error representing a failure internal to the client, itself.
+This indicates Fauna was never called - the client failed internally
+prior to sending the request.
23classNetworkError(FaunaException):
+24"""An error representing a failure due to the network.
+25 This indicates Fauna was never reached."""
+26pass
+
+
+
+
An error representing a failure due to the network.
+This indicates Fauna was never reached.
+
+ class
+ QueryRuntimeError(ServiceError):
+
+
+
+
+
+
344classQueryRuntimeError(ServiceError):
+345"""An error response that is the result of the query failing during execution.
+346 QueryRuntimeError's occur when a bug in your query causes an invalid execution
+347 to be requested.
+348 The 'code' field will vary based on the specific error cause."""
+349pass
+
+
+
+
An error response that is the result of the query failing during execution.
+QueryRuntimeError's occur when a bug in your query causes an invalid execution
+to be requested.
+The 'code' field will vary based on the specific error cause.
+
+ class
+ AuthorizationError(ServiceError):
+
+
+
+
+
+
357classAuthorizationError(ServiceError):
+358"""AuthorizationError indicates the credentials used do not have
+359 permission to perform the requested action."""
+360pass
+
+
+
+
AuthorizationError indicates the credentials used do not have
+permission to perform the requested action.
363classThrottlingError(ServiceError,RetryableFaunaException):
+364"""ThrottlingError indicates some capacity limit was exceeded
+365 and thus the request could not be served."""
+366pass
+
+
+
+
ThrottlingError indicates some capacity limit was exceeded
+and thus the request could not be served.
+
+ class
+ QueryTimeoutError(ServiceError):
+
+
+
+
+
+
369classQueryTimeoutError(ServiceError):
+370"""A failure due to the timeout being exceeded, but the timeout
+371 was set lower than the query's expected processing time.
+372 This response is distinguished from a ServiceTimeoutException
+373 in that a QueryTimeoutError shows Fauna behaving in an expected manner."""
+374pass
+
+
+
+
A failure due to the timeout being exceeded, but the timeout
+was set lower than the query's expected processing time.
+This response is distinguished from a ServiceTimeoutException
+in that a QueryTimeoutError shows Fauna behaving in an expected manner.
+
+ class
+ ServiceTimeoutError(ServiceError):
+
+
+
+
+
+
382classServiceTimeoutError(ServiceError):
+383"""ServiceTimeoutError indicates Fauna was not available to service
+384 the request before the timeout was reached."""
+385pass
+
+
+
+
ServiceTimeoutError indicates Fauna was not available to service
+the request before the timeout was reached.
24defjson(self)->Any:
+25try:
+26decoded=self._r.read().decode("utf-8")
+27returnjson.loads(decoded)
+28except(JSONDecodeError,UnicodeDecodeError)ase:
+29raiseClientError(
+30f"Unable to decode response from endpoint {self._r.request.url}. Check that your endpoint is valid."
+31)frome
+
76classBaseReference:
+77_collection:Module
+78
+79@property
+80defcoll(self)->Module:
+81returnself._collection
+82
+83def__init__(self,coll:Union[str,Module]):
+84ifisinstance(coll,Module):
+85self._collection=coll
+86elifisinstance(coll,str):
+87self._collection=Module(coll)
+88else:
+89raiseTypeError(
+90f"'coll' should be of type Module or str, but was {type(coll)}")
+91
+92def__repr__(self):
+93returnf"{self.__class__.__name__}(coll={repr(self._collection)})"
+94
+95def__eq__(self,other):
+96returnisinstance(other,type(self))andstr(self)==str(other)
+
83def__init__(self,coll:Union[str,Module]):
+84ifisinstance(coll,Module):
+85self._collection=coll
+86elifisinstance(coll,str):
+87self._collection=Module(coll)
+88else:
+89raiseTypeError(
+90f"'coll' should be of type Module or str, but was {type(coll)}")
+
+
+ class
+ DocumentReference(BaseReference):
+
+
+
+
+
+
99classDocumentReference(BaseReference):
+100"""A class representing a reference to a :class:`Document` stored in Fauna.
+101 """
+102
+103@property
+104defid(self)->str:
+105"""The ID for the :class:`Document`. Valid IDs are 64-bit integers, stored as strings.
+106
+107 :rtype: str
+108 """
+109returnself._id
+110
+111def__init__(self,coll:Union[str,Module],id:str):
+112super().__init__(coll)
+113
+114ifnotisinstance(id,str):
+115raiseTypeError(f"'id' should be of type str, but was {type(id)}")
+116self._id=id
+117
+118def__hash__(self):
+119returnhash((type(self),self._collection,self._id))
+120
+121def__repr__(self):
+122returnf"{self.__class__.__name__}(id={repr(self._id)},coll={repr(self._collection)})"
+123
+124@staticmethod
+125deffrom_string(ref:str):
+126rs=ref.split(":")
+127iflen(rs)!=2:
+128raiseValueError("Expects string of format <CollectionName>:<ID>")
+129returnDocumentReference(rs[0],rs[1])
+
+
+
+
A class representing a reference to a Document stored in Fauna.
111def__init__(self,coll:Union[str,Module],id:str):
+112super().__init__(coll)
+113
+114ifnotisinstance(id,str):
+115raiseTypeError(f"'id' should be of type str, but was {type(id)}")
+116self._id=id
+
+
+
+
+
+
+
+
+
+ id: str
+
+
+
+
+
+
103@property
+104defid(self)->str:
+105"""The ID for the :class:`Document`. Valid IDs are 64-bit integers, stored as strings.
+106
+107 :rtype: str
+108 """
+109returnself._id
+
+
+
+
The ID for the Document. Valid IDs are 64-bit integers, stored as strings.
+
+
+
+
+
+
+
+
@staticmethod
+
+ def
+ from_string(ref:str):
+
+
+
+
+
+
124@staticmethod
+125deffrom_string(ref:str):
+126rs=ref.split(":")
+127iflen(rs)!=2:
+128raiseValueError("Expects string of format <CollectionName>:<ID>")
+129returnDocumentReference(rs[0],rs[1])
+
+
+ class
+ NamedDocumentReference(BaseReference):
+
+
+
+
+
+
132classNamedDocumentReference(BaseReference):
+133"""A class representing a reference to a :class:`NamedDocument` stored in Fauna.
+134 """
+135
+136@property
+137defname(self)->str:
+138"""The name of the :class:`NamedDocument`.
+139
+140 :rtype: str
+141 """
+142returnself._name
+143
+144def__init__(self,coll:Union[str,Module],name:str):
+145super().__init__(coll)
+146
+147ifnotisinstance(name,str):
+148raiseTypeError(f"'name' should be of type str, but was {type(name)}")
+149
+150self._name=name
+151
+152def__hash__(self):
+153returnhash((type(self),self._collection,self._name))
+154
+155def__repr__(self):
+156returnf"{self.__class__.__name__}(name={repr(self._name)},coll={repr(self._collection)})"
+
+
+
+
A class representing a reference to a NamedDocument stored in Fauna.
144def__init__(self,coll:Union[str,Module],name:str):
+145super().__init__(coll)
+146
+147ifnotisinstance(name,str):
+148raiseTypeError(f"'name' should be of type str, but was {type(name)}")
+149
+150self._name=name
+
+
+
+
+
+
+
+
+
+ name: str
+
+
+
+
+
+
136@property
+137defname(self)->str:
+138"""The name of the :class:`NamedDocument`.
+139
+140 :rtype: str
+141 """
+142returnself._name
+
225classDocument(BaseDocument):
+226"""A class representing a user document stored in Fauna.
+227
+228 User data should be stored directly on the map, while id, ts, and coll should only be stored on the related
+229 properties. When working with a :class:`Document` in code, it should be considered immutable.
+230 """
+231
+232@property
+233defid(self)->str:
+234returnself._id
+235
+236@property
+237defts(self)->datetime:
+238returnself._ts
+239
+240@property
+241defcoll(self)->Module:
+242returnself._coll
+243
+244def__init__(self,
+245id:str,
+246ts:datetime,
+247coll:Union[str,Module],
+248data:Optional[Mapping]=None):
+249ifnotisinstance(id,str):
+250raiseTypeError(f"'id' should be of type str, but was {type(id)}")
+251
+252ifnotisinstance(ts,datetime):
+253raiseTypeError(f"'ts' should be of type datetime, but was {type(ts)}")
+254
+255ifnot(isinstance(coll,str)orisinstance(coll,Module)):
+256raiseTypeError(
+257f"'coll' should be of type Module or str, but was {type(coll)}")
+258
+259ifisinstance(coll,str):
+260coll=Module(coll)
+261
+262self._id=id
+263self._ts=ts
+264self._coll=coll
+265
+266super().__init__(dataor{})
+267
+268def__eq__(self,other):
+269returntype(self)==type(other) \
+270andself.id==other.id \
+271andself.coll==other.coll \
+272andself.ts==other.ts \
+273andsuper().__eq__(other)
+274
+275def__ne__(self,other):
+276returnnotself.__eq__(other)
+277
+278def__repr__(self):
+279kvs=",".join([f"{repr(k)}:{repr(v)}"fork,vinself.items()])
+280
+281returnf"{self.__class__.__name__}(" \
+282f"id={repr(self.id)}," \
+283f"coll={repr(self.coll)}," \
+284f"ts={repr(self.ts)}," \
+285f"data={{{kvs}}})"
+
+
+
+
A class representing a user document stored in Fauna.
+
+
User data should be stored directly on the map, while id, ts, and coll should only be stored on the related
+properties. When working with a Document in code, it should be considered immutable.
244def__init__(self,
+245id:str,
+246ts:datetime,
+247coll:Union[str,Module],
+248data:Optional[Mapping]=None):
+249ifnotisinstance(id,str):
+250raiseTypeError(f"'id' should be of type str, but was {type(id)}")
+251
+252ifnotisinstance(ts,datetime):
+253raiseTypeError(f"'ts' should be of type datetime, but was {type(ts)}")
+254
+255ifnot(isinstance(coll,str)orisinstance(coll,Module)):
+256raiseTypeError(
+257f"'coll' should be of type Module or str, but was {type(coll)}")
+258
+259ifisinstance(coll,str):
+260coll=Module(coll)
+261
+262self._id=id
+263self._ts=ts
+264self._coll=coll
+265
+266super().__init__(dataor{})
+
288classNamedDocument(BaseDocument):
+289"""A class representing a named document stored in Fauna. Examples of named documents include Collection
+290 definitions, Index definitions, and Roles, among others.
+291
+292 When working with a :class:`NamedDocument` in code, it should be considered immutable.
+293 """
+294
+295@property
+296defname(self)->str:
+297returnself._name
+298
+299@property
+300defts(self)->datetime:
+301returnself._ts
+302
+303@property
+304defcoll(self)->Module:
+305returnself._coll
+306
+307def__init__(self,
+308name:str,
+309ts:datetime,
+310coll:Union[Module,str],
+311data:Optional[Mapping]=None):
+312ifnotisinstance(name,str):
+313raiseTypeError(f"'name' should be of type str, but was {type(name)}")
+314
+315ifnotisinstance(ts,datetime):
+316raiseTypeError(f"'ts' should be of type datetime, but was {type(ts)}")
+317
+318ifnot(isinstance(coll,str)orisinstance(coll,Module)):
+319raiseTypeError(
+320f"'coll' should be of type Module or str, but was {type(coll)}")
+321
+322ifisinstance(coll,str):
+323coll=Module(coll)
+324
+325self._name=name
+326self._ts=ts
+327self._coll=coll
+328
+329super().__init__(dataor{})
+330
+331def__eq__(self,other):
+332returntype(self)==type(other) \
+333andself.name==other.name \
+334andself.coll==other.coll \
+335andself.ts==other.ts \
+336andsuper().__eq__(other)
+337
+338def__ne__(self,other):
+339returnnotself.__eq__(other)
+340
+341def__repr__(self):
+342kvs=",".join([f"{repr(k)}:{repr(v)}"fork,vinself.items()])
+343
+344returnf"{self.__class__.__name__}(" \
+345f"name={repr(self.name)}," \
+346f"coll={repr(self.coll)}," \
+347f"ts={repr(self.ts)}," \
+348f"data={{{kvs}}})"
+
+
+
+
A class representing a named document stored in Fauna. Examples of named documents include Collection
+definitions, Index definitions, and Roles, among others.
+
+
When working with a NamedDocument in code, it should be considered immutable.
307def__init__(self,
+308name:str,
+309ts:datetime,
+310coll:Union[Module,str],
+311data:Optional[Mapping]=None):
+312ifnotisinstance(name,str):
+313raiseTypeError(f"'name' should be of type str, but was {type(name)}")
+314
+315ifnotisinstance(ts,datetime):
+316raiseTypeError(f"'ts' should be of type datetime, but was {type(ts)}")
+317
+318ifnot(isinstance(coll,str)orisinstance(coll,Module)):
+319raiseTypeError(
+320f"'coll' should be of type Module or str, but was {type(coll)}")
+321
+322ifisinstance(coll,str):
+323coll=Module(coll)
+324
+325self._name=name
+326self._ts=ts
+327self._coll=coll
+328
+329super().__init__(dataor{})
+
1importabc
+ 2fromtypingimportAny,Optional,List
+ 3
+ 4from.templateimportFaunaTemplate
+ 5
+ 6
+ 7classFragment(abc.ABC):
+ 8"""An abstract class representing a Fragment of a query.
+ 9 """
+ 10
+ 11@abc.abstractmethod
+ 12defget(self)->Any:
+ 13"""An abstract method for returning a stored value.
+ 14 """
+ 15pass
+ 16
+ 17
+ 18classValueFragment(Fragment):
+ 19"""A concrete :class:`Fragment` representing a part of a query that can represent a template variable.
+ 20 For example, if a template contains a variable ``${foo}``, and an object ``{ "prop": 1 }`` is provided for foo,
+ 21 then ``{ "prop": 1 }`` should be wrapped as a :class:`ValueFragment`.
+ 22
+ 23 :param Any val: The value to be used as a fragment.
+ 24 """
+ 25
+ 26def__init__(self,val:Any):
+ 27self._val=val
+ 28
+ 29defget(self)->Any:
+ 30"""Gets the stored value.
+ 31
+ 32 :returns: The stored value.
+ 33 """
+ 34returnself._val
+ 35
+ 36
+ 37classLiteralFragment(Fragment):
+ 38"""A concrete :class:`Fragment` representing a query literal For example, in the template ```let x = ${foo}```,
+ 39 the portion ```let x = ``` is a query literal and should be wrapped as a :class:`LiteralFragment`.
+ 40
+ 41 :param str val: The query literal to be used as a fragment.
+ 42 """
+ 43
+ 44def__init__(self,val:str):
+ 45self._val=val
+ 46
+ 47defget(self)->str:
+ 48"""Returns the stored value.
+ 49
+ 50 :returns: The stored value.
+ 51 """
+ 52returnself._val
+ 53
+ 54
+ 55classQuery:
+ 56"""A class for representing a query.
+ 57
+ 58 e.g. { "fql": [...] }
+ 59 """
+ 60_fragments:List[Fragment]
+ 61
+ 62def__init__(self,fragments:Optional[List[Fragment]]=None):
+ 63self._fragments=fragmentsor[]
+ 64
+ 65@property
+ 66deffragments(self)->List[Fragment]:
+ 67"""The list of stored Fragments"""
+ 68returnself._fragments
+ 69
+ 70def__str__(self)->str:
+ 71res=""
+ 72forfinself._fragments:
+ 73res+=str(f.get())
+ 74
+ 75returnres
+ 76
+ 77
+ 78deffql(query:str,**kwargs:Any)->Query:
+ 79"""Creates a Query - capable of performing query composition and simple querying. It can accept a
+ 80 simple string query, or can perform composition using ``${}`` sigil string template with ``**kwargs`` as
+ 81 substitutions.
+ 82
+ 83 The ``**kwargs`` can be Fauna data types - such as strings, document references, or modules - and embedded
+ 84 Query - allowing you to compose arbitrarily complex queries.
+ 85
+ 86 When providing ``**kwargs``, following types are accepted:
+ 87 - :class:`str`, :class:`int`, :class:`float`, :class:`bool`, :class:`datetime.datetime`, :class:`datetime.date`,
+ 88 :class:`dict`, :class:`list`, :class:`Query`, :class:`DocumentReference`, :class:`Module`
+ 89
+ 90 :raises ValueError: If there is an invalid template placeholder or a value that cannot be encoded.
+ 91 :returns: A :class:`Query` that can be passed to the client for evaluation against Fauna.
+ 92
+ 93 Examples:
+ 94
+ 95 .. code-block:: python
+ 96 :name: Simple-FQL-Example
+ 97 :caption: Simple query declaration using this function.
+ 98
+ 99 fql('Dogs.byName("Fido")')
+100
+101 .. code-block:: python
+102 :name: Composition-FQL-Example
+103 :caption: Query composition using this function.
+104
+105 def get_dog(id):
+106 return fql('Dogs.byId(${id})', id=id)
+107
+108 def get_vet_phone(id):
+109 return fql('${dog} { .vet_phone_number }', dog=get_dog(id))
+110
+111 get_vet_phone('d123')
+112
+113 """
+114
+115fragments:List[Any]=[]
+116template=FaunaTemplate(query)
+117fortext,field_nameintemplate.iter():
+118iftextisnotNoneandlen(text)>0:
+119fragments.append(LiteralFragment(text))
+120
+121iffield_nameisnotNone:
+122iffield_namenotinkwargs:
+123raiseValueError(
+124f"template variable `{field_name}` not found in provided kwargs")
+125
+126# TODO: Reject if it's already a fragment, or accept *Fragment? Decide on API here
+127fragments.append(ValueFragment(kwargs[field_name]))
+128returnQuery(fragments)
+
+
+
+
+
+
+
+
+ class
+ Fragment(abc.ABC):
+
+
+
+
+
+
8classFragment(abc.ABC):
+ 9"""An abstract class representing a Fragment of a query.
+10 """
+11
+12@abc.abstractmethod
+13defget(self)->Any:
+14"""An abstract method for returning a stored value.
+15 """
+16pass
+
+
+
+
An abstract class representing a Fragment of a query.
+
+
+
+
+
+
+
@abc.abstractmethod
+
+ def
+ get(self) -> Any:
+
+
+
+
+
+
12@abc.abstractmethod
+13defget(self)->Any:
+14"""An abstract method for returning a stored value.
+15 """
+16pass
+
19classValueFragment(Fragment):
+20"""A concrete :class:`Fragment` representing a part of a query that can represent a template variable.
+21 For example, if a template contains a variable ``${foo}``, and an object ``{ "prop": 1 }`` is provided for foo,
+22 then ``{ "prop": 1 }`` should be wrapped as a :class:`ValueFragment`.
+23
+24 :param Any val: The value to be used as a fragment.
+25 """
+26
+27def__init__(self,val:Any):
+28self._val=val
+29
+30defget(self)->Any:
+31"""Gets the stored value.
+32
+33 :returns: The stored value.
+34 """
+35returnself._val
+
+
+
+
A concrete Fragment representing a part of a query that can represent a template variable.
+For example, if a template contains a variable ${foo}, and an object { "prop": 1 } is provided for foo,
+then { "prop": 1 } should be wrapped as a ValueFragment.
38classLiteralFragment(Fragment):
+39"""A concrete :class:`Fragment` representing a query literal For example, in the template ```let x = ${foo}```,
+40 the portion ```let x = ``` is a query literal and should be wrapped as a :class:`LiteralFragment`.
+41
+42 :param str val: The query literal to be used as a fragment.
+43 """
+44
+45def__init__(self,val:str):
+46self._val=val
+47
+48defget(self)->str:
+49"""Returns the stored value.
+50
+51 :returns: The stored value.
+52 """
+53returnself._val
+
+
+
+
A concrete Fragment representing a query literal For example, in the template let x = ${foo},
+the portion let x = is a query literal and should be wrapped as a LiteralFragment.
+
+
Parameters
+
+
+
str val: The query literal to be used as a fragment.
79deffql(query:str,**kwargs:Any)->Query:
+ 80"""Creates a Query - capable of performing query composition and simple querying. It can accept a
+ 81 simple string query, or can perform composition using ``${}`` sigil string template with ``**kwargs`` as
+ 82 substitutions.
+ 83
+ 84 The ``**kwargs`` can be Fauna data types - such as strings, document references, or modules - and embedded
+ 85 Query - allowing you to compose arbitrarily complex queries.
+ 86
+ 87 When providing ``**kwargs``, following types are accepted:
+ 88 - :class:`str`, :class:`int`, :class:`float`, :class:`bool`, :class:`datetime.datetime`, :class:`datetime.date`,
+ 89 :class:`dict`, :class:`list`, :class:`Query`, :class:`DocumentReference`, :class:`Module`
+ 90
+ 91 :raises ValueError: If there is an invalid template placeholder or a value that cannot be encoded.
+ 92 :returns: A :class:`Query` that can be passed to the client for evaluation against Fauna.
+ 93
+ 94 Examples:
+ 95
+ 96 .. code-block:: python
+ 97 :name: Simple-FQL-Example
+ 98 :caption: Simple query declaration using this function.
+ 99
+100 fql('Dogs.byName("Fido")')
+101
+102 .. code-block:: python
+103 :name: Composition-FQL-Example
+104 :caption: Query composition using this function.
+105
+106 def get_dog(id):
+107 return fql('Dogs.byId(${id})', id=id)
+108
+109 def get_vet_phone(id):
+110 return fql('${dog} { .vet_phone_number }', dog=get_dog(id))
+111
+112 get_vet_phone('d123')
+113
+114 """
+115
+116fragments:List[Any]=[]
+117template=FaunaTemplate(query)
+118fortext,field_nameintemplate.iter():
+119iftextisnotNoneandlen(text)>0:
+120fragments.append(LiteralFragment(text))
+121
+122iffield_nameisnotNone:
+123iffield_namenotinkwargs:
+124raiseValueError(
+125f"template variable `{field_name}` not found in provided kwargs")
+126
+127# TODO: Reject if it's already a fragment, or accept *Fragment? Decide on API here
+128fragments.append(ValueFragment(kwargs[field_name]))
+129returnQuery(fragments)
+
+
+
+
Creates a Query - capable of performing query composition and simple querying. It can accept a
+simple string query, or can perform composition using ${} sigil string template with **kwargs as
+substitutions.
+
+
The **kwargs can be Fauna data types - such as strings, document references, or modules - and embedded
+Query - allowing you to compose arbitrarily complex queries.
+
+
When providing **kwargs, following types are accepted:
+ - str, int, float, bool, datetime.datetime, datetime.date,
+ dict, list, Query, DocumentReference, Module
+
+
Raises
+
+
+
ValueError: If there is an invalid template placeholder or a value that cannot be encoded.
+:returns: A Query that can be passed to the client for evaluation against Fauna.
1importreas_re
+ 2fromtypingimportOptional,Tuple,Iterator,Match
+ 3
+ 4
+ 5classFaunaTemplate:
+ 6"""A template class that supports variables marked with a ${}-sigil. Its primary purpose
+ 7 is to expose an iterator for the template parts that support composition of FQL queries.
+ 8
+ 9 Implementation adapted from https://github.com/python/cpython/blob/main/Lib/string.py
+10
+11 :param template: A string template e.g. "${my_var} { name }"
+12 :type template: str
+13 """
+14
+15_delimiter='$'
+16_idpattern=r'[_a-zA-Z][_a-zA-Z0-9]*'
+17_flags=_re.VERBOSE
+18
+19def__init__(self,template:str):
+20"""The initializer"""
+21delim=_re.escape(self._delimiter)
+22pattern=fr"""
+23{delim}(?:
+24 (?P<escaped>{delim}) | # Escape sequence of two delimiters
+25{{(?P<braced>{self._idpattern})}} | # delimiter and a braced identifier
+26 (?P<invalid>) # Other ill-formed delimiter exprs
+27 )
+28 """
+29self._pattern=_re.compile(pattern,self._flags)
+30self._template=template
+31
+32defiter(self)->Iterator[Tuple[Optional[str],Optional[str]]]:
+33"""A method that returns an iterator over tuples representing template parts. The
+34 first value of the tuple, if not None, is a template literal. The second value of
+35 the tuple, if not None, is a template variable. If both are not None, then the
+36 template literal comes *before* the variable.
+37
+38 :raises ValueError: If there is an invalid template placeholder
+39
+40 :return: An iterator of template parts
+41 :rtype: collections.Iterable[Tuple[Optional[str], Optional[str]]]
+42 """
+43match_objects=self._pattern.finditer(self._template)
+44cur_pos=0
+45formoinmatch_objects:
+46ifmo.group("invalid")isnotNone:
+47self._handle_invalid(mo)
+48
+49span_start_pos=mo.span()[0]
+50span_end_pos=mo.span()[1]
+51escaped_part=mo.group("escaped")or""
+52variable_part=mo.group("braced")
+53literal_part:Optional[str]=None
+54
+55ifcur_pos!=span_start_pos:
+56literal_part= \
+57self._template[cur_pos:span_start_pos] \
+58+escaped_part
+59
+60cur_pos=span_end_pos
+61
+62yieldliteral_part,variable_part
+63
+64ifcur_pos!=len(self._template):
+65yieldself._template[cur_pos:],None
+66
+67def_handle_invalid(self,mo:Match)->None:
+68i=mo.start("invalid")
+69lines=self._template[:i].splitlines(keepends=True)
+70
+71ifnotlines:
+72colno=1
+73lineno=1
+74else:
+75colno=i-len(''.join(lines[:-1]))
+76lineno=len(lines)
+77
+78raiseValueError(
+79f"Invalid placeholder in template: line {lineno}, col {colno}")
+
+
+
+
+
+
+
+
+ class
+ FaunaTemplate:
+
+
+
+
+
+
6classFaunaTemplate:
+ 7"""A template class that supports variables marked with a ${}-sigil. Its primary purpose
+ 8 is to expose an iterator for the template parts that support composition of FQL queries.
+ 9
+10 Implementation adapted from https://github.com/python/cpython/blob/main/Lib/string.py
+11
+12 :param template: A string template e.g. "${my_var} { name }"
+13 :type template: str
+14 """
+15
+16_delimiter='$'
+17_idpattern=r'[_a-zA-Z][_a-zA-Z0-9]*'
+18_flags=_re.VERBOSE
+19
+20def__init__(self,template:str):
+21"""The initializer"""
+22delim=_re.escape(self._delimiter)
+23pattern=fr"""
+24{delim}(?:
+25 (?P<escaped>{delim}) | # Escape sequence of two delimiters
+26{{(?P<braced>{self._idpattern})}} | # delimiter and a braced identifier
+27 (?P<invalid>) # Other ill-formed delimiter exprs
+28 )
+29 """
+30self._pattern=_re.compile(pattern,self._flags)
+31self._template=template
+32
+33defiter(self)->Iterator[Tuple[Optional[str],Optional[str]]]:
+34"""A method that returns an iterator over tuples representing template parts. The
+35 first value of the tuple, if not None, is a template literal. The second value of
+36 the tuple, if not None, is a template variable. If both are not None, then the
+37 template literal comes *before* the variable.
+38
+39 :raises ValueError: If there is an invalid template placeholder
+40
+41 :return: An iterator of template parts
+42 :rtype: collections.Iterable[Tuple[Optional[str], Optional[str]]]
+43 """
+44match_objects=self._pattern.finditer(self._template)
+45cur_pos=0
+46formoinmatch_objects:
+47ifmo.group("invalid")isnotNone:
+48self._handle_invalid(mo)
+49
+50span_start_pos=mo.span()[0]
+51span_end_pos=mo.span()[1]
+52escaped_part=mo.group("escaped")or""
+53variable_part=mo.group("braced")
+54literal_part:Optional[str]=None
+55
+56ifcur_pos!=span_start_pos:
+57literal_part= \
+58self._template[cur_pos:span_start_pos] \
+59+escaped_part
+60
+61cur_pos=span_end_pos
+62
+63yieldliteral_part,variable_part
+64
+65ifcur_pos!=len(self._template):
+66yieldself._template[cur_pos:],None
+67
+68def_handle_invalid(self,mo:Match)->None:
+69i=mo.start("invalid")
+70lines=self._template[:i].splitlines(keepends=True)
+71
+72ifnotlines:
+73colno=1
+74lineno=1
+75else:
+76colno=i-len(''.join(lines[:-1]))
+77lineno=len(lines)
+78
+79raiseValueError(
+80f"Invalid placeholder in template: line {lineno}, col {colno}")
+
+
+
+
A template class that supports variables marked with a ${}-sigil. Its primary purpose
+is to expose an iterator for the template parts that support composition of FQL queries.
33defiter(self)->Iterator[Tuple[Optional[str],Optional[str]]]:
+34"""A method that returns an iterator over tuples representing template parts. The
+35 first value of the tuple, if not None, is a template literal. The second value of
+36 the tuple, if not None, is a template variable. If both are not None, then the
+37 template literal comes *before* the variable.
+38
+39 :raises ValueError: If there is an invalid template placeholder
+40
+41 :return: An iterator of template parts
+42 :rtype: collections.Iterable[Tuple[Optional[str], Optional[str]]]
+43 """
+44match_objects=self._pattern.finditer(self._template)
+45cur_pos=0
+46formoinmatch_objects:
+47ifmo.group("invalid")isnotNone:
+48self._handle_invalid(mo)
+49
+50span_start_pos=mo.span()[0]
+51span_end_pos=mo.span()[1]
+52escaped_part=mo.group("escaped")or""
+53variable_part=mo.group("braced")
+54literal_part:Optional[str]=None
+55
+56ifcur_pos!=span_start_pos:
+57literal_part= \
+58self._template[cur_pos:span_start_pos] \
+59+escaped_part
+60
+61cur_pos=span_end_pos
+62
+63yieldliteral_part,variable_part
+64
+65ifcur_pos!=len(self._template):
+66yieldself._template[cur_pos:],None
+
+
+
+
A method that returns an iterator over tuples representing template parts. The
+first value of the tuple, if not None, is a template literal. The second value of
+the tuple, if not None, is a template variable. If both are not None, then the
+template literal comes before the variable.
+
+
Raises
+
+
+
ValueError: If there is an invalid template placeholder
+
+
+
Returns
+
+
+
An iterator of template parts
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/2.2.0/api/index.html b/2.2.0/api/index.html
new file mode 100644
index 00000000..cd7f994c
--- /dev/null
+++ b/2.2.0/api/index.html
@@ -0,0 +1,7 @@
+
+
+
+
+
+
+
diff --git a/2.2.0/api/search.js b/2.2.0/api/search.js
new file mode 100644
index 00000000..ab6a3437
--- /dev/null
+++ b/2.2.0/api/search.js
@@ -0,0 +1,46 @@
+window.pdocSearch = (function(){
+/** elasticlunr - http://weixsong.github.io * Copyright (C) 2017 Oliver Nightingale * Copyright (C) 2017 Wei Song * MIT Licensed */!function(){function e(e){if(null===e||"object"!=typeof e)return e;var t=e.constructor();for(var n in e)e.hasOwnProperty(n)&&(t[n]=e[n]);return t}var t=function(e){var n=new t.Index;return n.pipeline.add(t.trimmer,t.stopWordFilter,t.stemmer),e&&e.call(n,n),n};t.version="0.9.5",lunr=t,t.utils={},t.utils.warn=function(e){return function(t){e.console&&console.warn&&console.warn(t)}}(this),t.utils.toString=function(e){return void 0===e||null===e?"":e.toString()},t.EventEmitter=function(){this.events={}},t.EventEmitter.prototype.addListener=function(){var e=Array.prototype.slice.call(arguments),t=e.pop(),n=e;if("function"!=typeof t)throw new TypeError("last argument must be a function");n.forEach(function(e){this.hasHandler(e)||(this.events[e]=[]),this.events[e].push(t)},this)},t.EventEmitter.prototype.removeListener=function(e,t){if(this.hasHandler(e)){var n=this.events[e].indexOf(t);-1!==n&&(this.events[e].splice(n,1),0==this.events[e].length&&delete this.events[e])}},t.EventEmitter.prototype.emit=function(e){if(this.hasHandler(e)){var t=Array.prototype.slice.call(arguments,1);this.events[e].forEach(function(e){e.apply(void 0,t)},this)}},t.EventEmitter.prototype.hasHandler=function(e){return e in this.events},t.tokenizer=function(e){if(!arguments.length||null===e||void 0===e)return[];if(Array.isArray(e)){var n=e.filter(function(e){return null===e||void 0===e?!1:!0});n=n.map(function(e){return t.utils.toString(e).toLowerCase()});var i=[];return n.forEach(function(e){var n=e.split(t.tokenizer.seperator);i=i.concat(n)},this),i}return e.toString().trim().toLowerCase().split(t.tokenizer.seperator)},t.tokenizer.defaultSeperator=/[\s\-]+/,t.tokenizer.seperator=t.tokenizer.defaultSeperator,t.tokenizer.setSeperator=function(e){null!==e&&void 0!==e&&"object"==typeof e&&(t.tokenizer.seperator=e)},t.tokenizer.resetSeperator=function(){t.tokenizer.seperator=t.tokenizer.defaultSeperator},t.tokenizer.getSeperator=function(){return t.tokenizer.seperator},t.Pipeline=function(){this._queue=[]},t.Pipeline.registeredFunctions={},t.Pipeline.registerFunction=function(e,n){n in t.Pipeline.registeredFunctions&&t.utils.warn("Overwriting existing registered function: "+n),e.label=n,t.Pipeline.registeredFunctions[n]=e},t.Pipeline.getRegisteredFunction=function(e){return e in t.Pipeline.registeredFunctions!=!0?null:t.Pipeline.registeredFunctions[e]},t.Pipeline.warnIfFunctionNotRegistered=function(e){var n=e.label&&e.label in this.registeredFunctions;n||t.utils.warn("Function is not registered with pipeline. This may cause problems when serialising the index.\n",e)},t.Pipeline.load=function(e){var n=new t.Pipeline;return e.forEach(function(e){var i=t.Pipeline.getRegisteredFunction(e);if(!i)throw new Error("Cannot load un-registered function: "+e);n.add(i)}),n},t.Pipeline.prototype.add=function(){var e=Array.prototype.slice.call(arguments);e.forEach(function(e){t.Pipeline.warnIfFunctionNotRegistered(e),this._queue.push(e)},this)},t.Pipeline.prototype.after=function(e,n){t.Pipeline.warnIfFunctionNotRegistered(n);var i=this._queue.indexOf(e);if(-1===i)throw new Error("Cannot find existingFn");this._queue.splice(i+1,0,n)},t.Pipeline.prototype.before=function(e,n){t.Pipeline.warnIfFunctionNotRegistered(n);var i=this._queue.indexOf(e);if(-1===i)throw new Error("Cannot find existingFn");this._queue.splice(i,0,n)},t.Pipeline.prototype.remove=function(e){var t=this._queue.indexOf(e);-1!==t&&this._queue.splice(t,1)},t.Pipeline.prototype.run=function(e){for(var t=[],n=e.length,i=this._queue.length,o=0;n>o;o++){for(var r=e[o],s=0;i>s&&(r=this._queue[s](r,o,e),void 0!==r&&null!==r);s++);void 0!==r&&null!==r&&t.push(r)}return t},t.Pipeline.prototype.reset=function(){this._queue=[]},t.Pipeline.prototype.get=function(){return this._queue},t.Pipeline.prototype.toJSON=function(){return this._queue.map(function(e){return t.Pipeline.warnIfFunctionNotRegistered(e),e.label})},t.Index=function(){this._fields=[],this._ref="id",this.pipeline=new t.Pipeline,this.documentStore=new t.DocumentStore,this.index={},this.eventEmitter=new t.EventEmitter,this._idfCache={},this.on("add","remove","update",function(){this._idfCache={}}.bind(this))},t.Index.prototype.on=function(){var e=Array.prototype.slice.call(arguments);return this.eventEmitter.addListener.apply(this.eventEmitter,e)},t.Index.prototype.off=function(e,t){return this.eventEmitter.removeListener(e,t)},t.Index.load=function(e){e.version!==t.version&&t.utils.warn("version mismatch: current "+t.version+" importing "+e.version);var n=new this;n._fields=e.fields,n._ref=e.ref,n.documentStore=t.DocumentStore.load(e.documentStore),n.pipeline=t.Pipeline.load(e.pipeline),n.index={};for(var i in e.index)n.index[i]=t.InvertedIndex.load(e.index[i]);return n},t.Index.prototype.addField=function(e){return this._fields.push(e),this.index[e]=new t.InvertedIndex,this},t.Index.prototype.setRef=function(e){return this._ref=e,this},t.Index.prototype.saveDocument=function(e){return this.documentStore=new t.DocumentStore(e),this},t.Index.prototype.addDoc=function(e,n){if(e){var n=void 0===n?!0:n,i=e[this._ref];this.documentStore.addDoc(i,e),this._fields.forEach(function(n){var o=this.pipeline.run(t.tokenizer(e[n]));this.documentStore.addFieldLength(i,n,o.length);var r={};o.forEach(function(e){e in r?r[e]+=1:r[e]=1},this);for(var s in r){var u=r[s];u=Math.sqrt(u),this.index[n].addToken(s,{ref:i,tf:u})}},this),n&&this.eventEmitter.emit("add",e,this)}},t.Index.prototype.removeDocByRef=function(e){if(e&&this.documentStore.isDocStored()!==!1&&this.documentStore.hasDoc(e)){var t=this.documentStore.getDoc(e);this.removeDoc(t,!1)}},t.Index.prototype.removeDoc=function(e,n){if(e){var n=void 0===n?!0:n,i=e[this._ref];this.documentStore.hasDoc(i)&&(this.documentStore.removeDoc(i),this._fields.forEach(function(n){var o=this.pipeline.run(t.tokenizer(e[n]));o.forEach(function(e){this.index[n].removeToken(e,i)},this)},this),n&&this.eventEmitter.emit("remove",e,this))}},t.Index.prototype.updateDoc=function(e,t){var t=void 0===t?!0:t;this.removeDocByRef(e[this._ref],!1),this.addDoc(e,!1),t&&this.eventEmitter.emit("update",e,this)},t.Index.prototype.idf=function(e,t){var n="@"+t+"/"+e;if(Object.prototype.hasOwnProperty.call(this._idfCache,n))return this._idfCache[n];var i=this.index[t].getDocFreq(e),o=1+Math.log(this.documentStore.length/(i+1));return this._idfCache[n]=o,o},t.Index.prototype.getFields=function(){return this._fields.slice()},t.Index.prototype.search=function(e,n){if(!e)return[];e="string"==typeof e?{any:e}:JSON.parse(JSON.stringify(e));var i=null;null!=n&&(i=JSON.stringify(n));for(var o=new t.Configuration(i,this.getFields()).get(),r={},s=Object.keys(e),u=0;u0&&t.push(e);for(var i in n)"docs"!==i&&"df"!==i&&this.expandToken(e+i,t,n[i]);return t},t.InvertedIndex.prototype.toJSON=function(){return{root:this.root}},t.Configuration=function(e,n){var e=e||"";if(void 0==n||null==n)throw new Error("fields should not be null");this.config={};var i;try{i=JSON.parse(e),this.buildUserConfig(i,n)}catch(o){t.utils.warn("user configuration parse failed, will use default configuration"),this.buildDefaultConfig(n)}},t.Configuration.prototype.buildDefaultConfig=function(e){this.reset(),e.forEach(function(e){this.config[e]={boost:1,bool:"OR",expand:!1}},this)},t.Configuration.prototype.buildUserConfig=function(e,n){var i="OR",o=!1;if(this.reset(),"bool"in e&&(i=e.bool||i),"expand"in e&&(o=e.expand||o),"fields"in e)for(var r in e.fields)if(n.indexOf(r)>-1){var s=e.fields[r],u=o;void 0!=s.expand&&(u=s.expand),this.config[r]={boost:s.boost||0===s.boost?s.boost:1,bool:s.bool||i,expand:u}}else t.utils.warn("field name in user configuration not found in index instance fields");else this.addAllFields2UserConfig(i,o,n)},t.Configuration.prototype.addAllFields2UserConfig=function(e,t,n){n.forEach(function(n){this.config[n]={boost:1,bool:e,expand:t}},this)},t.Configuration.prototype.get=function(){return this.config},t.Configuration.prototype.reset=function(){this.config={}},lunr.SortedSet=function(){this.length=0,this.elements=[]},lunr.SortedSet.load=function(e){var t=new this;return t.elements=e,t.length=e.length,t},lunr.SortedSet.prototype.add=function(){var e,t;for(e=0;e1;){if(r===e)return o;e>r&&(t=o),r>e&&(n=o),i=n-t,o=t+Math.floor(i/2),r=this.elements[o]}return r===e?o:-1},lunr.SortedSet.prototype.locationFor=function(e){for(var t=0,n=this.elements.length,i=n-t,o=t+Math.floor(i/2),r=this.elements[o];i>1;)e>r&&(t=o),r>e&&(n=o),i=n-t,o=t+Math.floor(i/2),r=this.elements[o];return r>e?o:e>r?o+1:void 0},lunr.SortedSet.prototype.intersect=function(e){for(var t=new lunr.SortedSet,n=0,i=0,o=this.length,r=e.length,s=this.elements,u=e.elements;;){if(n>o-1||i>r-1)break;s[n]!==u[i]?s[n]u[i]&&i++:(t.add(s[n]),n++,i++)}return t},lunr.SortedSet.prototype.clone=function(){var e=new lunr.SortedSet;return e.elements=this.toArray(),e.length=e.elements.length,e},lunr.SortedSet.prototype.union=function(e){var t,n,i;this.length>=e.length?(t=this,n=e):(t=e,n=this),i=t.clone();for(var o=0,r=n.toArray();o
A dataclass representing options available for a query.
\n\n
\n
linearized - If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
\n
max_contention_retries - The max number of times to retry the query if contention is encountered.
\n
query_timeout - Controls the maximum amount of time Fauna will execute your query before marking it failed.
\n
query_tags - Tags to associate with the query. See logging
typecheck - Enable or disable typechecking of the query before evaluation. If not set, the value configured on the Client will be used. If neither is set, Fauna will use the value of the \"typechecked\" flag on the database configuration.
\n
additional_headers - Add/update HTTP request headers for the query. In general, this should not be necessary.
A dataclass representing options available for a stream.
\n\n
\n
max_attempts - The maximum number of times to attempt a stream query when a retryable exception is thrown.
\n
max_backoff - The maximum backoff in seconds for an individual retry.
\n
start_ts - The starting timestamp of the stream, exclusive. If set, Fauna will return events starting after\nthe timestamp.
\n
cursor - The starting event cursor, exclusive. If set, Fauna will return events starting after the cursor.
\n
status_events - Indicates if stream should include status events. Status events are periodic events that\nupdate the client with the latest valid timestamp (in the event of a dropped connection) as well as metrics\nabout the cost of maintaining the stream other than the cost of the received events.
endpoint: The Fauna Endpoint to use. Defaults to https: //db.fauna.com, or the FAUNA_ENDPOINT env variable.
\n
secret: The Fauna Secret to use. Defaults to empty, or the FAUNA_SECRET env variable.
\n
http_client: An HTTPClient implementation. Defaults to a global HTTPXClient.
\n
**query_tags: Tags to associate with the query. See logging
\n
linearized: If true, unconditionally run the query as strictly serialized. This affects read-only transactions. Transactions which write will always be strictly serialized.
\n
max_contention_retries: The max number of times to retry the query if contention is encountered.
\n
typecheck: Enable or disable typechecking of the query before evaluation. If not set, Fauna will use the value of the \"typechecked\" flag on the database configuration.
\n
additional_headers: Add/update HTTP request headers for the query. In general, this should not be necessary.
\n
query_timeout: Controls the maximum amount of time Fauna will execute your query before marking it failed, default is DefaultQueryTimeout.
\n
client_buffer_timeout: Time in milliseconds beyond query_timeout at which the client will abort a request if it has not received a response. The default is DefaultClientBufferTimeout, which should account for network latency for most clients. The value must be greater than zero. The closer to zero the value is, the more likely the client is to abort the request before the server can report a legitimate response or error.
\n
http_read_timeout: Set HTTP Read timeout, default is DefaultHttpReadTimeout.
\n
http_write_timeout: Set HTTP Write timeout, default is DefaultHttpWriteTimeout.
\n
http_connect_timeout: Set HTTP Connect timeout, default is DefaultHttpConnectTimeout.
\n
http_pool_timeout: Set HTTP Pool timeout, default is DefaultHttpPoolTimeout.
\n
http_idle_timeout: Set HTTP Idle timeout, default is DefaultIdleConnectionTimeout.
\n
max_attempts: The maximum number of times to attempt a query when a retryable exception is thrown. Defaults to 3.
\n
max_backoff: The maximum backoff in seconds for an individual retry. Defaults to 20.
Set the last timestamp seen by this client.\nThis has no effect if earlier than stored timestamp.
\n\n
.. WARNING:: This should be used only when coordinating timestamps across\nmultiple clients. Moving the timestamp arbitrarily forward into\nthe future will cause transactions to stall.
Run a query on Fauna and returning an iterator of results. If the query\nreturns a Page, the iterator will fetch additional Pages until the\nafter token is null. Each call for a page will be retried with exponential\nbackoff up to the max_attempts set in the client's retry policy in the\nevent of a 429 or 502.
A generator function that immediately fetches and yields the results of\nthe stored query. Yields additional pages on subsequent iterations if\nthey exist
A generator function that immediately fetches and yields the results of\nthe stored query. Yields each item individually, rather than a whole\nPage at a time. Fetches additional pages as required if they exist.
Runs the wrapped function. Retries up to max_attempts if the function throws a RetryableFaunaException. It propagates\nthe thrown exception if max_attempts is reached or if a non-retryable is thrown.
Updates the internal transaction time.\nIn order to maintain a monotonically-increasing value, newTxnTime\nis discarded if it is behind the current timestamp.
Decodes supported objects from the tagged typed into untagged.
\n\n
Examples:\n - { \"@int\": \"100\" } decodes to 100 of type int\n - { \"@double\": \"100\" } decodes to 100.0 of type float\n - { \"@long\": \"100\" } decodes to 100 of type int\n - { \"@time\": \"...\" } decodes to a datetime\n - { \"@date\": \"...\" } decodes to a date\n - { \"@doc\": ... } decodes to a Document or NamedDocument\n - { \"@ref\": ... } decodes to a DocumentReference or NamedDocumentReference\n - { \"@mod\": ... } decodes to a Module\n - { \"@set\": ... } decodes to a Page\n - { \"@stream\": ... } decodes to a StreamToken\n - { \"@bytes\": ... } decodes to a bytearray
Examples:\n - Up to 32-bit ints encode to { \"@int\": \"...\" }\n - Up to 64-bit ints encode to { \"@long\": \"...\" }\n - Floats encode to { \"@double\": \"...\" }\n - datetime encodes to { \"@time\": \"...\" }\n - date encodes to { \"@date\": \"...\" }\n - DocumentReference encodes to { \"@doc\": \"...\" }\n - Module encodes to { \"@mod\": \"...\" }\n - Query encodes to { \"fql\": [...] }\n - ValueFragment encodes to { \"value\": }\n - LiteralFragment encodes to a string\n - StreamToken encodes to a string
\n\n
Raises
\n\n
\n
ValueError: If value cannot be encoded, cannot be encoded safely, or there's a circular reference.
An error representing a failure internal to the client, itself.\nThis indicates Fauna was never called - the client failed internally\nprior to sending the request.
An error response that is the result of the query failing during execution.\nQueryRuntimeError's occur when a bug in your query causes an invalid execution\nto be requested.\nThe 'code' field will vary based on the specific error cause.
A failure due to the timeout being exceeded, but the timeout\nwas set lower than the query's expected processing time.\nThis response is distinguished from a ServiceTimeoutException\nin that a QueryTimeoutError shows Fauna behaving in an expected manner.
A class representing a user document stored in Fauna.
\n\n
User data should be stored directly on the map, while id, ts, and coll should only be stored on the related\nproperties. When working with a Document in code, it should be considered immutable.
A class representing a named document stored in Fauna. Examples of named documents include Collection\ndefinitions, Index definitions, and Roles, among others.
\n\n
When working with a NamedDocument in code, it should be considered immutable.
A concrete Fragment representing a part of a query that can represent a template variable.\nFor example, if a template contains a variable ${foo}, and an object { \"prop\": 1 } is provided for foo,\nthen { \"prop\": 1 } should be wrapped as a ValueFragment.
A concrete Fragment representing a query literal For example, in the template let x = ${foo},\nthe portion let x = is a query literal and should be wrapped as a LiteralFragment.
\n\n
Parameters
\n\n
\n
str val: The query literal to be used as a fragment.
Creates a Query - capable of performing query composition and simple querying. It can accept a\nsimple string query, or can perform composition using ${} sigil string template with **kwargs as\nsubstitutions.
\n\n
The **kwargs can be Fauna data types - such as strings, document references, or modules - and embedded\nQuery - allowing you to compose arbitrarily complex queries.
\n\n
When providing **kwargs, following types are accepted:\n - str, int, float, bool, datetime.datetime, datetime.date,\n dict, list, Query, DocumentReference, Module
\n\n
Raises
\n\n
\n
ValueError: If there is an invalid template placeholder or a value that cannot be encoded.\n:returns: A Query that can be passed to the client for evaluation against Fauna.
A template class that supports variables marked with a ${}-sigil. Its primary purpose\nis to expose an iterator for the template parts that support composition of FQL queries.
A method that returns an iterator over tuples representing template parts. The\nfirst value of the tuple, if not None, is a template literal. The second value of\nthe tuple, if not None, is a template variable. If both are not None, then the\ntemplate literal comes before the variable.
\n\n
Raises
\n\n
\n
ValueError: If there is an invalid template placeholder
\n
\n\n
Returns
\n\n
\n
An iterator of template parts
\n
\n", "signature": "(self) -> Iterator[Tuple[Optional[str],Optional[str]]]:", "funcdef": "def"}];
+
+ // mirrored in build-search-index.js (part 1)
+ // Also split on html tags. this is a cheap heuristic, but good enough.
+ elasticlunr.tokenizer.setSeperator(/[\s\-.;&_'"=,()]+|<[^>]*>/);
+
+ let searchIndex;
+ if (docs._isPrebuiltIndex) {
+ console.info("using precompiled search index");
+ searchIndex = elasticlunr.Index.load(docs);
+ } else {
+ console.time("building search index");
+ // mirrored in build-search-index.js (part 2)
+ searchIndex = elasticlunr(function () {
+ this.pipeline.remove(elasticlunr.stemmer);
+ this.pipeline.remove(elasticlunr.stopWordFilter);
+ this.addField("qualname");
+ this.addField("fullname");
+ this.addField("annotation");
+ this.addField("default_value");
+ this.addField("signature");
+ this.addField("bases");
+ this.addField("doc");
+ this.setRef("fullname");
+ });
+ for (let doc of docs) {
+ searchIndex.addDoc(doc);
+ }
+ console.timeEnd("building search index");
+ }
+
+ return (term) => searchIndex.search(term, {
+ fields: {
+ qualname: {boost: 4},
+ fullname: {boost: 2},
+ annotation: {boost: 2},
+ default_value: {boost: 2},
+ signature: {boost: 2},
+ bases: {boost: 2},
+ doc: {boost: 1},
+ },
+ expand: true
+ });
+})();
\ No newline at end of file
diff --git a/latest b/latest
index 50aea0e7..e3a4f193 120000
--- a/latest
+++ b/latest
@@ -1 +1 @@
-2.1.0
\ No newline at end of file
+2.2.0
\ No newline at end of file