idf-build-apps 2.5.0rc2__py3-none-any.whl → 2.5.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.
@@ -8,7 +8,7 @@ Tools for building ESP-IDF related apps.
8
8
  # ruff: noqa: E402
9
9
  # avoid circular imports
10
10
 
11
- __version__ = '2.5.0rc2'
11
+ __version__ = '2.5.2'
12
12
 
13
13
  from .session_args import (
14
14
  SessionArgs,
idf_build_apps/args.py CHANGED
@@ -11,6 +11,7 @@ import sys
11
11
  import typing as t
12
12
  from copy import deepcopy
13
13
  from dataclasses import dataclass
14
+ from io import TextIOWrapper
14
15
  from pathlib import Path
15
16
  from typing import Any
16
17
 
@@ -51,6 +52,8 @@ class FieldMetadata:
51
52
  :param choices: choices for the argument, used in argparse
52
53
  :param type: type for the argument, used in argparse
53
54
  :param required: whether the argument is required, used in argparse
55
+ :param default: default value for the argument, used in argparse
56
+ :param hidden: whether the argument is hidden, used in argparse
54
57
  """
55
58
 
56
59
  # validate method
@@ -66,6 +69,8 @@ class FieldMetadata:
66
69
  required: bool = False
67
70
  # usually default is not needed. only set it when different from the default value of the field
68
71
  default: t.Any = None
72
+ # hidden field, use deprecated instead, or hide it in the argparse
73
+ hidden: bool = False
69
74
 
70
75
 
71
76
  P = ParamSpec('P')
@@ -112,6 +117,7 @@ class BaseArguments(BaseSettings):
112
117
  toml_file='.idf_build_apps.toml',
113
118
  pyproject_toml_table_header=('tool', 'idf-build-apps'),
114
119
  pyproject_toml_depth=sys.maxsize,
120
+ extra='ignore',
115
121
  )
116
122
 
117
123
  @classmethod
@@ -135,8 +141,9 @@ class BaseArguments(BaseSettings):
135
141
  if info.field_name and info.field_name in cls.model_fields:
136
142
  f = cls.model_fields[info.field_name]
137
143
  meta = get_meta(f)
138
- if meta and meta.validate_method and ValidateMethod.TO_LIST in meta.validate_method:
139
- return to_list(v)
144
+ if meta and meta.validate_method:
145
+ if ValidateMethod.TO_LIST in meta.validate_method:
146
+ return to_list(v)
140
147
 
141
148
  return v
142
149
 
@@ -195,7 +202,9 @@ class DependencyDrivenBuildArguments(GlobalArguments):
195
202
  validate_method=[ValidateMethod.TO_LIST],
196
203
  type=semicolon_separated_str_to_list,
197
204
  ),
198
- description='semicolon-separated list of modified components',
205
+ description='semicolon-separated list of modified components. '
206
+ 'If set to "", the value would be considered as None. '
207
+ 'If set to ";", the value would be considered as an empty list.',
199
208
  default=None,
200
209
  )
201
210
  modified_files: t.Optional[t.List[str]] = field(
@@ -203,7 +212,9 @@ class DependencyDrivenBuildArguments(GlobalArguments):
203
212
  validate_method=[ValidateMethod.TO_LIST],
204
213
  type=semicolon_separated_str_to_list,
205
214
  ),
206
- description='semicolon-separated list of modified files',
215
+ description='semicolon-separated list of modified files. '
216
+ 'If set to "", the value would be considered as None. '
217
+ 'If set to ";", the value would be considered as an empty list.',
207
218
  default=None,
208
219
  )
209
220
  deactivate_dependency_driven_build_by_components: t.Optional[t.List[str]] = field(
@@ -219,7 +230,10 @@ class DependencyDrivenBuildArguments(GlobalArguments):
219
230
  shorthand='-dc',
220
231
  ),
221
232
  description='semicolon-separated list of components. '
222
- 'dependency-driven build feature will be deactivated when any of these components are modified',
233
+ 'dependency-driven build feature will be deactivated when any of these components are modified. '
234
+ 'Must be specified together with --modified-components. '
235
+ 'If set to "", the value would be considered as None. '
236
+ 'If set to ";", the value would be considered as an empty list.',
223
237
  validation_alias=AliasChoices(
224
238
  'deactivate_dependency_driven_build_by_components', 'ignore_app_dependencies_components'
225
239
  ),
@@ -238,7 +252,10 @@ class DependencyDrivenBuildArguments(GlobalArguments):
238
252
  shorthand='-df',
239
253
  ),
240
254
  description='semicolon-separated list of file patterns. '
241
- 'dependency-driven build feature will be deactivated when any of matched files are modified',
255
+ 'dependency-driven build feature will be deactivated when any of matched files are modified. '
256
+ 'Must be specified together with --modified-files. '
257
+ 'If set to "", the value would be considered as None. '
258
+ 'If set to ";", the value would be considered as an empty list.',
242
259
  validation_alias=AliasChoices(
243
260
  'deactivate_dependency_driven_build_by_filepatterns', 'ignore_app_dependencies_filepatterns'
244
261
  ),
@@ -253,7 +270,7 @@ class DependencyDrivenBuildArguments(GlobalArguments):
253
270
  )
254
271
  compare_manifest_sha_filepath: t.Optional[str] = field(
255
272
  None,
256
- description='Path to the file containing the sha256 hash of the manifest rules. '
273
+ description='Path to the file containing the hash of the manifest rules. '
257
274
  'Compare the hash with the current manifest rules. '
258
275
  'All matched apps will be built if the corresponding manifest rule is modified',
259
276
  default=None,
@@ -443,7 +460,10 @@ class FindBuildArguments(DependencyDrivenBuildArguments):
443
460
  default=False,
444
461
  )
445
462
  default_build_targets: t.Optional[t.List[str]] = field(
446
- None,
463
+ FieldMetadata(
464
+ validate_method=[ValidateMethod.TO_LIST],
465
+ nargs='+',
466
+ ),
447
467
  description='space-separated list of the default enabled build targets for the apps. '
448
468
  'When not specified, the default value is the targets listed by `idf.py --list-targets`',
449
469
  default=None,
@@ -602,10 +622,16 @@ class BuildArguments(FindBuildArguments):
602
622
  validation_alias=AliasChoices('ignore_warning_strs', 'ignore_warning_str'),
603
623
  default=None,
604
624
  )
605
- ignore_warning_files: t.Optional[t.List[str]] = field(
625
+ ignore_warning_files: t.Optional[t.List[t.Union[str, TextIOWrapper]]] = field(
606
626
  FieldMetadata(
607
- deprecates={'ignore_warning_file': {}},
627
+ validate_method=[ValidateMethod.TO_LIST],
628
+ deprecates={
629
+ 'ignore_warning_file': {
630
+ 'type': argparse.FileType('r'),
631
+ }
632
+ },
608
633
  nargs='+',
634
+ type=argparse.FileType('r'),
609
635
  ),
610
636
  description='Path to the files containing the patterns to ignore the warnings in the build output',
611
637
  validation_alias=AliasChoices('ignore_warning_files', 'ignore_warning_file'),
@@ -621,7 +647,10 @@ class BuildArguments(FindBuildArguments):
621
647
 
622
648
  # Attrs that support placeholders
623
649
  collect_size_info_filename: t.Optional[str] = field(
624
- None,
650
+ FieldMetadata(
651
+ deprecates={'collect_size_info': {}},
652
+ hidden=True,
653
+ ),
625
654
  description='Record size json filepath of the built apps to the specified file. '
626
655
  'Each line is a json string. Can expand placeholders @p',
627
656
  validation_alias=AliasChoices('collect_size_info_filename', 'collect_size_info'),
@@ -629,7 +658,10 @@ class BuildArguments(FindBuildArguments):
629
658
  exclude=True, # computed field is used
630
659
  )
631
660
  collect_app_info_filename: t.Optional[str] = field(
632
- None,
661
+ FieldMetadata(
662
+ deprecates={'collect_app_info': {}},
663
+ hidden=True,
664
+ ),
633
665
  description='Record serialized app model of the built apps to the specified file. '
634
666
  'Each line is a json string. Can expand placeholders @p',
635
667
  validation_alias=AliasChoices('collect_app_info_filename', 'collect_app_info'),
@@ -637,7 +669,10 @@ class BuildArguments(FindBuildArguments):
637
669
  exclude=True, # computed field is used
638
670
  )
639
671
  junitxml_filename: t.Optional[str] = field(
640
- None,
672
+ FieldMetadata(
673
+ deprecates={'junitxml': {}},
674
+ hidden=True,
675
+ ),
641
676
  description='Path to the junitxml file to record the build results. Can expand placeholder @p',
642
677
  validation_alias=AliasChoices('junitxml_filename', 'junitxml'),
643
678
  default=None,
@@ -654,8 +689,14 @@ class BuildArguments(FindBuildArguments):
654
689
  for s in self.ignore_warning_strs:
655
690
  ignore_warnings_regexes.append(re.compile(s))
656
691
  if self.ignore_warning_files:
657
- for s in self.ignore_warning_files:
658
- ignore_warnings_regexes.append(re.compile(s.strip()))
692
+ for f in self.ignore_warning_files:
693
+ if isinstance(f, str):
694
+ with open(f) as fr:
695
+ for s in fr:
696
+ ignore_warnings_regexes.append(re.compile(s.strip()))
697
+ else:
698
+ for s in f:
699
+ ignore_warnings_regexes.append(re.compile(s.strip()))
659
700
  App.IGNORE_WARNS_REGEXES = ignore_warnings_regexes
660
701
 
661
702
  @computed_field # type: ignore
@@ -734,12 +775,20 @@ def add_args_to_parser(argument_cls: t.Type[BaseArguments], parser: argparse.Arg
734
775
  if _shorthand:
735
776
  _names.append(_shorthand)
736
777
 
778
+ if f_meta.hidden: # f is hidden, use deprecated field instead
779
+ help_msg = f.description
780
+ else:
781
+ help_msg = f'[Deprecated] Use {_snake_case_to_cli_arg_name(f_name)} instead'
782
+
737
783
  parser.add_argument(
738
784
  *_names,
739
785
  **dep_f_kwargs,
740
- help=f'[Deprecated] Use {_snake_case_to_cli_arg_name(f_name)} instead',
786
+ help=help_msg,
741
787
  )
742
788
 
789
+ if f_meta and f_meta.hidden:
790
+ continue
791
+
743
792
  names = [_snake_case_to_cli_arg_name(f_name)]
744
793
  if f_meta and f_meta.shorthand:
745
794
  names.append(f_meta.shorthand)
@@ -752,14 +801,19 @@ def add_args_to_parser(argument_cls: t.Type[BaseArguments], parser: argparse.Arg
752
801
  kwargs['required'] = True
753
802
  if f_meta.action:
754
803
  kwargs['action'] = f_meta.action
804
+ # to make the CLI override config file work
805
+ if f_meta.action == 'store_true':
806
+ kwargs['default'] = None
807
+
755
808
  if f_meta.nargs:
756
809
  kwargs['nargs'] = f_meta.nargs
757
810
  if f_meta.choices:
758
811
  kwargs['choices'] = f_meta.choices
759
812
  if f_meta.default:
760
813
  kwargs['default'] = f_meta.default
761
- if 'default' not in kwargs:
762
- kwargs['default'] = f.default
814
+
815
+ # here in CLI arguments, don't set the default to field.default
816
+ # otherwise it will override the config file settings
763
817
 
764
818
  parser.add_argument(
765
819
  *names,
idf_build_apps/finder.py CHANGED
@@ -33,6 +33,7 @@ def _get_apps_from_path(
33
33
  app_cls: t.Type[App] = CMakeApp,
34
34
  args: FindArguments,
35
35
  ) -> t.List[App]:
36
+ # trigger test
36
37
  def _validate_app(_app: App) -> bool:
37
38
  if target not in _app.supported_targets:
38
39
  LOGGER.debug('=> Ignored. %s only supports targets: %s', _app, ', '.join(_app.supported_targets))
idf_build_apps/main.py CHANGED
@@ -70,6 +70,15 @@ def find_apps(
70
70
  """
71
71
  apply_config_file(config_file)
72
72
 
73
+ # compatible with old usage
74
+ ## `preserve`
75
+ if 'preserve' in kwargs:
76
+ LOGGER.warning(
77
+ 'Passing "preserve" directly is deprecated. '
78
+ 'Pass "no_preserve" instead to disable preserving the build directory'
79
+ )
80
+ kwargs['no_preserve'] = not kwargs.pop('preserve')
81
+
73
82
  if find_arguments is None:
74
83
  find_arguments = FindArguments(
75
84
  paths=to_list(paths), # type: ignore
@@ -125,7 +134,8 @@ def build_apps(
125
134
  """
126
135
  apply_config_file(config_file)
127
136
 
128
- apps = to_list(apps)
137
+ # compatible with old usage
138
+ ## `check_app_dependencies`
129
139
  if 'check_app_dependencies' in kwargs:
130
140
  LOGGER.warning(
131
141
  'Passing "check_app_dependencies" directly is deprecated. '
@@ -138,6 +148,7 @@ def build_apps(
138
148
  **kwargs,
139
149
  )
140
150
 
151
+ apps = to_list(apps)
141
152
  if apps is None:
142
153
  apps = find_apps(
143
154
  find_arguments=FindArguments(
@@ -431,7 +442,8 @@ def main():
431
442
  for app in failed_apps:
432
443
  print(f' {app}')
433
444
 
434
- sys.exit(ret_code)
445
+ if ret_code != 0:
446
+ sys.exit(ret_code)
435
447
 
436
448
 
437
449
  def json_to_app(json_str: str, extra_classes: t.Optional[t.List[t.Type[App]]] = None) -> App:
@@ -1,6 +1,5 @@
1
1
  # SPDX-FileCopyrightText: 2022-2024 Espressif Systems (Shanghai) CO LTD
2
2
  # SPDX-License-Identifier: Apache-2.0
3
-
4
3
  import operator
5
4
  import os
6
5
  from ast import (
@@ -16,6 +15,7 @@ from packaging.version import (
16
15
  from pyparsing import (
17
16
  Keyword,
18
17
  Literal,
18
+ MatchFirst,
19
19
  ParseResults,
20
20
  QuotedString,
21
21
  Suppress,
@@ -35,6 +35,7 @@ from ..constants import (
35
35
  IDF_VERSION_PATCH,
36
36
  )
37
37
  from ..utils import (
38
+ InvalidIfClause,
38
39
  InvalidInput,
39
40
  to_version,
40
41
  )
@@ -177,22 +178,31 @@ class BoolExpr(Stmt):
177
178
  pass
178
179
 
179
180
 
180
- class BoolAnd(BoolExpr):
181
- def __init__(self, t: ParseResults):
182
- self.left: BoolStmt = t[0][0]
183
- self.right: BoolStmt = t[0][2]
181
+ def _and(_l, _r):
182
+ return _l and _r
184
183
 
185
- def get_value(self, target: str, config_name: str) -> Any:
186
- return self.left.get_value(target, config_name) and self.right.get_value(target, config_name)
184
+
185
+ def _or(_l, _r):
186
+ return _l or _r
187
187
 
188
188
 
189
- class BoolOr(BoolExpr):
189
+ class BoolOrAnd(BoolExpr):
190
190
  def __init__(self, t: ParseResults):
191
+ if len(t[0]) > 3:
192
+ raise InvalidIfClause(
193
+ 'Chaining "and"/"or" is not allowed. Please use paratheses instead. '
194
+ 'For example: "a and b and c" should be "(a and b) and c".'
195
+ )
191
196
  self.left: BoolStmt = t[0][0]
192
197
  self.right: BoolStmt = t[0][2]
193
198
 
199
+ if t[0][1] == 'and':
200
+ self.operation = _and
201
+ if t[0][1] == 'or':
202
+ self.operation = _or
203
+
194
204
  def get_value(self, target: str, config_name: str) -> Any:
195
- return self.left.get_value(target, config_name) or self.right.get_value(target, config_name)
205
+ return self.operation(self.left.get_value(target, config_name), self.right.get_value(target, config_name))
196
206
 
197
207
 
198
208
  CAP_WORD = Word(alphas.upper(), nums + alphas.upper() + '_').setParseAction(ChipAttr)
@@ -225,7 +235,6 @@ OR = Keyword('or')
225
235
  BOOL_EXPR = infixNotation(
226
236
  BOOL_STMT,
227
237
  [
228
- (AND, 2, opAssoc.LEFT, BoolAnd),
229
- (OR, 2, opAssoc.LEFT, BoolOr),
238
+ (MatchFirst((AND, OR)), 2, opAssoc.LEFT, BoolOrAnd),
230
239
  ],
231
240
  )
@@ -35,14 +35,21 @@ class IfClause:
35
35
  def __init__(self, stmt: str, temporary: bool = False, reason: t.Optional[str] = None) -> None:
36
36
  try:
37
37
  self.stmt: BoolStmt = BOOL_EXPR.parseString(stmt)[0]
38
- except ParseException:
39
- raise InvalidIfClause(f'Invalid if statement: {stmt}')
38
+ except (ParseException, InvalidIfClause) as ex:
39
+ raise InvalidIfClause(f'Invalid if clause: {stmt}. {ex}')
40
40
 
41
41
  self.temporary = temporary
42
42
  self.reason = reason
43
43
 
44
44
  if self.temporary is True and not self.reason:
45
- raise InvalidIfClause('"reason" must be set when "temporary: true"')
45
+ raise InvalidIfClause(
46
+ f'Invalid if clause "{stmt}". '
47
+ f'"reason" must be set when "temporary: true". '
48
+ f'For example:\n'
49
+ f' - if: {stmt}\n'
50
+ f' temporary: true\n'
51
+ f' reason: lack of ci runners'
52
+ )
46
53
 
47
54
  def get_value(self, target: str, config_name: str) -> t.Any:
48
55
  return self.stmt.get_value(target, config_name)
@@ -249,6 +256,7 @@ class Manifest:
249
256
 
250
257
  rules: t.List[FolderRule] = []
251
258
  for path in paths:
259
+ LOGGER.debug('Loading manifest file %s', path)
252
260
  _manifest = cls.from_file(path, root_path=root_path)
253
261
 
254
262
  for rule in _manifest.rules:
@@ -344,11 +352,14 @@ class Manifest:
344
352
  return x
345
353
 
346
354
  for folder, sha_value in recorded__rel_folder__sha__dict.items():
355
+ # removed
347
356
  if folder not in self__rel_folder__sha__dict:
348
357
  diff_folders.add(_path(folder))
358
+ # modified
349
359
  elif sha_value != self__rel_folder__sha__dict[folder]:
350
360
  diff_folders.add(_path(folder))
351
361
 
362
+ # new
352
363
  for folder in self__rel_folder__sha__dict:
353
364
  if folder not in recorded__rel_folder__sha__dict:
354
365
  diff_folders.add(_path(folder))
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: idf-build-apps
3
- Version: 2.5.0rc2
3
+ Version: 2.5.2
4
4
  Summary: Tools for building ESP-IDF related apps.
5
5
  Author-email: Fu Hanxi <fuhanxi@espressif.com>
6
6
  Requires-Python: >=3.7
@@ -1,27 +1,27 @@
1
- idf_build_apps/__init__.py,sha256=qNCNTH1dk1j7euO_o2AVw2usSipP9euNQd2j5_sDuY4,653
1
+ idf_build_apps/__init__.py,sha256=_czizzLpsxKDZG61HbrIaZjdeib-_E4ZqTMOcxbxNxo,650
2
2
  idf_build_apps/__main__.py,sha256=8E-5xHm2MlRun0L88XJleNh5U50dpE0Q1nK5KqomA7I,182
3
3
  idf_build_apps/app.py,sha256=F-MKOsaz7cJ0H2wsEE4gpO4kkkEdkyFmIZBHDoM2qgs,37359
4
- idf_build_apps/args.py,sha256=bNVrFsB3V1vhukNGgF3YOL-92GHqQZF3rOkJhqIDDvU,29627
4
+ idf_build_apps/args.py,sha256=L1r7iS3gvV5UDqaZpu9M4Y36LUp-m-BK0rOqfpJjXu4,31918
5
5
  idf_build_apps/autocompletions.py,sha256=g-bx0pzXoFKI0VQqftkHyGVWN6MLjuFOdozeuAf45yo,2138
6
6
  idf_build_apps/constants.py,sha256=07ve2FtWuLBuc_6LFzbs1XncB1VNi9HJUqGjQQauRNM,3952
7
- idf_build_apps/finder.py,sha256=kfZaGWJfPUwWAbaOj_W3Fu97SIIFEsv1R_dJucjbFHw,5691
7
+ idf_build_apps/finder.py,sha256=Wv0QYArXmVZZDslLxfPwGgt202s94112AlAHk0s3N1g,5710
8
8
  idf_build_apps/log.py,sha256=pyvT7N4MWzGjIXph5mThQCGBiSt53RNPW0WrFfLr0Kw,2650
9
- idf_build_apps/main.py,sha256=OeWzXkA-68MIv6MmgBN_8XhDJQUvYjJiYxR5zkdQiT0,16094
9
+ idf_build_apps/main.py,sha256=Z_hetbOavgCJZQPaP01_jx57fR1w0DefiFz0J2cCwp0,16498
10
10
  idf_build_apps/session_args.py,sha256=2WDTy40IFAc0KQ57HaeBcYj_k10eUXRKkDOWLrFCaHY,2985
11
11
  idf_build_apps/utils.py,sha256=s4D8P7QA17XcaCUQ_EoiNOW_VpU3cPQgiZVV9KQ8I30,10171
12
12
  idf_build_apps/junit/__init__.py,sha256=IxvdaS6eSXp7kZxRuXqyZyGxuA_A1nOW1jF1HMi8Gns,231
13
13
  idf_build_apps/junit/report.py,sha256=T7dVU3Sz5tqjfbcFW7wjsb65PDH6C2HFf73ePJqBhMs,6555
14
14
  idf_build_apps/junit/utils.py,sha256=j0PYhFTZjXtTwkENdeL4bFJcX24ktf1CsOOVXz65yNo,1297
15
15
  idf_build_apps/manifest/__init__.py,sha256=Q2-cb3ngNjnl6_zWhUfzZZB10f_-Rv2JYNck3Lk7UkQ,133
16
- idf_build_apps/manifest/if_parser.py,sha256=r0pivV9gmniPn3Ia6sTMbW5tFAKInhOXk-Lfd6GokqE,6381
17
- idf_build_apps/manifest/manifest.py,sha256=5BekZkLutVrpe6HaUYeBiNqbxkRAFjQpUo9hG1eN1zg,14090
16
+ idf_build_apps/manifest/if_parser.py,sha256=AAEyYPgcBWL2ToCmcvhx3fl8ebYdC5-LMvvDkYtWn8o,6546
17
+ idf_build_apps/manifest/manifest.py,sha256=6F6Nt93eaoBOHlGHZtv2hHT8Zw0HmJf0d_MESlgye6M,14478
18
18
  idf_build_apps/manifest/soc_header.py,sha256=PzJ37xFspt5f0AXWvAFNA_avHZA9fMXHBrwDYLi3qEI,4344
19
19
  idf_build_apps/vendors/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
20
20
  idf_build_apps/vendors/pydantic_sources.py,sha256=2IN6opo6qjCwaqhERFbgA4PtEwqKaTtEkccy5_fYAT0,4130
21
21
  idf_build_apps/yaml/__init__.py,sha256=W-3z5no07RQ6eYKGyOAPA8Z2CLiMPob8DD91I4URjrA,162
22
22
  idf_build_apps/yaml/parser.py,sha256=b3LvogO6do-eJPRsYzT-8xk8AT2MnXpLCzQutJqyC7M,2128
23
- idf_build_apps-2.5.0rc2.dist-info/entry_points.txt,sha256=3pVUirUEsb6jsDRikkQWNUt4hqLK2ci1HvW_Vf8b6uE,59
24
- idf_build_apps-2.5.0rc2.dist-info/LICENSE,sha256=z8d0m5b2O9McPEK1xHG_dWgUBT6EfBDz6wA0F7xSPTA,11358
25
- idf_build_apps-2.5.0rc2.dist-info/WHEEL,sha256=EZbGkh7Ie4PoZfRQ8I0ZuP9VklN_TvcZ6DSE5Uar4z4,81
26
- idf_build_apps-2.5.0rc2.dist-info/METADATA,sha256=i65JKj_Ul2N4ItjFbehykUlkzxKyjy1hSudWtnSPliw,4613
27
- idf_build_apps-2.5.0rc2.dist-info/RECORD,,
23
+ idf_build_apps-2.5.2.dist-info/entry_points.txt,sha256=3pVUirUEsb6jsDRikkQWNUt4hqLK2ci1HvW_Vf8b6uE,59
24
+ idf_build_apps-2.5.2.dist-info/LICENSE,sha256=z8d0m5b2O9McPEK1xHG_dWgUBT6EfBDz6wA0F7xSPTA,11358
25
+ idf_build_apps-2.5.2.dist-info/WHEEL,sha256=EZbGkh7Ie4PoZfRQ8I0ZuP9VklN_TvcZ6DSE5Uar4z4,81
26
+ idf_build_apps-2.5.2.dist-info/METADATA,sha256=ZegY4svIIuKIUgCFWJdgiiMIZzhSzLWRbOCiwBAVoKo,4610
27
+ idf_build_apps-2.5.2.dist-info/RECORD,,