fal 1.2.0__py3-none-any.whl → 1.2.2__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.

Potentially problematic release.


This version of fal might be problematic. Click here for more details.

fal/_fal_version.py CHANGED
@@ -12,5 +12,5 @@ __version__: str
12
12
  __version_tuple__: VERSION_TUPLE
13
13
  version_tuple: VERSION_TUPLE
14
14
 
15
- __version__ = version = '1.2.0'
16
- __version_tuple__ = version_tuple = (1, 2, 0)
15
+ __version__ = version = '1.2.2'
16
+ __version_tuple__ = version_tuple = (1, 2, 2)
fal/api.py CHANGED
@@ -1048,7 +1048,9 @@ class BaseServable:
1048
1048
  from uvicorn import Config
1049
1049
 
1050
1050
  app = self._build_app()
1051
- server = Server(config=Config(app, host="0.0.0.0", port=8080))
1051
+ server = Server(
1052
+ config=Config(app, host="0.0.0.0", port=8080, timeout_keep_alive=300)
1053
+ )
1052
1054
  metrics_app = FastAPI()
1053
1055
  metrics_app.add_route("/metrics", handle_metrics)
1054
1056
  metrics_server = Server(config=Config(metrics_app, host="0.0.0.0", port=9090))
fal/app.py CHANGED
@@ -7,7 +7,7 @@ import re
7
7
  import time
8
8
  import typing
9
9
  from contextlib import asynccontextmanager, contextmanager
10
- from typing import Any, Callable, ClassVar, TypeVar
10
+ from typing import Any, Callable, ClassVar, Literal, TypeVar
11
11
 
12
12
  import httpx
13
13
  from fastapi import FastAPI
@@ -152,6 +152,7 @@ class App(fal.api.BaseServable):
152
152
  "keep_alive": 60,
153
153
  }
154
154
  app_name: ClassVar[str]
155
+ app_auth: ClassVar[Literal["private", "public", "shared"]] = "private"
155
156
 
156
157
  def __init_subclass__(cls, **kwargs):
157
158
  app_name = kwargs.pop("name", None) or _to_fal_app_name(cls.__name__)
fal/cli/apps.py CHANGED
@@ -1,3 +1,5 @@
1
+ from __future__ import annotations
2
+
1
3
  from typing import TYPE_CHECKING
2
4
 
3
5
  from .parser import FalClientParser
@@ -6,7 +8,7 @@ if TYPE_CHECKING:
6
8
  from fal.sdk import AliasInfo, ApplicationInfo
7
9
 
8
10
 
9
- def _apps_table(apps: list["AliasInfo"]):
11
+ def _apps_table(apps: list[AliasInfo]):
10
12
  from rich.table import Table
11
13
 
12
14
  table = Table()
@@ -56,7 +58,7 @@ def _add_list_parser(subparsers, parents):
56
58
  parser.set_defaults(func=_list)
57
59
 
58
60
 
59
- def _app_rev_table(revs: list["ApplicationInfo"]):
61
+ def _app_rev_table(revs: list[ApplicationInfo]):
60
62
  from rich.table import Table
61
63
 
62
64
  table = Table()
fal/cli/create.py ADDED
@@ -0,0 +1,26 @@
1
+ PROJECT_TYPES = ["app"]
2
+
3
+
4
+ def _create_project(project_type: str):
5
+ from cookiecutter.main import cookiecutter
6
+
7
+ cookiecutter("https://github.com/fal-ai/cookiecutter-fal.git")
8
+
9
+
10
+ def add_parser(main_subparsers, parents):
11
+ apps_help = "Create fal applications."
12
+ parser = main_subparsers.add_parser(
13
+ "create",
14
+ description=apps_help,
15
+ help=apps_help,
16
+ parents=parents,
17
+ )
18
+
19
+ parser.add_argument(
20
+ metavar="project_type",
21
+ choices=PROJECT_TYPES,
22
+ help="Type of project to create.",
23
+ dest="project_type",
24
+ )
25
+
26
+ parser.set_defaults(func=_create_project)
fal/cli/deploy.py CHANGED
@@ -81,17 +81,18 @@ def _deploy(args):
81
81
 
