U
    h                     @   st  d dl Z d dlZd dlZd dlZd dlZd dlT d dlmZ d dlmZ d dlmZ d dlm	Z	 d dlm
Z
 d dlmZ d d	lmZ d d
lmZ d dlmZ d dlmZ d dlmZ z<d dlmZmZmZmZmZmZmZmZmZmZmZ dZW n ek
r
   dZY nX ejd  dkr e Z!dZ"dZ#edk	rPej$dd dkrJdndZ%ndZ%dZ&G dd de'Z(G dd de(Z)G dd de'Z*G dd de+Z,G dd  d eZ-G d!d" d"e-Z.G d#d$ d$e/Z0G d%d& d&e0Z1G d'd( d(e2Z3G d)d* d*e4Z5G d+d, d,e6Z7G d-d. d.e7Z8G d/d0 d0e8Z9d1Z:e;d2e;e:B e;e:< B e;e=d3fB Z>e;d4d5 e?d6D Z@[:[>eAd7ZBG d8d9 d9e8ZCd]d:d;ZDG d<d= d=e7ZEd>e_Fd?d@ ZGG dAdB dBeHZIer@d ZJdCZKdDZLdZMdZNdEZOdFZPdGZQdHZRdIZSd ZTdCZUdDZVdZWdZXdEZYdFZZdGZ[dHZ\dIZ]dJZ^d^dKdLZ_d_dMdNZ`G dOdP dPeIZadQdR ZbdSdT ZcdUdV ZddWdX ZedYdZ Zfd[d\ ZgdS )`    N)*)
ColumnBase)EnclosedNodeList)Entity)
Expression)Insert)Node)NodeList)OP)VirtualField)
merge_dict)sqlite3)backupbackup_to_fileBlobConnectionHelperregister_bloomfilterregister_hash_functionsregister_rank_functionssqlite_get_db_statussqlite_get_statusTableFunctionZeroBlobTF   ZpcxZpcnalx)r         r   )r   	   r   c                       s,   e Zd ZdZd Z ZZ fddZ  ZS )
RowIDFieldTrowidc                    s<   || j kr tdt| | j f tt| j||f|  d S )Nz%s must be named "%s".)required_name
ValueErrortypesuperr   bind)selfmodelnameargs	__class__ 8/tmp/pip-unpacked-wheel-5j60pwdk/playhouse/sqlite_ext.pyr#   8   s
    
zRowIDField.bind)	__name__
__module____qualname__Zauto_incrementcolumn_namer&   r   r#   __classcell__r*   r*   r(   r+   r   4   s   r   c                   @   s   e Zd Zd Z ZZdS )
DocIDFielddocidN)r,   r-   r.   r/   r&   r   r*   r*   r*   r+   r1   ?   s   r1   c                       s   e Zd Z fddZ  ZS )AutoIncrementFieldc                    s    t t| |}t|tdfS )NZAUTOINCREMENT)r"   r3   ddlr	   SQL)r$   ctxZ	node_listr(   r*   r+   r4   D   s    zAutoIncrementField.ddl)r,   r-   r.   r4   r0   r*   r*   r(   r+   r3   C   s   r3   c                   @   s   e Zd ZdZdd ZdS )TDecimalFieldTEXTc                 C   s   d S Nr*   r$   r*   r*   r+   get_modifiersK       zTDecimalField.get_modifiersN)r,   r-   r.   
field_typer;   r*   r*   r*   r+   r7   I   s   r7   c                       s   e Zd Zd  fdd	Zedd Zdd Zd!dd	Zd"d
dZd#ddZ	d$ddZ
d%ddZdd Zdd Zdd Zdd Zdd Zdd Zdd Z  ZS )&JSONPathNc                    s"   t t|   || _|pd| _d S )Nr*   )r"   r>   __init___field_path)r$   fieldpathr(   r*   r+   r?   O   s    zJSONPath.__init__c                 C   s   t dd| j S )Nz$%s )ValuejoinrA   r:   r*   r*   r+   rC   T   s    zJSONPath.pathc                 C   s<   t |ts|dkrd| }nd| }t| | j| j|f S )N#z[%s]z.%s)
isinstanceintr!   r@   rA   )r$   idxitemr*   r*   r+   __getitem__X   s    
zJSONPath.__getitem__c                 C   s:   |st |ttfr$t| j|}t| j| d j|S NrG   )	rH   listdictfnjsonr@   _json_dumpsjson_setrC   r$   valueas_jsonr*   r*   r+   append_   s    zJSONPath.appendc                 C   s4   |st |ttfr$t| j|}|| j| j|S r9   )rH   rN   rO   rP   rQ   r@   rR   rC   r$   funcrU   rV   r*   r*   r+   _json_operationd   s    zJSONPath._json_operationc                 C   s   |  tj||S r9   )rZ   rP   Zjson_insertrT   r*   r*   r+   inserti   s    zJSONPath.insertc                 C   s   |  tj||S r9   )rZ   rP   rS   rT   r*   r*   r+   setl   s    zJSONPath.setc                 C   s   |  tj||S r9   )rZ   rP   Zjson_replacerT   r*   r*   r+   replaceo   s    zJSONPath.replacec                 C   s   |  t| | j|S r9   )r\   rP   Z
json_patchr@   rR   r$   rU   r*   r*   r+   updater   s    zJSONPath.updatec                 C   s   t | j| jS r9   )rP   json_remover@   rC   r:   r*   r*   r+   removeu   s    zJSONPath.removec                 C   s   t | j| jS r9   )rP   	json_typer@   rC   r:   r*   r*   r+   rb   x   s    zJSONPath.json_typec                 C   s   t | j| jS r9   )rP   json_array_lengthr@   rC   r:   r*   r*   r+   length{   s    zJSONPath.lengthc                 C   s   t | j| jS r9   )rP   	json_eachr@   rC   r:   r*   r*   r+   children~   s    zJSONPath.childrenc                 C   s   t | j| jS r9   )rP   	json_treer@   rC   r:   r*   r*   r+   tree   s    zJSONPath.treec                 C   s"   | | jrt| j| jn| jS r9   )sqlrA   rP   json_extractr@   rC   r$   r6   r*   r*   r+   __sql__   s
    zJSONPath.__sql__)N)N)N)N)N)N)r,   r-   r.   r?   propertyrC   rL   rW   rZ   r[   r\   r]   r_   ra   rb   rd   rf   rh   rl   r0   r*   r*   r(   r+   r>   N   s    





r>   c                   @   sV   e Zd ZdddZdddZdddZddd	Zdd
dZdd Zdd Z	dd Z
dS )	JSONBPathNc                 C   s:   |st |ttfr$t| j|}t| j| d j|S rM   )	rH   rN   rO   rP   jsonbr@   rR   	jsonb_setrC   rT   r*   r*   r+   rW      s    zJSONBPath.appendc                 C   s4   |st |ttfr$t| j|}|| j| j|S r9   )rH   rN   rO   rP   ro   r@   rR   rC   rX   r*   r*   r+   rZ      s    zJSONBPath._json_operationc                 C   s   |  tj||S r9   )rZ   rP   Zjsonb_insertrT   r*   r*   r+   r[      s    zJSONBPath.insertc                 C   s   |  tj||S r9   )rZ   rP   rp   rT   r*   r*   r+   r\      s    zJSONBPath.setc                 C   s   |  tj||S r9   )rZ   rP   Zjsonb_replacerT   r*   r*   r+   r]      s    zJSONBPath.replacec                 C   s   |  t| | j|S r9   )r\   rP   Zjsonb_patchr@   rR   r^   r*   r*   r+   r_      s    zJSONBPath.updatec                 C   s   t | j| jS r9   )rP   jsonb_remover@   rC   r:   r*   r*   r+   ra      s    zJSONBPath.removec                 C   s"   | | jrt| j| jn| jS r9   )ri   rA   rP   jsonb_extractr@   rC   rk   r*   r*   r+   rl      s
    zJSONBPath.__sql__)N)N)N)N)N)r,   r-   r.   rW   rZ   r[   r\   r]   r_   ra   rl   r*   r*   r*   r+   rn      s   




