11
11
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12
12
# License for the specific language governing permissions and limitations
13
13
# under the License.
14
+ from dataclasses import asdict
15
+ from enum import Enum
16
+ from http import HTTPStatus
17
+ from typing import List , Optional , TypeVar , Any
18
+
19
+ from urllib .parse import quote
20
+
21
+ from httpx import Response
22
+
14
23
from centraldogma .base_client import BaseClient
15
24
from centraldogma .data .change import Change
16
25
from centraldogma .data .commit import Commit
17
26
from centraldogma .data .content import Content
27
+ from centraldogma .data .entry import Entry , EntryType
18
28
from centraldogma .data .push_result import PushResult
19
- from dataclasses import asdict
20
- from enum import Enum
21
- from http import HTTPStatus
22
- from typing import List , Optional
29
+ from centraldogma .data .revision import Revision
30
+ from centraldogma .exceptions import CentralDogmaException
31
+ from centraldogma .query import Query , QueryType
32
+
33
+ T = TypeVar ('T' )
23
34
24
35
25
36
class ContentService :
26
37
def __init__ (self , client : BaseClient ):
27
38
self .client = client
28
39
29
40
def get_files (
30
- self ,
31
- project_name : str ,
32
- repo_name : str ,
33
- path_pattern : Optional [str ],
34
- revision : Optional [int ],
35
- include_content : bool = False ,
41
+ self ,
42
+ project_name : str ,
43
+ repo_name : str ,
44
+ path_pattern : Optional [str ],
45
+ revision : Optional [int ],
46
+ include_content : bool = False ,
36
47
) -> List [Content ]:
37
48
params = {"revision" : revision } if revision else None
38
49
path = f"/projects/{ project_name } /repos/{ repo_name } /"
@@ -67,11 +78,11 @@ def get_file(
67
78
return Content .from_dict (resp .json ())
68
79
69
80
def push (
70
- self ,
71
- project_name : str ,
72
- repo_name : str ,
73
- commit : Commit ,
74
- changes : List [Change ],
81
+ self ,
82
+ project_name : str ,
83
+ repo_name : str ,
84
+ commit : Commit ,
85
+ changes : List [Change ],
75
86
) -> PushResult :
76
87
params = {
77
88
"commitMessage" : asdict (commit ),
@@ -81,7 +92,79 @@ def push(
81
92
}
82
93
path = f"/projects/{ project_name } /repos/{ repo_name } /contents"
83
94
resp = self .client .request ("post" , path , json = params )
84
- return PushResult .from_dict (resp .json ())
95
+ json : object = resp .json ()
96
+ return PushResult .from_dict (json )
97
+
98
+ def watch_repository (self , project_name : str , repo_name : str , last_known_revision : Revision , path_pattern : str ,
99
+ timeout_millis : int ) -> Optional [Revision ]:
100
+ path = f"/projects/{ project_name } /repos/{ repo_name } /contents"
101
+ if path_pattern [0 ] != "/" :
102
+ path += "/**/"
103
+
104
+ if path_pattern in ' ' :
105
+ path_pattern = path_pattern .replace (" " , "%20" )
106
+ path += path_pattern
107
+
108
+ response = self ._watch (last_known_revision , timeout_millis , path )
109
+ if response .status_code == HTTPStatus .OK :
110
+ json = response .json ()
111
+ return Revision (json ["revision" ])
112
+ elif response .status_code == HTTPStatus .NOT_MODIFIED :
113
+ return None
114
+ else :
115
+ # TODO(ikhoon): Handle excepitons after https://github.com/line/centraldogma-python/pull/11/ is merged.
116
+ pass
117
+
118
+ def watch_file (self , project_name : str , repo_name : str , last_known_revision : Revision , query : Query [T ],
119
+ timeout_millis ) -> Optional [Entry [T ]]:
120
+ path = f"/projects/{ project_name } /repos/{ repo_name } /contents/{ query .path } "
121
+ if query .query_type == QueryType .JSON_PATH :
122
+ queries = [f"jsonpath={ quote (expr )} " for expr in query .expressions ]
123
+ path = f"{ path } ?{ '&' .join (queries )} "
124
+
125
+ response = self ._watch (last_known_revision , timeout_millis , path )
126
+ if response .status_code == HTTPStatus .OK :
127
+ json = response .json ()
128
+ revision = Revision (json ["revision" ])
129
+ return self ._to_entry (revision , json ["entry" ], query .query_type )
130
+ elif response .status_code == HTTPStatus .NOT_MODIFIED :
131
+ return None
132
+ else :
133
+ # TODO(ikhoon): Handle excepitons after https://github.com/line/centraldogma-python/pull/11/ is merged.
134
+ pass
135
+
136
+ @staticmethod
137
+ def _to_entry (revision : Revision , json : Any , query_type : QueryType ) -> Entry :
138
+ entry_path = json ["path" ]
139
+ received_entry_type = EntryType [json ["type" ]]
140
+ content = json ["content" ]
141
+ if query_type == QueryType .IDENTITY_TEXT :
142
+ return Entry .text (revision , entry_path , content )
143
+ elif query_type == QueryType .IDENTITY or query_type == QueryType .JSON_PATH :
144
+ if received_entry_type != EntryType .JSON :
145
+ raise CentralDogmaException (
146
+ f"invalid entry type. entry type: { received_entry_type } (expected: { query_type } )" )
147
+
148
+ return Entry .json (revision , entry_path , content )
149
+ else : # query_type == QueryType.IDENTITY
150
+ if received_entry_type == EntryType .JSON :
151
+ return Entry .json (revision , entry_path , content )
152
+ elif received_entry_type == EntryType .TEXT :
153
+ return Entry .text (revision , entry_path , content )
154
+ else : # received_entry_type == EntryType.DIRECTORY
155
+ return Entry .directory (revision , entry_path )
156
+
157
+ def _watch (
158
+ self ,
159
+ last_known_revision : Revision ,
160
+ timeout_millis : int ,
161
+ path : str ) -> Response :
162
+ normalized_timeout = (timeout_millis + 999 ) // 1000
163
+ headers = {
164
+ "if-none-match" : f"{ last_known_revision .major } " ,
165
+ "prefer" : f"wait={ normalized_timeout } "
166
+ }
167
+ return self .client .request ("get" , path , headers = headers , timeout = normalized_timeout )
85
168
86
169
def _change_dict (self , data ):
87
170
return {
0 commit comments