82
82
  user = _get_user()
83
83
  host = FalServerlessHost(args.host)
84
- isolated_function, app_name = load_function_from(
84
+ isolated_function, app_name, app_auth = load_function_from(
85
85
  host,
86
86
  file_path,
87
87
  func_name,
88
88
  )
89
89
  app_name = args.app_name or app_name
90
+ app_auth = args.auth or app_auth or "private"
90
91
  app_id = host.register(
91
92
  func=isolated_function.func,
92
93
  options=isolated_function.options,
93
94
  application_name=app_name,
94
- application_auth_mode=args.auth,
95
+ application_auth_mode=app_auth,
95
96
  metadata=isolated_function.options.host.get("metadata", {}),
96
97
  )
97
98
 
@@ -151,7 +152,6 @@ def add_parser(main_subparsers, parents):
151
152
  parser.add_argument(
152
153
  "--auth",
153
154
  type=valid_auth_option,
154
- default="private",
155
155
  help="Application authentication mode (private, public).",
156
156
  )
157
157
  parser.set_defaults(func=_deploy)
fal/cli/main.py CHANGED
@@ -6,7 +6,7 @@ from fal import __version__
6
6
  from fal.console import console
7
7
  from fal.console.icons import CROSS_ICON
8
8
 
9
- from . import apps, auth, deploy, doctor, keys, run, secrets
9
+ from . import apps, auth, create, deploy, doctor, keys, run, secrets
10
10
  from .debug import debugtools, get_debug_parser
11
11
  from .parser import FalParser, FalParserExit
12
12
 
@@ -31,7 +31,7 @@ def _get_main_parser() -> argparse.ArgumentParser:
31
31
  required=True,
32
32
  )
33
33
 
34
- for cmd in [auth, apps, deploy, run, keys, secrets, doctor]:
34
+ for cmd in [auth, apps, deploy, run, keys, secrets, doctor, create]:
35
35
  cmd.add_parser(subparsers, parents)
36
36
 
37
37
  return parser
fal/cli/run.py CHANGED
@@ -6,7 +6,7 @@ def _run(args):
6
6
  from fal.utils import load_function_from
7
7
 
8
8
  host = FalServerlessHost(args.host)
9
- isolated_function, _ = load_function_from(host, *args.func_ref)
9
+ isolated_function, _, _ = load_function_from(host, *args.func_ref)
10
10
  # let our exc handlers handle UserFunctionException
11
11
  isolated_function.reraise = False
12
12
  isolated_function()
fal/logging/isolate.py CHANGED
@@ -17,11 +17,31 @@ class IsolateLogPrinter:
17
17
 
18
18
  def __init__(self, debug: bool = False) -> None:
19
19
  self.debug = debug
20
+ self._current_source: LogSource | None = None
21
+
22
+ def _maybe_print_header(self, source: LogSource):
23
+ from fal.console import console
24
+
25
+ if source == self._current_source:
26
+ return
27
+
28
+ msg = {
29
+ LogSource.BUILDER: "Building the environment",
30
+ LogSource.BRIDGE: "Unpacking user code",
31
+ LogSource.USER: "Running",
32
+ }.get(source)
33
+
34
+ if msg:
35
+ console.print(f"==> {msg}", style="bold green")
36
+
37
+ self._current_source = source
20
38
 
21
39
  def print(self, log: Log):
22
40
  if log.level < LogLevel.INFO and not self.debug:
23
41
  return
24
42
 
43
+ self._maybe_print_header(log.source)
44
+
25
45
  if log.source == LogSource.USER:
26
46
  stream = sys.stderr if log.level == LogLevel.STDERR else sys.stdout