rn   c                       s   e Zd ZdZdZeZd( fdd	Zdd Zdd	 Z	d
d Z
e
ejZe
ejZe
ejZe
ejZe
ejZe
ejZejZdd Zdd Zdd Zdd Zd)ddZd*ddZd+ddZ d,ddZ!dd Z"dd Z#d d! Z$d-d"d#Z%d$d% Z&d&d' Z'  Z(S ).	JSONFieldZJSONFNc                    s.   |pt j| _|pt j| _tt| jf | d S r9   )rQ   dumpsrR   loads_json_loadsr"   rs   r?   )r$   Z
json_dumpsZ
json_loadskwargsr(   r*   r+   r?      s    zJSONField.__init__c              	   C   s6   |d k	r2z|  |W S  ttfk
r0   | Y S X d S r9   )rv   	TypeErrorr    r^   r*   r*   r+   python_value   s
    zJSONField.python_valuec                 C   s*   |d k	r&t |ts"t| |}|S d S r9   )rH   r   rP   rQ   rR   r^   r*   r*   r+   db_value   s    
zJSONField.db_valuec                    s    fdd}|S )Nc                    s*   t |ttfrt|| jdd}t|  |S )NF)	converterunpack)rH   rN   rO   rE   rz   r   )r$   rhsopr*   r+   inner   s    zJSONField._e.<locals>.innerr*   )r   r   r*   r~   r+   _e   s    zJSONField._ec                 C   s   |  | | S r9   )Path)r$   rK   r*   r*   r+   rL      s    zJSONField.__getitem__c                 G   s   dd |D }t j| f| S )Nc                 S   s   g | ]}t |d dqS Fr{   rE   .0pr*   r*   r+   
<listcomp>   s     z%JSONField.extract.<locals>.<listcomp>)rP   rj   r$   pathsr*   r*   r+   extract   s    zJSONField.extractc                 C   s   t | dt|ddS )Nz->Fr   r   rE   r$   rC   r*   r*   r+   extract_json   s    zJSONField.extract_jsonc                 C   s   t | dt|ddS )Nz->>Fr   r   r   r*   r*   r+   extract_text   s    zJSONField.extract_textc                 C   s   |  | ||S r9   )r   rW   rT   r*   r*   r+   rW      s    zJSONField.appendc                 C   s   |  | ||S r9   )r   r[   rT   r*   r*   r+   r[      s    zJSONField.insertc                 C   s   |  | ||S r9   )r   r\   rT   r*   r*   r+   r\      s    zJSONField.setc                 C   s   |  | ||S r9   )r   r]   rT   r*   r*   r+   r]      s    zJSONField.replacec                 C   s   |  | |S r9   )r   r_   )r$   datar*   r*   r+   r_      s    zJSONField.updatec                 G   s"   |s|  |  S tj| f| S r9   )r   ra   rP   r`   r   r*   r*   r+   ra      s    zJSONField.removec                 C   s
   t | S r9   )rP   rb   r:   r*   r*   r+   rb      s    zJSONField.json_typec                 C   s   |r| |fn| f}t j| S r9   )rP   rc   )r$   rC   r'   r*   r*   r+   rd      s    zJSONField.lengthc                 C   s
   t | S )a?  
        Schema of `json_each` and `json_tree`:

        key,
        value,
        type TEXT (object, array, string, etc),
        atom (value for primitive/scalar types, NULL for array and object)
        id INTEGER (unique identifier for element)
        parent INTEGER (unique identifier of parent element or NULL)
        fullkey TEXT (full path describing element)
        path TEXT (path to the container of the current element)
        json JSON hidden (1st input parameter to function)
        root TEXT hidden (2nd input parameter, path at which to start)
        )rP   re   r:   r*   r*   r+   rf      s    zJSONField.childrenc                 C   s
   t | S r9   )rP   rg   r:   r*   r*   r+   rh     s    zJSONField.tree)NN)N)N)N)N)N))r,   r-   r.   r=   r|   r>   r   r?   ry   rz   r   r
   ZEQ__eq__ZNE__ne__GT__gt__ZGTE__ge__LT__lt__ZLTE__le__Field__hash__rL   r   r   r   rW   r[   r\   r]   r_   ra   rb   rd   rf   rh   r0   r*   r*   r(   r+   rs      s8   










rs   c                   @   s4   e Zd ZdZeZdd Zdd Zdd Zdd	 Z	d
S )
JSONBFieldZJSONBc                 C   s*   |d k	r&t |ts"t| |}|S d S r9   )rH   r   rP   ro   rR   r^   r*   r*   r+   rz     s    
zJSONBField.db_valuec                 C   s
   t | S r9   )rP   rQ   r:   r*   r*   r+   rQ     s    zJSONBField.jsonc                 G   s   dd |D }t j| f| S )Nc                 S   s   g | ]}t |d dqS r   r   r   r*   r*   r+   r     s     z&JSONBField.extract.<locals>.<listcomp>)rP   rr   r   r*   r*   r+   r     s    zJSONBField.extractc                 G   s"   |s|  |  S tj| f| S r9   )r   ra   rP   rq   r   r*   r*   r+   ra     s    zJSONBField.removeN)
