kaggle 1.7.4.5__py3-none-any.whl → 1.8.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- kaggle/__init__.py +10 -6
- kaggle/api/kaggle_api.py +574 -598
- kaggle/api/kaggle_api_extended.py +5251 -4769
- kaggle/cli.py +1335 -1585
- kaggle/models/api_blob_type.py +3 -3
- kaggle/models/dataset_column.py +165 -174
- kaggle/models/dataset_new_request.py +83 -41
- kaggle/models/dataset_new_version_request.py +32 -25
- kaggle/models/dataset_update_settings_request.py +35 -27
- kaggle/models/kaggle_models_extended.py +169 -172
- kaggle/models/kernel_push_request.py +66 -49
- kaggle/models/model_instance_new_version_request.py +10 -18
- kaggle/models/model_instance_update_request.py +103 -34
- kaggle/models/model_new_instance_request.py +138 -41
- kaggle/models/model_new_request.py +35 -27
- kaggle/models/model_update_request.py +32 -25
- kaggle/models/start_blob_upload_request.py +192 -195
- kaggle/models/start_blob_upload_response.py +98 -98
- kaggle/models/upload_file.py +114 -120
- kaggle/test/test_authenticate.py +23 -23
- {kaggle-1.7.4.5.dist-info → kaggle-1.8.0.dist-info}/METADATA +11 -15
- kaggle-1.8.0.dist-info/RECORD +148 -0
- kagglesdk/__init__.py +5 -1
- kagglesdk/benchmarks/services/__init__.py +0 -0
- kagglesdk/benchmarks/services/benchmarks_api_service.py +19 -0
- kagglesdk/benchmarks/types/__init__.py +0 -0
- kagglesdk/benchmarks/types/benchmark_types.py +307 -0
- kagglesdk/benchmarks/types/benchmarks_api_service.py +243 -0
- kagglesdk/blobs/services/blob_api_service.py +1 -1
- kagglesdk/blobs/types/blob_api_service.py +2 -2
- kagglesdk/common/services/__init__.py +0 -0
- kagglesdk/common/services/operations_service.py +46 -0
- kagglesdk/common/types/file_download.py +1 -1
- kagglesdk/common/types/http_redirect.py +1 -1
- kagglesdk/common/types/operations.py +194 -0
- kagglesdk/common/types/operations_service.py +48 -0
- kagglesdk/community/__init__.py +0 -0
- kagglesdk/community/types/__init__.py +0 -0
- kagglesdk/community/types/content_enums.py +44 -0
- kagglesdk/community/types/organization.py +410 -0
- kagglesdk/competitions/services/competition_api_service.py +49 -12
- kagglesdk/competitions/types/competition.py +14 -0
- kagglesdk/competitions/types/competition_api_service.py +1639 -1275
- kagglesdk/competitions/types/search_competitions.py +28 -0
- kagglesdk/datasets/databundles/__init__.py +0 -0
- kagglesdk/datasets/databundles/types/__init__.py +0 -0
- kagglesdk/datasets/databundles/types/databundle_api_types.py +540 -0
- kagglesdk/datasets/services/dataset_api_service.py +39 -14
- kagglesdk/datasets/types/dataset_api_service.py +554 -300
- kagglesdk/datasets/types/dataset_enums.py +21 -0
- kagglesdk/datasets/types/dataset_service.py +145 -0
- kagglesdk/datasets/types/dataset_types.py +74 -74
- kagglesdk/datasets/types/search_datasets.py +6 -0
- kagglesdk/discussions/__init__.py +0 -0
- kagglesdk/discussions/types/__init__.py +0 -0
- kagglesdk/discussions/types/search_discussions.py +43 -0
- kagglesdk/discussions/types/writeup_enums.py +11 -0
- kagglesdk/education/services/education_api_service.py +1 -1
- kagglesdk/education/types/education_api_service.py +1 -1
- kagglesdk/kaggle_client.py +46 -23
- kagglesdk/kaggle_creds.py +148 -0
- kagglesdk/kaggle_env.py +89 -25
- kagglesdk/kaggle_http_client.py +216 -306
- kagglesdk/kaggle_oauth.py +200 -0
- kagglesdk/kaggle_object.py +286 -293
- kagglesdk/kernels/services/kernels_api_service.py +46 -9
- kagglesdk/kernels/types/kernels_api_service.py +635 -159
- kagglesdk/kernels/types/kernels_enums.py +6 -0
- kagglesdk/kernels/types/search_kernels.py +6 -0
- kagglesdk/licenses/__init__.py +0 -0
- kagglesdk/licenses/types/__init__.py +0 -0
- kagglesdk/licenses/types/licenses_types.py +182 -0
- kagglesdk/models/services/model_api_service.py +41 -17
- kagglesdk/models/types/model_api_service.py +987 -637
- kagglesdk/models/types/model_enums.py +8 -0
- kagglesdk/models/types/model_service.py +71 -71
- kagglesdk/models/types/model_types.py +1057 -5
- kagglesdk/models/types/search_models.py +8 -0
- kagglesdk/search/__init__.py +0 -0
- kagglesdk/search/services/__init__.py +0 -0
- kagglesdk/search/services/search_api_service.py +19 -0
- kagglesdk/search/types/__init__.py +0 -0
- kagglesdk/search/types/search_api_service.py +2435 -0
- kagglesdk/search/types/search_content_shared.py +50 -0
- kagglesdk/search/types/search_enums.py +45 -0
- kagglesdk/search/types/search_service.py +303 -0
- kagglesdk/security/services/iam_service.py +31 -0
- kagglesdk/security/services/oauth_service.py +27 -1
- kagglesdk/security/types/authentication.py +63 -63
- kagglesdk/security/types/iam_service.py +496 -0
- kagglesdk/security/types/oauth_service.py +797 -10
- kagglesdk/security/types/roles.py +8 -0
- kagglesdk/security/types/security_types.py +159 -0
- kagglesdk/test/__init__.py +0 -0
- kagglesdk/test/test_client.py +20 -22
- kagglesdk/users/services/account_service.py +13 -1
- kagglesdk/users/services/group_api_service.py +31 -0
- kagglesdk/users/types/account_service.py +169 -28
- kagglesdk/users/types/group_api_service.py +315 -0
- kagglesdk/users/types/group_types.py +165 -0
- kagglesdk/users/types/groups_enum.py +8 -0
- kagglesdk/users/types/progression_service.py +9 -0
- kagglesdk/users/types/search_users.py +23 -0
- kagglesdk/users/types/user_avatar.py +226 -0
- kaggle/configuration.py +0 -206
- kaggle-1.7.4.5.dist-info/RECORD +0 -98
- {kaggle-1.7.4.5.dist-info → kaggle-1.8.0.dist-info}/WHEEL +0 -0
- {kaggle-1.7.4.5.dist-info → kaggle-1.8.0.dist-info}/entry_points.txt +0 -0
- {kaggle-1.7.4.5.dist-info → kaggle-1.8.0.dist-info}/licenses/LICENSE.txt +0 -0
- {kaggle/test → kagglesdk/benchmarks}/__init__.py +0 -0
kagglesdk/kaggle_object.py
CHANGED
|
@@ -6,346 +6,339 @@ from google.protobuf.field_mask_pb2 import FieldMask
|
|
|
6
6
|
|
|
7
7
|
class ObjectSerializer(object):
|
|
8
8
|
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
9
|
+
def __init__(self, to_dict_value, from_dict_value):
|
|
10
|
+
self.to_dict_value = to_dict_value
|
|
11
|
+
self.from_dict_value = from_dict_value
|
|
12
12
|
|
|
13
13
|
|
|
14
14
|
class PredefinedSerializer(ObjectSerializer):
|
|
15
15
|
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
16
|
+
def __init__(self):
|
|
17
|
+
"""Predefined objects such as int, float etc are serialized/deserialized directly."""
|
|
18
|
+
ObjectSerializer.__init__(self, lambda cls, v, _: v, lambda cls, v: v)
|
|
19
19
|
|
|
20
20
|
|
|
21
21
|
# Adapted from https://stackoverflow.com/questions/1175208/elegant-python-function-to-convert-camelcase-to-snake-case
|
|
22
|
-
_pascal_to_upper_snake_case_regex = re.compile(
|
|
23
|
-
'((?<=[a-z0-9])[A-Z]|(?!^)[A-Z](?=[a-z]))'
|
|
24
|
-
)
|
|
22
|
+
_pascal_to_upper_snake_case_regex = re.compile("((?<=[a-z0-9])[A-Z]|(?!^)[A-Z](?=[a-z]))")
|
|
25
23
|
|
|
26
24
|
|
|
27
25
|
def _pascal_case_to_upper_snake_case(string):
|
|
28
|
-
|
|
26
|
+
return _pascal_to_upper_snake_case_regex.sub(r"_\1", string).upper()
|
|
29
27
|
|
|
30
28
|
|
|
31
29
|
def _convert(camel_input):
|
|
32
|
-
|
|
33
|
-
|
|
30
|
+
words = re.findall(r"[A-Z]?[a-z]+|[A-Z]{2,}(?=[A-Z][a-z]|\d|\W|$)|\d+", camel_input)
|
|
31
|
+
return "_".join(map(str.lower, words))
|
|
34
32
|
|
|
35
33
|
|
|
36
34
|
class EnumSerializer(ObjectSerializer):
|
|
37
35
|
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
# ix_start = len(enum_prefix) + 1
|
|
79
|
-
# return enum_items[v[ix_start:]]
|
|
80
|
-
#
|
|
81
|
-
# return enum_items[f'{enum_prefix}_{v}']
|
|
82
|
-
try:
|
|
83
|
-
return cls[v]
|
|
84
|
-
except KeyError:
|
|
85
|
-
dct = vars(cls)
|
|
86
|
-
n = v.lower()
|
|
87
|
-
nn = _convert(v).lower()
|
|
88
|
-
enum_prefix = _pascal_case_to_upper_snake_case(cls.__name__).lower()
|
|
89
|
-
for key in dct.keys():
|
|
90
|
-
k = key.lower()
|
|
91
|
-
if k == n:
|
|
92
|
-
return dct[key]
|
|
93
|
-
if k.startswith(enum_prefix) and k.endswith(n) or k.endswith(nn):
|
|
94
|
-
return dct[key]
|
|
95
|
-
raise
|
|
36
|
+
def __init__(self):
|
|
37
|
+
"""
|
|
38
|
+
Enum objects are serialized using their ".name" field and deserialized by indexing the string in the Enum type.
|
|
39
|
+
Example:
|
|
40
|
+
class Foo(Enum):
|
|
41
|
+
TEST = 1
|
|
42
|
+
foo = Foo.TEST
|
|
43
|
+
foo.name # => returns "TEST"
|
|
44
|
+
Foo["TEST"] # => returns Foo.TEST enum value.
|
|
45
|
+
"""
|
|
46
|
+
ObjectSerializer.__init__(
|
|
47
|
+
self,
|
|
48
|
+
lambda cls, v, _: EnumSerializer._to_str(cls, v),
|
|
49
|
+
lambda cls, v: EnumSerializer._from_str(cls, v),
|
|
50
|
+
)
|
|
51
|
+
|
|
52
|
+
@staticmethod
|
|
53
|
+
def _to_str(cls, v):
|
|
54
|
+
# "v" corresponds to an enum instance: Example foo or Foo.Test above.
|
|
55
|
+
# "cls" corresponds to the enum type Foo above.
|
|
56
|
+
return v.name
|
|
57
|
+
|
|
58
|
+
@staticmethod
|
|
59
|
+
def _from_str(cls, v):
|
|
60
|
+
# "v" corresponds to enum string: Example "TEST" above.
|
|
61
|
+
# "cls" corresponds to the enum type Foo above.
|
|
62
|
+
try:
|
|
63
|
+
return cls[v]
|
|
64
|
+
except KeyError:
|
|
65
|
+
dct = vars(cls)
|
|
66
|
+
n = v.lower()
|
|
67
|
+
nn = _convert(v).lower()
|
|
68
|
+
enum_prefix = _pascal_case_to_upper_snake_case(cls.__name__).lower()
|
|
69
|
+
for key in dct.keys():
|
|
70
|
+
k = key.lower()
|
|
71
|
+
if k == n:
|
|
72
|
+
return dct[key]
|
|
73
|
+
if k.startswith(enum_prefix) and k.endswith(n) or k.endswith(nn):
|
|
74
|
+
return dct[key]
|
|
75
|
+
raise
|
|
96
76
|
|
|
97
77
|
|
|
98
78
|
class ListSerializer(ObjectSerializer):
|
|
99
79
|
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
item_serializer.
|
|
109
|
-
|
|
110
|
-
lambda cls, l: [item_serializer.from_dict_value(cls, v) for v in l],
|
|
111
|
-
)
|
|
80
|
+
def __init__(self, item_serializer: ObjectSerializer):
|
|
81
|
+
"""
|
|
82
|
+
Lists are serialized based on the type they contain. Since objects are generated from proto files, a list always
|
|
83
|
+
contains objects of the same type, which is serialized/deserialized using "item_serializer".
|
|
84
|
+
"""
|
|
85
|
+
ObjectSerializer.__init__(
|
|
86
|
+
self,
|
|
87
|
+
lambda cls, l, ignore_defaults: [item_serializer.to_dict_value(cls, v, ignore_defaults) for v in l],
|
|
88
|
+
lambda cls, l: [item_serializer.from_dict_value(cls, v) for v in l],
|
|
89
|
+
)
|
|
112
90
|
|
|
113
91
|
|
|
114
92
|
class MapSerializer(ObjectSerializer):
|
|
115
93
|
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
k: item_serializer.from_dict_value(cls, v) for k, v in d.items()
|
|
129
|
-
},
|
|
130
|
-
)
|
|
94
|
+
def __init__(self, item_serializer: ObjectSerializer):
|
|
95
|
+
"""
|
|
96
|
+
Maps are serialized based on type of their values. Since maps keys are always predefined types, we don't need a
|
|
97
|
+
serializer for them.
|
|
98
|
+
"""
|
|
99
|
+
ObjectSerializer.__init__(
|
|
100
|
+
self,
|
|
101
|
+
lambda cls, d, ignore_defaults: {
|
|
102
|
+
k: item_serializer.to_dict_value(cls, v, ignore_defaults) for k, v in d.items()
|
|
103
|
+
},
|
|
104
|
+
lambda cls, d: {k: item_serializer.from_dict_value(cls, v) for k, v in d.items()},
|
|
105
|
+
)
|
|
131
106
|
|
|
132
107
|
|
|
133
108
|
class DateTimeSerializer(ObjectSerializer):
|
|
134
109
|
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
110
|
+
def __init__(self):
|
|
111
|
+
"""Date times are serialized/deserialized as a string in iso format"""
|
|
112
|
+
ObjectSerializer.__init__(
|
|
113
|
+
self,
|
|
114
|
+
lambda cls, dt, _: DateTimeSerializer._to_str(dt),
|
|
115
|
+
lambda _, v: DateTimeSerializer._from_str(v),
|
|
116
|
+
)
|
|
117
|
+
|
|
118
|
+
@staticmethod
|
|
119
|
+
def _to_str(dt):
|
|
120
|
+
return dt.isoformat(timespec="milliseconds") + "Z"
|
|
121
|
+
|
|
122
|
+
@staticmethod
|
|
123
|
+
def _from_str(v):
|
|
124
|
+
v = v.rstrip("Z")
|
|
125
|
+
fields = v.rsplit(".", maxsplit=1)
|
|
126
|
+
if len(fields) == 1:
|
|
127
|
+
return datetime.fromisoformat(v)
|
|
128
|
+
(dt, nanos) = fields
|
|
129
|
+
millis = nanos[:3]
|
|
130
|
+
try:
|
|
131
|
+
return datetime.fromisoformat(f"{dt}.{millis}")
|
|
132
|
+
except ValueError:
|
|
133
|
+
return datetime.fromisoformat(dt) # Python 3.9, 3.10
|
|
159
134
|
|
|
160
135
|
|
|
161
136
|
class TimeDeltaSerializer(ObjectSerializer):
|
|
137
|
+
# Scaling factors idea from https://github.com/protocolbuffers/protobuf/blob/master/csharp/src/Google.Protobuf/JsonParser.cs
|
|
138
|
+
SUBSECOND_SCALING_FACTORS = [
|
|
139
|
+
0,
|
|
140
|
+
100_000_000,
|
|
141
|
+
10_000_000,
|
|
142
|
+
1_000_000,
|
|
143
|
+
100_000,
|
|
144
|
+
10_000,
|
|
145
|
+
1_000,
|
|
146
|
+
100,
|
|
147
|
+
10,
|
|
148
|
+
1,
|
|
149
|
+
]
|
|
162
150
|
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
(
|
|
181
|
-
|
|
151
|
+
def __init__(self):
|
|
152
|
+
"""Time deltas are serialized/deserialized as a string in "<seconds>.<nanoseconds>s" format. Example: 151.500s"""
|
|
153
|
+
ObjectSerializer.__init__(
|
|
154
|
+
self,
|
|
155
|
+
lambda cls, t, _: TimeDeltaSerializer._to_dict_value(t),
|
|
156
|
+
lambda cls, v: TimeDeltaSerializer._from_dict_value(v),
|
|
157
|
+
)
|
|
158
|
+
|
|
159
|
+
@staticmethod
|
|
160
|
+
def _to_dict_value(delta):
|
|
161
|
+
seconds = delta.seconds
|
|
162
|
+
nanos_str = TimeDeltaSerializer._nanos_to_str(delta.microseconds * 1000)
|
|
163
|
+
if nanos_str is None:
|
|
164
|
+
return "{}s".format(seconds)
|
|
165
|
+
return "{}.{}s".format(seconds, nanos_str)
|
|
166
|
+
|
|
167
|
+
@staticmethod
|
|
168
|
+
def _nanos_to_str(nanos):
|
|
169
|
+
if nanos == 0:
|
|
170
|
+
return None
|
|
171
|
+
if nanos < 0:
|
|
172
|
+
nanos *= -1
|
|
173
|
+
if nanos % 1000000 == 0:
|
|
174
|
+
return "{}".format(nanos / 1000000)
|
|
175
|
+
elif nanos % 1000 == 0:
|
|
176
|
+
return "{}".format(nanos / 1000)
|
|
177
|
+
else:
|
|
178
|
+
return "{}".format(nanos)
|
|
179
|
+
|
|
180
|
+
@staticmethod
|
|
181
|
+
def _from_dict_value(value):
|
|
182
|
+
(seconds, nanosRaw) = value.rstrip("s").split(".")
|
|
183
|
+
nanos = int(nanosRaw) * TimeDeltaSerializer.SUBSECOND_SCALING_FACTORS[len(nanosRaw)]
|
|
184
|
+
return timedelta(seconds=int(seconds), microseconds=int(int(nanos) / 1000))
|
|
182
185
|
|
|
183
186
|
|
|
184
187
|
class FieldMaskSerializer(ObjectSerializer):
|
|
185
188
|
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
189
|
+
def __init__(self):
|
|
190
|
+
"""Field masks are serialized/deserialized as a string that contains a list of paths with a comma delimiter"""
|
|
191
|
+
ObjectSerializer.__init__(
|
|
192
|
+
self,
|
|
193
|
+
lambda cls, m, _: m.ToJsonString(),
|
|
194
|
+
lambda cls, v: FieldMaskSerializer._from_joined_paths(v),
|
|
195
|
+
)
|
|
193
196
|
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
197
|
+
@staticmethod
|
|
198
|
+
def _from_joined_paths(joined_paths):
|
|
199
|
+
mask = FieldMask()
|
|
200
|
+
mask.FromJsonString(joined_paths)
|
|
201
|
+
return mask
|
|
199
202
|
|
|
200
203
|
|
|
201
204
|
class KaggleObjectSerializer(ObjectSerializer):
|
|
202
205
|
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
206
|
+
def __init__(self):
|
|
207
|
+
"""
|
|
208
|
+
Kaggle objects (i.e., proto-generated types that inherit from KaggleObject) have custom "to_dict" and "from_dict"
|
|
209
|
+
methods that serialize/deserialize them to/from dictionaries.
|
|
210
|
+
"""
|
|
211
|
+
ObjectSerializer.__init__(
|
|
212
|
+
self,
|
|
213
|
+
# "v" is an instance of a KaggleObject. For example: "req = ListCompetitionsRequest()".
|
|
214
|
+
# So "req.to_dict()" returns a dictionary with keys as json field names. Example:
|
|
215
|
+
# '{"pageSize": 10, "page": 2}'
|
|
216
|
+
lambda cls, v, ignore_defaults: cls.to_dict(v, ignore_defaults),
|
|
217
|
+
# "cls" is the type of a KaggleObject. For example: ListCompetitionsRequest. All
|
|
218
|
+
# generated Kaggle objects have "from_dict" class method that takes a dict to create a
|
|
219
|
+
# new instance of the object. See "KaggleObject" class definition below.
|
|
220
|
+
lambda cls, v: cls.from_dict(v),
|
|
221
|
+
)
|
|
219
222
|
|
|
220
223
|
|
|
221
224
|
class FieldMetadata(object):
|
|
222
225
|
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
226
|
+
def __init__(
|
|
227
|
+
self,
|
|
228
|
+
json_name,
|
|
229
|
+
field_name,
|
|
230
|
+
private_field_name,
|
|
231
|
+
field_type,
|
|
232
|
+
default_value,
|
|
233
|
+
serializer,
|
|
234
|
+
optional=False,
|
|
235
|
+
):
|
|
236
|
+
self.json_name = json_name
|
|
237
|
+
self.field_name = field_name
|
|
238
|
+
self.private_field_name = private_field_name
|
|
239
|
+
self.field_type = field_type
|
|
240
|
+
self.default_value = default_value
|
|
241
|
+
self.serializer = serializer
|
|
242
|
+
self.optional = optional
|
|
243
|
+
|
|
244
|
+
def get_as_dict_item(self, instance, ignore_defaults=True):
|
|
245
|
+
value = getattr(instance, self.private_field_name)
|
|
246
|
+
if ignore_defaults and value == self.default_value:
|
|
247
|
+
return None
|
|
248
|
+
if value is None:
|
|
249
|
+
return None
|
|
250
|
+
return self.serializer.to_dict_value(self.field_type, value, ignore_defaults)
|
|
251
|
+
|
|
252
|
+
def set_from_dict(self, instance, json_dict):
|
|
253
|
+
if self.json_name not in json_dict:
|
|
254
|
+
return # Ignore unknown fields
|
|
255
|
+
value = json_dict[self.json_name]
|
|
256
|
+
if value == self.default_value:
|
|
257
|
+
return # Ignore default values
|
|
258
|
+
try:
|
|
259
|
+
setattr(
|
|
260
|
+
instance,
|
|
261
|
+
self.private_field_name,
|
|
262
|
+
self.serializer.from_dict_value(self.field_type, value),
|
|
263
|
+
)
|
|
264
|
+
except Exception as e:
|
|
265
|
+
raise
|
|
263
266
|
|
|
264
267
|
|
|
265
268
|
class KaggleObject(object):
|
|
266
269
|
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
for field in
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
field = self._get_field(field_name)
|
|
343
|
-
setattr(self, field.private_field_name, field.default_value)
|
|
344
|
-
|
|
345
|
-
def _get_field(self, field_name):
|
|
346
|
-
field = next((f for f in self._fields if f.field_name == field_name), None)
|
|
347
|
-
if field is None:
|
|
348
|
-
raise ValueError(
|
|
349
|
-
f'Protocol message {self.__class__.__name__} has no "{field_name}" field.'
|
|
350
|
-
)
|
|
351
|
-
return field
|
|
270
|
+
def endpoint(self):
|
|
271
|
+
raise "Error: endpoint must be defined by the request object"
|
|
272
|
+
|
|
273
|
+
@staticmethod
|
|
274
|
+
def endpoint_path():
|
|
275
|
+
return None
|
|
276
|
+
|
|
277
|
+
@staticmethod
|
|
278
|
+
def body_fields():
|
|
279
|
+
return None
|
|
280
|
+
|
|
281
|
+
@classmethod
|
|
282
|
+
def prepare_from(cls, http_response):
|
|
283
|
+
return cls.from_json(http_response.text)
|
|
284
|
+
|
|
285
|
+
@staticmethod
|
|
286
|
+
def method():
|
|
287
|
+
return "GET"
|
|
288
|
+
|
|
289
|
+
def _freeze(self):
|
|
290
|
+
self._is_frozen = True
|
|
291
|
+
|
|
292
|
+
def __setattr__(self, key, value):
|
|
293
|
+
if hasattr(self, "_is_frozen") and not hasattr(self, key):
|
|
294
|
+
raise AttributeError(f"Unknown field for {self.__class__.__name__}: {key}")
|
|
295
|
+
object.__setattr__(self, key, value)
|
|
296
|
+
|
|
297
|
+
def to_dict(self, ignore_defaults=True):
|
|
298
|
+
kv_pairs = [(field.json_name, field.get_as_dict_item(self, ignore_defaults)) for field in self._fields]
|
|
299
|
+
return {k: v for (k, v) in kv_pairs if not ignore_defaults or v is not None}
|
|
300
|
+
|
|
301
|
+
def to_field_map(self, ignore_defaults=True):
|
|
302
|
+
kv_pairs = [(field.field_name, field.get_as_dict_item(self, ignore_defaults)) for field in self._fields]
|
|
303
|
+
return {k: v for (k, v) in kv_pairs if not ignore_defaults or v is not None}
|
|
304
|
+
|
|
305
|
+
@classmethod
|
|
306
|
+
def from_dict(cls, json_dict):
|
|
307
|
+
instance = cls()
|
|
308
|
+
for field in cls._fields:
|
|
309
|
+
field.set_from_dict(instance, json_dict)
|
|
310
|
+
return instance
|
|
311
|
+
|
|
312
|
+
@classmethod
|
|
313
|
+
def from_json(cls, json_str):
|
|
314
|
+
return cls.from_dict(json.loads(json_str))
|
|
315
|
+
|
|
316
|
+
def to_json(self, ignore_defaults=True):
|
|
317
|
+
return json.dumps(self.to_dict(ignore_defaults))
|
|
318
|
+
|
|
319
|
+
def __str__(self):
|
|
320
|
+
return self.to_json(ignore_defaults=False)
|
|
321
|
+
|
|
322
|
+
def __repr__(self):
|
|
323
|
+
return self.to_json(ignore_defaults=False)
|
|
324
|
+
|
|
325
|
+
def __contains__(self, field_name):
|
|
326
|
+
try:
|
|
327
|
+
field = self._get_field(field_name)
|
|
328
|
+
except ValueError:
|
|
329
|
+
return False
|
|
330
|
+
value = getattr(self, field.private_field_name)
|
|
331
|
+
if field.optional:
|
|
332
|
+
return value is not None
|
|
333
|
+
else:
|
|
334
|
+
return value != field.default_value
|
|
335
|
+
|
|
336
|
+
def __delattr__(self, field_name):
|
|
337
|
+
field = self._get_field(field_name)
|
|
338
|
+
setattr(self, field.private_field_name, field.default_value)
|
|
339
|
+
|
|
340
|
+
def _get_field(self, field_name):
|
|
341
|
+
field = next((f for f in self._fields if f.field_name == field_name), None)
|
|
342
|
+
if field is None:
|
|
343
|
+
raise ValueError(f'Protocol message {self.__class__.__name__} has no "{field_name}" field.')
|
|
344
|
+
return field
|