27
47
  print(log.message, file=stream)
@@ -1,11 +1,12 @@
1
1
  from __future__ import annotations
2
2
 
3
3
  import hashlib
4
+ import os
4
5
  import shutil
5
6
  import subprocess
6
7
  import sys
7
8
  from pathlib import Path, PurePath
8
- from tempfile import TemporaryDirectory
9
+ from tempfile import NamedTemporaryFile, TemporaryDirectory
9
10
  from urllib.parse import urlparse
10
11
  from urllib.request import Request, urlopen
11
12
 
@@ -215,10 +216,14 @@ def _download_file_python(
215
216
  Returns:
216
217
  The path where the downloaded file has been saved.
217
218
  """
218
- import shutil
219
- import tempfile
220
-
221
- with tempfile.NamedTemporaryFile(delete=False) as temp_file:
219
+ basename = os.path.basename(target_path)
220
+ # NOTE: using the same directory to avoid potential copies across temp fs and target
221
+ # fs, and also to be able to atomically rename a downloaded file into place.
222
+ with NamedTemporaryFile(
223
+ delete=False,
224
+ dir=os.path.dirname(target_path),
225
+ prefix=f"{basename}.tmp",
226
+ ) as temp_file:
222
227
  try:
223
228
  file_path = temp_file.name
224
229
 
@@ -232,13 +237,14 @@ def _download_file_python(
232
237
 
233
238
  print(progress_msg, end="\r\n")
234
239
 
235
- # Move the file when the file is downloaded completely. Since the
236
- # file used is temporary, in a case of an interruption, the downloaded
237
- # content will be lost. So, it is safe to redownload the file in such cases.
238
- shutil.move(file_path, target_path)
240
+ # NOTE: Atomically renaming the file into place when the file is downloaded
241
+ # completely.
242
+ #
243
+ # Since the file used is temporary, in a case of an interruption, the
244
+ # downloaded content will be lost. So, it is safe to redownload the file in
245
+ # such cases.
246
+ os.rename(file_path, target_path)
239
247
 
240
- except Exception as error:
241
- raise error
242
248
  finally:
243
249
  Path(temp_file.name).unlink(missing_ok=True)
244
250
 
@@ -403,35 +409,38 @@ def clone_repository(
403
409
  print(f"Removing the existing repository: {local_repo_path} ")
404
410
  shutil.rmtree(local_repo_path)
405
411
 
406
- # Temporary directory to download the repo into.
407
- temp_dir = TemporaryDirectory()
408
- temp_dir_path = temp_dir.name
409
-
410
- try:
411
- print(f"Cloning the repository '{https_url}' .")
412
-
413
- # Clone with disabling the logs and advices for detached HEAD state.
414
- clone_command = [
415
- "git",
416
- "clone",
417
- "--recursive",
418
- https_url,
419
- temp_dir_path,
420
- ]
421
- subprocess.check_call(clone_command)
422
-
423
- if commit_hash:
424
- checkout_command = ["git", "checkout", commit_hash]
425
- subprocess.check_call(checkout_command, cwd=temp_dir_path)
426
-
427
- # Move the repository when the clone and checkout is finished.
428
- shutil.move(temp_dir_path, local_repo_path)
429
-
430
- except Exception as error:
431
- print(f"{error}\nFailed to clone repository '{https_url}' .")
432
- temp_dir.cleanup()
412
+ # NOTE: using the target_dir to be able to avoid potential copies across temp fs
413
+ # and target fs, and also to be able to atomically rename repo_name dir into place
414
+ # when we are done setting it up.
415
+ os.makedirs(target_dir, exist_ok=True) # type: ignore[arg-type]
416
+ with TemporaryDirectory(
417
+ dir=target_dir,
418
+ suffix=f"{local_repo_path.name}.tmp",
419
+ ) as temp_dir:
420
+ try:
421
+ print(f"Cloning the repository '{https_url}' .")
422
+
423
+ # Clone with disabling the logs and advices for detached HEAD state.
424
+ clone_command = [
425
+ "git",
426
+ "clone",
427
+ "--recursive",
428
+ https_url,
429
+ temp_dir,
430
+ ]
431
+ subprocess.check_call(clone_command)
432
+
433
+ if commit_hash:
434
+ checkout_command = ["git", "checkout", commit_hash]
435
+ subprocess.check_call(checkout_command, cwd=temp_dir)
436
+
437
+ # NOTE: Atomically renaming the repository directory into place when the
438
+ # clone and checkout are done.
439
+ os.rename(temp_dir, local_repo_path)
433
440
 
434
- raise error
441
+ except Exception as error:
442
+ print(f"{error}\nFailed to clone repository '{https_url}' .")
443
+ raise error
435
444
 
436
445
  if include_to_path:
437
446
  __add_local_path_to_sys_path(local_repo_path)
fal/utils.py CHANGED
@@ -10,13 +10,13 @@ def load_function_from(
10
10
  host: FalServerlessHost,
11
11
  file_path: str,
12
12
  function_name: str | None = None,
13
- ) -> tuple[IsolatedFunction, str | None]:
13
+ ) -> tuple[IsolatedFunction, str | None, str | None]:
14
14
  import runpy
15
15
 
16
16
  module = runpy.run_path(file_path)
17
17
  if function_name is None:
18
18
  fal_objects = {
19
- obj.app_name: obj_name
19
+ obj_name: obj
20
20
  for obj_name, obj in module.items()
21
21
  if isinstance(obj, type)
22
22
  and issubclass(obj, fal.App)
@@ -30,9 +30,12 @@ def load_function_from(
30
30
  "Please specify the name of the app."
31
31
  )
32
32
 
33
- [(app_name, function_name)] = fal_objects.items()
33
+ [(function_name, obj)] = fal_objects.items()
34
+ app_name = obj.app_name
35
+ app_auth = obj.app_auth
34
36
  else:
35
37
  app_name = None
38
+ app_auth = None
36
39
 
37
40
  if function_name not in module:
38
41
  raise FalServerlessError(f"Function '{function_name}' not found in module")
@@ -44,10 +47,11 @@ def load_function_from(
44
47
  target = module[function_name]
45
48
  if isinstance(target, type) and issubclass(target, App):
46
49
  app_name = target.app_name
50
+ app_auth = target.app_auth
47
51
  target = wrap_app(target, host=host)
48
52
 
49
53
  if not isinstance(target, IsolatedFunction):
50
54
  raise FalServerlessError(
51
55
  f"Function '{function_name}' is not a fal.function or a fal.App"
52
56
  )
53
- return target, app_name
57
+ return target, app_name, app_auth
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: fal
3
- Version: 1.2.0
3
+ Version: 1.2.2
4
4
  Summary: fal is an easy-to-use Serverless Python Framework
5
5
  Author: Features & Labels <support@fal.ai>
6
6
  Requires-Python: >=3.8
@@ -34,6 +34,7 @@ Requires-Dist: websockets <13,>=12.0
34
34
  Requires-Dist: pillow <11,>=10.2.0
35
35
  Requires-Dist: pyjwt[crypto] <3,>=2.8.0
36
36
  Requires-Dist: uvicorn <1,>=0.29.0
37
+ Requires-Dist: cookiecutter
37
38
  Requires-Dist: importlib-metadata >=4.4 ; python_version < "3.10"
38
39
  Provides-Extra: dev
39
40
  Requires-Dist: fal[test] ; extra == 'dev'
@@ -1,10 +1,10 @@
1
1
  fal/__init__.py,sha256=wXs1G0gSc7ZK60-bHe-B2m0l_sA6TrFk4BxY0tMoLe8,784
2
2
  fal/__main__.py,sha256=MSmt_5Xg84uHqzTN38JwgseJK8rsJn_11A8WD99VtEo,61
3
- fal/_fal_version.py,sha256=zMnMemknXglcJs59xkicNzeEJTVgYd1omSfLWj76yWw,411
3
+ fal/_fal_version.py,sha256=XEVwqOPlIChKtEnSO5v_SvghWXnn9WeQSoJ436w3v9Y,411
4
4
  fal/_serialization.py,sha256=rD2YiSa8iuzCaZohZwN_MPEB-PpSKbWRDeaIDpTEjyY,7653
5
5
  fal/_version.py,sha256=EBGqrknaf1WygENX-H4fBefLvHryvJBBGtVJetaB0NY,266
6
- fal/api.py,sha256=x60GlBWynDd1yhHsVWeqf07WVTzgbwNC6cqCjhlTiFQ,40556
7
- fal/app.py,sha256=duOf_YKE8o30hmhNtF9zvkT8wlKYXW7hdQLJtPrXHik,15793
6
+ fal/api.py,sha256=LAPl5Hf6ZWzEjv4lFUtsisWgrnXH_qNUHdJrEHT_A5Y,40602
7
+ fal/app.py,sha256=oyN4PNULFJtjOwHYrR5lh4Ks_zBi-dEPzvFYRUXe0sI,15877
8
8
  fal/apps.py,sha256=FrKmaAUo8U9vE_fcva0GQvk4sCrzaTEr62lGtu3Ld5M,6825
9
9
  fal/container.py,sha256=V7riyyq8AZGwEX9QaqRQDZyDN_bUKeRKV1OOZArXjL0,622
10
10
  fal/flags.py,sha256=oWN_eidSUOcE9wdPK_77si3A1fpgOC0UEERPsvNLIMc,842
@@ -12,21 +12,22 @@ fal/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
12
12
  fal/rest_client.py,sha256=kGBGmuyHfX1lR910EoKCYPjsyU8MdXawT_cW2q8Sajc,568
13
13
  fal/sdk.py,sha256=wA58DYnSK1vdsBi8Or9Z8kvMMEyBNfeZYk_xulSfTWE,20078
14
14
  fal/sync.py,sha256=ZuIJA2-hTPNANG9B_NNJZUsO68EIdTH0dc9MzeVE2VU,4340
15
- fal/utils.py,sha256=pstF7-13xjB1gKOtzz4tIXc8b8nMSDd9HO79WCiVrt4,1753
15
+ fal/utils.py,sha256=4-V6iGSRd3kG_-UP6OdZp_-EhAkl3zectFlFKkCsS0Q,1884
16
16
  fal/workflows.py,sha256=jx3tGy2R7cN6lLvOzT6lhhlcjmiq64iZls2smVrmQj0,14657
17
17
  fal/auth/__init__.py,sha256=r8iA2-5ih7-Fik3gEC4HEWNFbGoxpYnXpZu1icPIoS0,3561
18
18
  fal/auth/auth0.py,sha256=rSG1mgH-QGyKfzd7XyAaj1AYsWt-ho8Y_LZ-FUVWzh4,5421
19
19
  fal/auth/local.py,sha256=sndkM6vKpeVny6NHTacVlTbiIFqaksOmw0Viqs_RN1U,1790
20
20
  fal/cli/__init__.py,sha256=padK4o0BFqq61kxAA1qQ0jYr2SuhA2mf90B3AaRkmJA,37
21
- fal/cli/apps.py,sha256=GlQSEE68HYfnhK90WIiOx611tcVkkbc9bu_10FMaFLY,8143
21
+ fal/cli/apps.py,sha256=-DDp-Gvxz5kHho5YjAhbri8vOny_9cftAI_wP2KR5nU,8175
22
22
  fal/cli/auth.py,sha256=--MhfHGwxmtHbRkGioyn1prKn_U-pBzbz0G_QeZou-U,1352
23
+ fal/cli/create.py,sha256=a8WDq-nJLFTeoIXqpb5cr7GR7YR9ZZrQCawNm34KXXE,627
23
24
  fal/cli/debug.py,sha256=u_urnyFzSlNnrq93zz_GXE9FX4VyVxDoamJJyrZpFI0,1312
24
- fal/cli/deploy.py,sha256=jA-C4Pvzie70MVDnaFKc6ByfBrNFeaMROAKX98RD8tQ,4785
25
+ fal/cli/deploy.py,sha256=1e4OERVGtfwgM0VEFjlCLpNyuOl1BiLI-dx8u-71PVs,4817
25
26
  fal/cli/doctor.py,sha256=U4ne9LX5gQwNblsYQ27XdO8AYDgbYjTO39EtxhwexRM,983
26
27
  fal/cli/keys.py,sha256=trDpA3LJu9S27qE_K8Hr6fKLK4vwVzbxUHq8TFrV4pw,3157
27
- fal/cli/main.py,sha256=MxETDhqIT37quMbmofSMxBcAFOhnEHjpQ_pYEtOhApM,1993
28
+ fal/cli/main.py,sha256=_Wh_DQc02qwh-ZN7v41lZm0lDR1WseViXVOcqUlyWLg,2009
28
29
  fal/cli/parser.py,sha256=r1hd5e8Jq6yzDZw8-S0On1EjJbjRtHMuVuHC6MlvUj4,2835
29
- fal/cli/run.py,sha256=enhnpqo07PzGK2uh8M522zenNWxrBFOqb3oMZ8XiWdg,870
30
+ fal/cli/run.py,sha256=8wHNDruIr8i21JwbfFzS389C-y0jktM5zN5iDnJHsvA,873
30
31
  fal/cli/secrets.py,sha256=740msFm7d41HruudlcfqUXlFl53N-WmChsQP9B9M9Po,2572
31
32
  fal/console/__init__.py,sha256=ernZ4bzvvliQh5SmrEqQ7lA5eVcbw6Ra2jalKtA7dxg,132
32
33
  fal/console/icons.py,sha256=De9MfFaSkO2Lqfne13n3PrYfTXJVIzYZVqYn5BWsdrA,108
@@ -35,7 +36,7 @@ fal/exceptions/__init__.py,sha256=x3fp97qMr5zCQJghMq6k2ESXWSrkWumO1BZebh3pWsI,92
35
36
  fal/exceptions/_base.py,sha256=oF2XfitbiDGObmSF1IX50uAdV8IUvOfR-YsGmMQSE0A,161
36
37
  fal/exceptions/auth.py,sha256=gxRago5coI__vSIcdcsqhhq1lRPkvCnwPAueIaXTAdw,329
37
38
  fal/logging/__init__.py,sha256=snqprf7-sKw6oAATS_Yxklf-a3XhLg0vIHICPwLp6TM,1583
38
- fal/logging/isolate.py,sha256=uKZpF5p02jqaiyXNYt7VUp43ShdKc_N4UUi7z65nmRc,1741
39
+ fal/logging/isolate.py,sha256=jJSgDHkFg4sB0xElYSqCYF6IAxy6jEgSfjwFuKJIZbA,2305
39
40
  fal/logging/style.py,sha256=ckIgHzvF4DShM5kQh8F133X53z_vF46snuDHVmo_h9g,386
40
41
  fal/logging/trace.py,sha256=OhzB6d4rQZimBc18WFLqH_9BGfqFFumKKTAGSsmWRMg,1904
41
42
  fal/logging/user.py,sha256=0Xvb8n6tSb9l_V51VDzv6SOdYEFNouV_6nF_W9e7uNQ,642
@@ -51,7 +52,7 @@ fal/toolkit/file/providers/r2.py,sha256=WxmOHF5WxHt6tKMcFjWj7ZWO8a1EXysO9lfYv_tB
51
52
  fal/toolkit/image/__init__.py,sha256=qNLyXsBWysionUjbeWbohLqWlw3G_UpzunamkZd_JLQ,71
52
53
  fal/toolkit/image/image.py,sha256=UDIHgkxae8LzmCvWBM9GayMnK8c0JMMfsrVlLnW5rto,4234
53
54
  fal/toolkit/utils/__init__.py,sha256=CrmM9DyCz5-SmcTzRSm5RaLgxy3kf0ZsSEN9uhnX2Xo,97
54
- fal/toolkit/utils/download_utils.py,sha256=fDUITgdGW0wRXLE0NuQg29YJnJS3yr6l0zbRKqX6zMU,17006
55
+ fal/toolkit/utils/download_utils.py,sha256=9WMpn0mFIhkFelQpPj5KG-pC7RMyyOzGHbNRDSyz07o,17664
55
56
  openapi_fal_rest/__init__.py,sha256=ziculmF_i6trw63LzZGFX-6W3Lwq9mCR8_UpkpvpaHI,152
56
57
  openapi_fal_rest/client.py,sha256=G6BpJg9j7-JsrAUGddYwkzeWRYickBjPdcVgXoPzxuE,2817
57
58
  openapi_fal_rest/errors.py,sha256=8mXSxdfSGzxT82srdhYbR0fHfgenxJXaUtMkaGgb6iU,470
@@ -115,8 +116,8 @@ openapi_fal_rest/models/workflow_node_type.py,sha256=-FzyeY2bxcNmizKbJI8joG7byRi
115
116
  openapi_fal_rest/models/workflow_schema.py,sha256=4K5gsv9u9pxx2ItkffoyHeNjBBYf6ur5bN4m_zePZNY,2019
116
117
  openapi_fal_rest/models/workflow_schema_input.py,sha256=2OkOXWHTNsCXHWS6EGDFzcJKkW5FIap-2gfO233EvZQ,1191
117
118
  openapi_fal_rest/models/workflow_schema_output.py,sha256=EblwSPAGfWfYVWw_WSSaBzQVju296is9o28rMBAd0mc,1196
118
- fal-1.2.0.dist-info/METADATA,sha256=sfrRbze0jlBv3qqW-eG6sDFb6L6G7zHfJ9UpivA--uY,3777
119
- fal-1.2.0.dist-info/WHEEL,sha256=Z4pYXqR_rTB7OWNDYFOm1qRk0RX6GFP2o8LgvP453Hk,91
120
- fal-1.2.0.dist-info/entry_points.txt,sha256=32zwTUC1U1E7nSTIGCoANQOQ3I7-qHG5wI6gsVz5pNU,37
121
- fal-1.2.0.dist-info/top_level.txt,sha256=r257X1L57oJL8_lM0tRrfGuXFwm66i1huwQygbpLmHw,21
122
- fal-1.2.0.dist-info/RECORD,,
119
+ fal-1.2.2.dist-info/METADATA,sha256=ip9ta9NozUhujfW9CmFlgVrVmrvZztXkFFOCDCboQi0,3805
120
+ fal-1.2.2.dist-info/WHEEL,sha256=Wyh-_nZ0DJYolHNn1_hMa4lM7uDedD_RGVwbmTjyItk,91
121
+ fal-1.2.2.dist-info/entry_points.txt,sha256=32zwTUC1U1E7nSTIGCoANQOQ3I7-qHG5wI6gsVz5pNU,37
122
+ fal-1.2.2.dist-info/top_level.txt,sha256=r257X1L57oJL8_lM0tRrfGuXFwm66i1huwQygbpLmHw,21
123
+ fal-1.2.2.dist-info/RECORD,,
@@ -1,5 +1,5 @@
1
1
  Wheel-Version: 1.0
2
- Generator: setuptools (70.3.0)
2
+ Generator: setuptools (71.1.0)
3
3
  Root-Is-Purelib: true
4
4
  Tag: py3-none-any
5
5