r,   r-   r.   r=   rn   r   rz   rQ   r   ra   r*   r*   r*   r+   r     s   r   c                       sD   e Zd Zd fdd	Zdd Zedd Zd	d
 ZdddZ  Z	S )SearchFieldFNc                    s.   |rt dt| tt| j||dd d S )Nz8SearchField does not accept these keyword arguments: %s.T)	unindexedr/   null)r    sortedr"   r   r?   )r$   r   r/   kr(   r*   r+   r?     s     zSearchField.__init__c                 C   s
   t | |S r9   )match)r$   termr*   r*   r+   r   &  s    zSearchField.matchc                 C   s2   t | ds,dd | jjjD }|| j| _| jS )N_fts_column_indexc                 S   s   g | ]}t |tr|jqS r*   )rH   r   r&   )r   fr*   r*   r+   r   ,  s    
z0SearchField.fts_column_index.<locals>.<listcomp>)hasattrr%   _metasorted_fieldsindexr&   r   )r$   Zsearch_fieldsr*   r*   r+   fts_column_index)  s    
zSearchField.fts_column_indexc                 C   s   | j }t| jjj|||S r9   )r   rP   	highlightr%   r   entity)r$   leftright
column_idxr*   r*   r+   r   1  s    zSearchField.highlight...   c                 C   s>   d|  k rdk sn t d| j}t| jjj|||||S )Nr   A   z/max_tokens must be between 1 and 64 (inclusive))r    r   rP   snippetr%   r   r   )r$   r   r   Zover_lengthZ
max_tokensr   r*   r*   r+   r   5  s     zSearchField.snippet)FN)r   r   )
r,   r-   r.   r?   r   rm   r   r   r   r0   r*   r*   r(   r+   r     s   
r   c                       s(   e Zd ZdddZd fdd	Z  ZS )VirtualTableSchemaManagerTc           	      K   s2  | j t| j jj|}|  }|d |r8|d || j d | j jj}t	|t
rh||S |t|d g }| j j}|jr|dd |jD  |jD ]D}t	|ts|jrqt|jg}|jr|td |t| q|jr|dd |jD  |r$|| | |t|S )	NzCREATE VIRTUAL TABLE zIF NOT EXISTS z USING  c                 S   s   g | ]}t |qS r*   r5   r   ar*   r*   r+   r   W  s     zCVirtualTableSchemaManager._create_virtual_table.<locals>.<listcomp>Z	UNINDEXEDc                 S   s   g | ]}t |qS r*   r   r   r*   r*   r+   r   c  s     )r%   clean_optionsr   r   optionsZ_create_contextliteralri   extension_modulerH   r   r5   prefix_argumentsextendr   r   _hiddenr   r/   r   rW   r	   	argumentsZ_create_table_option_sqlr   )	r$   safer   r6   Z
ext_moduler   metarB   Z	field_defr*   r*   r+   _create_virtual_table>  s@    





z/VirtualTableSchemaManager._create_virtual_tablec                    s.   t | jtr| j|f|S tt| j|f|S r9   )
issubclassr%   VirtualModelr   r"   r   _create_table)r$   r   r   r(   r*   r+   r   i  s    
z'VirtualTableSchemaManager._create_table)T)T)r,   r-   r.   r   r   r0   r*   r*   r(   r+   r   =  s   
+r   c                   @   s&   e Zd ZG dd dZedd ZdS )r   c                   @   s    e Zd ZdZdZdZdZeZdS )zVirtualModel.MetaNF)	r,   r-   r.   r   r   r   primary_keyr   Zschema_manager_classr*   r*   r*   r+   Metar  s
   r   c                 C   s   |S r9   r*   )clsr   r*   r*   r+   r   y  s    zVirtualModel.clean_optionsN)r,   r-   r.   r   classmethodr   r*   r*   r*   r+   r   q  s   r   c                   @   s   e Zd Zedd ZdS )BaseFTSModelc                 C   s   | d}| d}| d}t|tr:|dkr:d|d< n t|trZt|jjj|j|d< |rt|t	t
frddd |D }d	|d
 |d< |r| jj dkrd| |d< |S )NcontentprefixtokenizerD   z'',c                 S   s   g | ]}t |qS r*   )strr   ir*   r*   r+   r     s     z.BaseFTSModel.clean_options.<locals>.<listcomp>z'%s'z' fts5"%s")getrH   
basestringr   r   r%   r   
table_namer/   rN   tuplerF   stripr   lower)r   r   r   r   r   r*   r*   r+   r     s     





zBaseFTSModel.clean_optionsN)r,   r-   r.   r   r   r*   r*   r*   r+   r   ~  s   r   c                   @   s   e Zd ZdZe ZG dd dZedd Zedd Z	edd	 Z
ed
d Zed*ddZed+ddZedd Zedd Zedd Zedd Zedd Zedd Zed,d"d#Zed-d$d%Zed.d&d'Zed/d(d)ZdS )0FTSModelz
    VirtualModel class for creating tables that use either the FTS3 or FTS4
    search extensions. Peewee automatically determines which version of the
    FTS extension is supported and will use FTS4 if possible.
    c                   @   s   e Zd Zde ZdS )zFTSModel.MetazFTS%sN)r,   r-   r.   FTS_VERSIONr   r*   r*   r*   r+   r     s   r   c                 C   s(   | j j}| j jd|||f }| S )Nz INSERT INTO %s(%s) VALUES('%s');)r   r   databaseZexecute_sqlZfetchone)r   cmdtblresr*   r*   r+   _fts_cmd  s
    zFTSModel._fts_cmdc                 C   s
   |  dS Noptimizer   r   r*   r*   r+   r     s    zFTSModel.optimizec                 C   s
   |  dS Nrebuildr   r   r*   r*   r+   r     s    zFTSModel.rebuildc                 C   s
   |  dS )Nintegrity-checkr   r   r*   r*   r+   integrity_check  s    zFTSModel.integrity_check      c                 C   s   |  d||f S )Nzmerge=%s,%sr   )r   blockssegmentsr*   r*   r+   merge  s    zFTSModel.mergeTc                 C   s   |  d|rdpd S )Nzautomerge=%s10r   )r   stater*   r*   r+   	automerge  s    zFTSModel.automergec                 C   s   t | jj|S zU
        Generate a `MATCH` expression appropriate for searching this table.
        r   r   r   r   r   r*   r*   r+   r     s    zFTSModel.matchc                 G   s    t | jjt}t j|f| S r9   )rP   	matchinfor   r   FTS3_MATCHINFOfts_rank)r   weightsr   r*   r*   r+   rank  s    zFTSModel.rankc                 G   s    t | jjt}t j|f| S r9   )rP   r   r   r   FTS4_MATCHINFOfts_bm25r   r   
match_infor*   r*   r+   bm25  s    zFTSModel.bm25c                 G   s    t | jjt}t j|f| S r9   )rP   r   r   r   r   	fts_bm25fr   r*   r*   r+   bm25f  s    zFTSModel.bm25fc                 G   s    t | jjt}t j|f| S r9   )rP   r   r   r   r   
fts_lucener   r*   r*   r+   lucene  s    zFTSModel.lucenec                 C   s   |s| }nNt |trRg }| jjD ]$}	||	||	jd}
||
 q"|| }n|| }d}|}|rt| ||f}|r|st|}| j	| 
| ||S )N      ?r*   )rH   rO   r   r   r   r&   rW   aliasr5   selectwherer   order_by)r   r   r   
with_scorescore_aliasZscore_fnexplicit_orderingr   weight_argsrB   Zfield_weight	selectionr  r*   r*   r+   _search  s,    

zFTSModel._searchNFscorec                 C   s   |  ||||| j|S 'Full-text search using selected `term`.)r  r   r   r   r   r	  r
  r  r*   r*   r+   search  s    zFTSModel.searchc                 C   s   |  ||||| j|S z:Full-text search for selected `term` using BM25 algorithm.)r  r   r  r*   r*   r+   search_bm25  s    zFTSModel.search_bm25c                 C   s   |  ||||| j|S r  )r  r  r  r*   r*   r+   search_bm25f  s    zFTSModel.search_bm25fc                 C   s   |  ||||| j|S r  )r  r  r  r*   r*   r+   search_lucene  s    zFTSModel.search_lucene)r   r   )T)NFr  F)NFr  F)NFr  F)NFr  F)r,   r-   r.   __doc__r1   r2   r   r   r   r   r   r   r   r   r   r   r   r  r  r  r  r  r  r  r*   r*   r*   r+   r     s\   









              r   abcdefghijklmnopqrstuvwxyzz	 ,"(){}*:_+0123456789   c                 c   s"   | ]}t |tkrt |V  qd S r9   )chr	_alphanumr   r*   r*   r+   	<genexpr>.  s      r     z(?:[^\s"]|"(?:\\.|[^"])*")+c                   @   s0  e Zd ZdZe ZG dd dZddddZedd	 Z	ed
d Z
edd ZeedfddZedd Zedd Zedd Zed6ddZed7ddZedd Zed d! Zed"d# Zed$d% Zed&d' Zed(d) Zed*d+ Zed,d- Zed.d/ Zed8d1d2Zed9d4d5ZdS ):	FTS5Modelan  
    Requires SQLite >= 3.9.0.

    Table options:

    content: table name of external content, or empty string for "contentless"
    content_rowid: column name of external content primary key
    prefix: integer(s). Ex: '2' or '2 3 4'
    tokenize: porter, unicode61, ascii. Ex: 'porter unicode61'

    The unicode tokenizer supports the following parameters:

    * remove_diacritics (1 or 0, default is 1)
    * tokenchars (string of characters, e.g. '-_'
    * separators (string of characters)

    Parameters are passed as alternating parameter name and value, so:

    {'tokenize': "unicode61 remove_diacritics 0 tokenchars '-_'"}

    Content-less tables:

    If you don't need the full-text content in it's original form, you can
    specify a content-less table. Searches and auxiliary functions will work
    as usual, but the only values returned when SELECT-ing can be rowid. Also
    content-less tables do not support UPDATE or DELETE.

    External content tables:

    You can set up triggers to sync these, e.g.

    -- Create a table. And an external content fts5 table to index it.
    CREATE TABLE tbl(a INTEGER PRIMARY KEY, b);
    CREATE VIRTUAL TABLE ft USING fts5(b, content='tbl', content_rowid='a');

    -- Triggers to keep the FTS index up to date.
    CREATE TRIGGER tbl_ai AFTER INSERT ON tbl BEGIN
      INSERT INTO ft(rowid, b) VALUES (new.a, new.b);
    END;
    CREATE TRIGGER tbl_ad AFTER DELETE ON tbl BEGIN
      INSERT INTO ft(fts_idx, rowid, b) VALUES('delete', old.a, old.b);
    END;
    CREATE TRIGGER tbl_au AFTER UPDATE ON tbl BEGIN
      INSERT INTO ft(fts_idx, rowid, b) VALUES('delete', old.a, old.b);
      INSERT INTO ft(rowid, b) VALUES (new.a, new.b);
    END;

    Built-in auxiliary functions:

    * bm25(tbl[, weight_0, ... weight_n])
    * highlight(tbl, col_idx, prefix, suffix)
    * snippet(tbl, col_idx, prefix, suffix, ?, max_tokens)
    c                   @   s   e Zd ZdZdS )zFTS5Model.Metar   Nr,   r-   r.   r   r*   r*   r*   r+   r   n  s   r   zQBesides the implicit `rowid` column, all columns must be instances of SearchFieldz3Secondary indexes are not supported for FTS5 modelsz4FTS5 models must use the default `rowid` primary key)r=   r   pkc                 C   sd   | j jjdkrt| jd | j j D ] }t|tt	fs(t| jd q(| j j
r`t| jd d S )Nr   r!  r=   r   )r   r   r&   ImproperlyConfigured_error_messagesfieldsvaluesrH   r   r   Zindexes)r   rB   r*   r*   r+   validate_modelx  s    zFTS5Model.validate_modelc              	   C   s   t jd d tk rdS t d}z\z|d W nH   z|d |d W n   Y Y W dS X | jj	d Y nX W 5 |  X dS )Nr   Fz:memory:z0CREATE VIRTUAL TABLE fts5test USING fts5 (data);Tr   )
r   sqlite_version_infoFTS5_MIN_SQLITE_VERSIONconnectcloseexecuteZenable_load_extensionload_extensionr   r   )r   Ztmp_dbr*   r*   r+   fts5_installed  s    


zFTS5Model.fts5_installedc                 C   s@   t | }|D ],}|dr(|dr(qt|t@ r dS qdS )z
        Simple helper function to indicate whether a search query is a
        valid FTS5 query. Note: this simply looks at the characters being
        used, and is not guaranteed to catch all problematic queries.
        "FT)	_quote_refindall
startswithendswithr\   _invalid_ascii)querytokenstokenr*   r*   r+   validate_query  s    
zFTS5Model.validate_queryr  c           	      C   s   g }d}t | }|D ]\}|dr:|dr:|| qt|}|t@ }|rhd}|D ]}|||}qV|| q|rd|S | S )z2
        Clean a query of invalid tokens.
        Fr.  Tr   )	r/  r0  r1  r2  rW   r\   r3  r]   rF   )	r4  r]   accumZany_invalidr5  r6  Z	token_setZinvalid_for_tokencr*   r*   r+   clean_query  s"    


zFTS5Model.clean_queryc                 C   s   t | jj|S r   r   r   r*   r*   r+   r     s    zFTS5Model.matchc                 G   s   |r| j | S tdS )Nr   )r   r5   )r   r'   r*   r*   r+   r     s    zFTS5Model.rankc                 G   s   t j| jjf| S r9   )rP   r   r   r   )r   r   r*   r*   r+   r     s    zFTS5Model.bm25NFr  c                 C   s   |  t|||||S r  )r  r  r:  r  r*   r*   r+   r    s    zFTS5Model.searchc              
   C   s   |st d}nrt|trlg }| jjD ]0}t|tr$|js$|||||j	d q$t
j| jjf| }nt
j| jjf| }d}	|}
|r| ||f}	|r|st |}
| j|	 | t||
S )r  r   r  r*   )r5   rH   rO   r   r   r   r   rW   r   r&   rP   r   r   r  r  r  r   r  r:  r  )r   r   r   r	  r
  r  r   r  rB   r  r  r*   r*   r+   r    s0    

zFTS5Model.search_bm25c                 K   sd   | j j}|g}|g}| D ] \}}|t| || qttd| j jt|tdt|fS )NzINSERT INTOZVALUES)r   r   itemsrW   r   r	   r5   r   )r   r   extra_paramsr   columnsr%  keyrU   r*   r*   r+   _fts_cmd_sql  s    zFTS5Model._fts_cmd_sqlc                 K   s   | j |f|}| jj|S r9   )r?  r   r   r+  )r   r   r<  r4  r*   r*   r+   r     s    zFTS5Model._fts_cmdc                 C   s,   d|  krdksn t d| jd|dS )Nr   r   zlevel must be between 0 and 16r   r   )r    r   )r   levelr*   r*   r+   r     s    zFTS5Model.automergec                 C   s   | j d|dS )Nr   r@  r   )r   Znpagesr*   r*   r+   r     s    zFTS5Model.mergec                 C   s
   |  dS r   r   r   r*   r*   r+   r     s    zFTS5Model.optimizec                 C   s
   |  dS r   r   r   r*   r*   r+   r     s    zFTS5Model.rebuildc                 C   s   | j d|dS )Npgszr@  r   )r   rB  r*   r*   r+   set_pgsz  s    zFTS5Model.set_pgszc                 C   s   | j d|dS )Nr   r@  r   )r   Zrank_expressionr*   r*   r+   set_rank!  s    zFTS5Model.set_rankc                 C   s
   |  dS )Nz
delete-allr   r   r*   r*   r+   
delete_all%  s    zFTS5Model.delete_allr   c                 C   s   | j d|dS )Nr   r@  r   )r   r   r*   r*   r+   r   )  s    zFTS5Model.integrity_checkrowc                    s   dkrt dd }t |sG  fddd}ttt t t |d}dkrhtt|d< ndkr|tt|d	< d
 j }t |t|t	f| t
 |S )N)rF  colinstancez5table_type must be either "row", "col" or "instance".z_vocab_model_%sc                       s8   e Zd Z jjZp jjd Ze jje	Z
dS )z"FTS5Model.VocabModel.<locals>.MetaZ_vN)r,   r-   r.   r   r   r   rP   Z	fts5vocabr   r5   r   r*   r   table
table_typer*   r+   r   6  s   r   )r   docZcntr   r   rG  rH  offsetz%sVocab)r    r   r   	TextFieldIntegerFieldr   r,   setattrr!   r   getattr)r   rK  rJ  attrr   attrs
class_namer*   rI  r+   
VocabModel-  s$    

zFTS5Model.VocabModel)NFr  F)NFr  F)r   )rF  N)r,   r-   r.   r  r   r   r   r#  r   r&  r-  staticmethodr7  r  r:  r   r   r   r  r  r?  r   r   r   r   r   rC  rD  rE  r   rU  r*   r*   r*   r+   r  4  sh   7






  
    








r  c                    s   dkr dkr<j jD ]}|jkr|  q<qtdj jdkrPG  fdddt}G  fddd}dj }t||fd|iS )z3Model factory for the transitive closure extension.Nz,Unable to find self-referential foreign key.c                       s   e Zd ZeeZeeZeeZeeZ	eeZ
eeZG dd dZedfdd	Zedfdd	Zed fd	d
	ZdS )z&ClosureTable.<locals>.BaseClosureTablec                   @   s   e Zd ZdZdS )z+ClosureTable.<locals>.BaseClosureTable.MetaZtransitive_closureNr   r*   r*   r*   r+   r   i  s   r   NFc                    sf      | jdj| | jkd| j|k }|d k	rN|| j|k}n|sb|| jdk}|S Ndepthonr   )r  rX  r  rF   idr  rootobjectsr   noderX  include_noder4  model_class
source_keyr*   r+   descendantsl  s     
 z2ClosureTable.<locals>.BaseClosureTable.descendantsc                    sb      | jdj| | jkd| j|k }|rJ|| j|k}n|s^|| jdk}|S rW  )r  rX  r  rF   r\  r  r[  r]  r^  ra  r*   r+   	ancestorsy  s     
 z0ClosureTable.<locals>.BaseClosureTable.ancestorsc                    s   kr*|j  j}  |k}nFj|  | jkd| j|k| jdk@ } |> 	 }|s||k}|S )NrY     )
Z__data__r   r&   r  r  rF   r\  r[  rX  r]  )r   r_  r`  Zfk_valuer4  siblingsforeign_keyrb  referencing_classreferencing_keyrc  r*   r+   rg    s"     z/ClosureTable.<locals>.BaseClosureTable.siblings)NF)NF)F)r,   r-   r.   r   rO  rX  r[  rN  idcolumnparentcolumnr\  	tablenamer   r   rd  re  rg  r*   rh  r*   r+   BaseClosureTablea  s   ro  c                       s,   e Zd ZjjZjjj jdZdZdS )zClosureTable.<locals>.Meta)rn  rl  rm  FN)	r,   r-   r.   r   r   r   r/   r   r   r*   )ri  rj  rk  r*   r+   r     s   r   z	%sClosure)r   refsZ	rel_modelr    r   r   r,   r!   )rb  ri  rj  rk  Z	field_objro  r   r&   r*   rh  r+   ClosureTableO  s    
=
rq  c                   @   sp   e Zd ZG dd dZedd ZedddZedd	 Zed
d Z	edd Z
edd Zedd ZdS )LSMTablec                   @   s   e Zd ZdZdZdS )zLSMTable.MetaZlsm1N)r,   r-   r.   r   filenamer*   r*   r*   r+   r     s   r   c                 C   s   | j j}|stdn t|dkr6|d dkr6d| }| j jsFtd| j j}t|tr`tdt|ttt	fsxtdd	|_
t|t	rd
}nt|trd}nd}|d|j |g| j _t| j jdkr| j jd | j _nd | j _|S )NzILSM1 extension requires that you specify a filename for the LSM database.   r   r.  r   z-LSM1 models must specify a primary-key field.z8LSM1 models must explicitly declare a primary key field.z9LSM1 key must be a TextField, BlobField, or IntegerField.TZUINTZBLOBr8   rf  )r   rs  r    lenr   rH   	AutoFieldrN  Z	BlobFieldrO  r   r&   r   r   _value_field)r   r   rs  r>  Z	data_typer*   r*   r+   r     s.    



zLSMTable.clean_optionslsm.soc                 C   s   | j j| d S r9   )r   r   r,  )r   rC   r*   r*   r+   r,    s    zLSMTable.load_extensionc                 C   sP   |j d k	r$|jd k	r$| |j |jS |j d k	r8| |j kS |jd k	rL| |jkS d S r9   )startstopZbetween)r>  rJ   r*   r*   r+   slice_to_expr  s    


zLSMTable.slice_to_exprc                 C   s^   t |tr0t||}|d k	r(| |} | dfS t |trH| |dfS | ||kdfS d S NFT)rH   slicerr  r{  r  r   )r4  r>  lookupexprr*   r*   r+   _apply_lookup_to_query  s    


zLSMTable._apply_lookup_to_queryc                 C   sJ   |  |   | jj|\}}|rB| }| jjd k	r>|d S |S |S d S Nrf  )r  r  Znamedtuplesr   r   r   rw  )r   r!  r4  	is_singlerF  r*   r*   r+   	get_by_id  s    
zLSMTable.get_by_idc                 C   s   | j jd k	r| j j|i}nXt|trRi }t| j jdd  |D ]\}}|||< q>n t|trb|}nt|| rr|j}||| j j< | 	|
  d S r  )r   rw  rH   r   zipr   rO   __dict__r   r]   r+  )r   r>  rU   r   rB   Zfvalr*   r*   r+   	set_by_id  s    


zLSMTable.set_by_idc                 C   s"   |  |  | jj|\}}| S r9   )r  deleter   r   r+  )r   r!  r4  r  r*   r*   r+   delete_by_id  s    zLSMTable.delete_by_idN)rx  )r,   r-   r.   r   r   r   r,  rV  r{  r  r  r  r  r*   r*   r*   r+   rr    s   
#



rr  MATCHc                 C   s   t | |d k	S r9   )rer  )regexrU   r*   r*   r+   _sqlite_regexp  s    r  c                       s2   e Zd Zd
 fdd	Z fddZdd	 Z  ZS )SqliteExtDatabaseNTFc                    s   t t| j|f||	 d | _|r.ts.tdto8|dk	}
|r||
rLt|  n0| td | t	d | td | td |r|
st
dt|  |r| tdd	 |r|
st
d
t|  |r| td |
| _d S )NzRSqliteExtDatabase initialized with C extensions, but shared library was not found!Fr   r   r   r  z0C extension required to register hash functions.regexprt  z(C extension required to use bloomfilter.json_contains)r"   r  r?   _row_factoryCYTHON_SQLITE_EXTENSIONSr"  r   Zregister_functionr   r   r    r   r  r   _json_containsZ_c_extensions)r$   r   Zc_extensionsZrank_functionsZhash_functionsZregexp_functionZbloomfilterr  r'   rw   Zprefer_cr(   r*   r+   r?     s2    
zSqliteExtDatabase.__init__c                    s"   t t| | | jr| j|_d S r9   )r"   r  _add_conn_hooksr  row_factoryr$   connr(   r*   r+   r  ;  s    z!SqliteExtDatabase._add_conn_hooksc                 C   s
   || _ d S r9   )r  r$   rP   r*   r*   r+   r  @  s    zSqliteExtDatabase.row_factory)NTFFFF)r,   r-   r.   r?   r  r  r0   r*   r*   r(   r+   r    s          #r  rf  rt        r   r   r   
   c                    s    fdd}t |S )zt
        Expose a sqlite3_status() call for a particular flag as a property of
        the Database object.
        c                    s   t  }r|d S |S r  )r   r$   resultflagreturn_highwaterr*   r+   gettera  s    z__status__.<locals>.getterrm   )r  r  r  r*   r  r+   
__status__\  s    r  c                    s    fdd}t |S )z
        Expose a sqlite3_dbstatus() call for a particular flag as a property of
        the Database instance. Unlike sqlite3_status(), the dbstatus properties
        pertain to the current connection.
        c                    s>   | j jd krtdt| j j }r.|d S r:|d S |S )Nzdatabase connection not opened.r   rf  )_stater  r"  r   r  r  return_currentr  r*   r+   r  l  s    z__dbstatus__.<locals>.getterr  )r  r  r  r  r*   r  r+   __dbstatus__f  s    r  c                       sT  e Zd Z fddZd fdd	Z fddZ fdd	Zd
d Zdd Zdd Z	dd Z
edd Zedd ZdddZd ddZd!ddZeeZeedZeeZeeZeeZeedZeeZee Z!ee"dZ#e$e%Z&e$e'dZ(e$e)dZ*e$e+dZ,e$e-ddZ.e$e/ddZ0e$e1ddZ2e$e3ddZ4e$e5ddZ6e$e7ddZ8  Z9S )"CSqliteExtDatabasec                    s4   d | _ d  | _ | _| _d| _tt| j|| d S )NF)_conn_helper_commit_hook_rollback_hook_update_hook_replace_busy_handlerr"   r  r?   )r$   r'   rw   r(   r*   r+   r?   v  s    zCSqliteExtDatabase.__init__Fc                    s   t t| j|f| || _d S r9   )r"   r  initr  )r$   r   Zreplace_busy_handlerrw   r(   r*   r+   r  |  s    zCSqliteExtDatabase.initc                    sF   | j r| jd  | jr$| jd  | jr6| jd  tt| 	|S r9   )
r  r  set_commit_hookr  set_rollback_hookr  set_update_hookr"   r  _closer  r(   r*   r+   r    s    zCSqliteExtDatabase._closec                    s   t t| | t|| _| jd k	r2| j| j | jd k	rJ| j| j | j	d k	rb| j
| j	 | jr| jppd}| j|d  d S )Nr  i  )r"   r  r  r   r  r  r  r  r  r  r  r  _timeoutZset_busy_handler)r$   r  timeoutr(   r*   r+   r    s    




z"CSqliteExtDatabase._add_conn_hooksc                 C   s   || _ |  s| j| |S r9   )r  	is_closedr  r  r  r*   r*   r+   	on_commit  s    zCSqliteExtDatabase.on_commitc                 C   s   || _ |  s| j| |S r9   )r  r  r  r  r  r*   r*   r+   on_rollback  s    zCSqliteExtDatabase.on_rollbackc                 C   s   || _ |  s| j| |S r9   )r  r  r  r  r  r*   r*   r+   	on_update  s    zCSqliteExtDatabase.on_updatec                 C   s
   | j  S r9   )r  changesr:   r*   r*   r+   r    s    zCSqliteExtDatabase.changesc                 C   s
   | j  S r9   )r  last_insert_rowidr:   r*   r*   r+   r    s    z$CSqliteExtDatabase.last_insert_rowidc                 C   s
   | j  S r9   )r  
autocommitr:   r*   r*   r+   r    s    zCSqliteExtDatabase.autocommitNc                 C   s   t |  | |||dS N)pagesr&   progress)r   
connection)r$   Zdestinationr  r&   r  r*   r*   r+   r     s
      zCSqliteExtDatabase.backupc                 C   s   t |  ||||dS r  )r   r  )r$   rs  r  r&   r  r*   r*   r+   r     s     z!CSqliteExtDatabase.backup_to_filec                 C   s   t | ||||S r9   )r   )r$   rJ  columnr   Z	read_onlyr*   r*   r+   	blob_open  s    zCSqliteExtDatabase.blob_openT)F)NNN)NNN)F):r,   r-   r.   r?   r  r  r  r  r  r  r  rm   r  r  r   r   r  r  SQLITE_STATUS_MEMORY_USEDZmemory_usedSQLITE_STATUS_MALLOC_SIZEZmalloc_sizeSQLITE_STATUS_MALLOC_COUNTZmalloc_countSQLITE_STATUS_PAGECACHE_USEDZpagecache_used SQLITE_STATUS_PAGECACHE_OVERFLOWZpagecache_overflowSQLITE_STATUS_PAGECACHE_SIZEZpagecache_sizeSQLITE_STATUS_SCRATCH_USEDZscratch_usedSQLITE_STATUS_SCRATCH_OVERFLOWZscratch_overflowSQLITE_STATUS_SCRATCH_SIZEZscratch_sizer  SQLITE_DBSTATUS_LOOKASIDE_USEDZlookaside_usedSQLITE_DBSTATUS_LOOKASIDE_HITZlookaside_hit#SQLITE_DBSTATUS_LOOKASIDE_MISS_SIZEZlookaside_miss#SQLITE_DBSTATUS_LOOKASIDE_MISS_FULLZlookaside_miss_fullSQLITE_DBSTATUS_CACHE_USEDZ
cache_usedSQLITE_DBSTATUS_SCHEMA_USEDZschema_usedSQLITE_DBSTATUS_STMT_USEDZstatement_usedSQLITE_DBSTATUS_CACHE_HITZ	cache_hitSQLITE_DBSTATUS_CACHE_MISSZ
cache_missSQLITE_DBSTATUS_CACHE_WRITEZcache_writer0   r*   r*   r(   r+   r  u  sP   	


  





r  c                 C   s   t | tj|S r9   )r   r
   r  )lhsr}   r*   r*   r+   r     s    r   c                    s"   t  } fddtd|dD S )Nc              	      s(   g | ] }t d  ||d  d qS )z@Ir   r   )structr|   r   bufr*   r+   r     s     z%_parse_match_info.<locals>.<listcomp>r   r   )ru  range)r  bufsizer*   r  r+   _parse_match_info  s    r  c                 C   s6   |sdg|  S dg|  }t |D ]\}}|||< q |S )Nrf  r   )	enumerate)Zncolraw_weightsr   r   weightr*   r*   r+   get_weights  s    


r  c                 G   s   t | }d}|d d \}}t||}t|D ]l}d|| d  }t|D ]N}	||	 }
|
s\qJ||	d  }|| }||d  }|dkrJ||
t||  7 }qJq.| S )N        rt  r   rf  r   )r  r  r  float)raw_match_infor  r   r  r   r9  r   Z
phrase_numZphrase_info_idxZcol_numr  Zcol_idxZrow_hitsZall_rows_hitsr*   r*   r+   r     s     
r   c                 G   sL  t | }d}d}d}td\}}}}	|| }
|| }|| }|	| }|| }t||}t|
D ]}t|D ]}|| }|dkrqj|d|||    }t|| }t||d  }t|| d |d  }|dkrd	}t|||  }t||	|  pd
}|| }||d
  }d
| ||  }|||  }|||  }||| 7 }qjq^| S )z
    Usage:

        # Format string *must* be pcnalx
        # Second parameter to bm25 specifies the index of the column, on
        # the table being queries.
        bm25(matchinfo(document_tbl, 'pcnalx'), 1) AS rank
    g333333?g      ?r  r   r   r   rt  g      ?gư>r  )r  r  r  r  mathlog)r  r'   r   KBr  ZP_OZC_OZN_OZA_OZ
term_countZ	col_countZ
total_docsZL_OZX_Or   r   jr  xZterm_frequencyZdocs_with_termZidfZ
doc_lengthZ
avg_lengthZrationumZb_partZdenomZpc_scorer*   r*   r+   r     sF    	

r   c                 C   sJ  g }z| t|t| f W n   Y dS X |rF| \}}t|trt|tr|D ](}||krn dS | || || f q\n0t|tr|D ]}||kr dS qn||krdS q2t|tr8t|trdS t|tr(z,tt|D ]}| || || f qW n t	k
r$   Y dS X n||krDdS q2||kr2dS q2dS r|  )
rW   rQ   ru   poprH   rO   rN   r  ru  
IndexError)Zsrc_jsonZobj_jsonstackobjsrcr>  rK   r   r*   r*   r+   r  b  sB    






r  )NNN)F)FF)hrQ   r  r  r  sysZpeeweer   r   r   r   r   r   r	   r
   r   r   r   Zplayhouse._sqlite_extr   r   r   r   r   r   r   r   r   r   r   r  ImportErrorversion_infor   r   r   r   r'  r   r(  rv  r   r1   r3   ZDecimalFieldr7   r>   rn   rN  rs   r   r   r   ZSchemaManagerr   ZModelr   r   r   Z	_alphabetr\   upperr  r  r  r3  compiler/  r  rq  rr  r  r  ZSqliteDatabaser  r  r  r  r  r  r  ZSQLITE_STATUS_PARSER_STACKr  r  r  r  r  r  r  r  r  r  r  r  r  ZSQLITE_DBSTATUS_DEFERRED_FKSr  r  r  r   r  r  r   r   r  r*   r*   r*   r+   <module>   s   4

:`4 

    
[g-


f
$Q