hh-applicant-tool 0.2.0__tar.gz → 0.2.1__tar.gz

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 hh-applicant-tool might be problematic. Click here for more details.

Files changed (23) hide show
  1. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/PKG-INFO +13 -12
  2. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/README.md +12 -11
  3. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/api/errors.py +7 -3
  4. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/apply_similar.py +2 -2
  5. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/call_api.py +6 -4
  6. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/clear_negotiations.py +2 -3
  7. hh_applicant_tool-0.2.1/hh_applicant_tool/operations/refresh_token.py +42 -0
  8. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/update_resumes.py +2 -2
  9. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/whoami.py +2 -2
  10. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/utils.py +4 -9
  11. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/pyproject.toml +1 -1
  12. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/__init__.py +0 -0
  13. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/__main__.py +0 -0
  14. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/api/__init__.py +0 -0
  15. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/api/client.py +0 -0
  16. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/color_log.py +0 -0
  17. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/constants.py +0 -0
  18. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/main.py +0 -0
  19. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/__init__.py +0 -0
  20. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/add_handler.py +0 -0
  21. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/authorize.py +0 -0
  22. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/operations/list_resumes.py +0 -0
  23. {hh_applicant_tool-0.2.0 → hh_applicant_tool-0.2.1}/hh_applicant_tool/types.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: hh-applicant-tool
3
- Version: 0.2.0
3
+ Version: 0.2.1
4
4
  Summary:
5
5
  Author: Senior YAML Developer
6
6
  Author-email: yamldeveloper@proton.me
@@ -110,16 +110,17 @@ https://hh.ru/employer/1918903
110
110
  - `-v` используется для вывода отладочной информации. Два таких флага, например, выводят запросы к **API**.
111
111
  - `-c <path>` можно создать путь до конфига. С помощью этого флага можно одновременно использовать несколько профилей.
112
112
 
113
- | Операция | Описание |
114
- | ---------------------- | --------------------------------------------------------------------------------------------------------------------------------------------- |
115
- | **add-handler** | Добавляет обработчик протокола `hhandroid` |
116
- | **authorize** | Открывает сайт hh.ru для авторизации и перехватывает перенаправление на `hhadnroid://oauthresponse` |
117
- | **whoami** | Выводит информацию об авторизованном пользователе |
118
- | **list-resumes** | Список резюме |
119
- | **update-resumes** | Обновить все резюме. Аналогично нажатию кнопки «Обновить дату». |
120
- | **apply-similar** | Откликнуться на все подходящие вакансии. Лимит = 200 в день |
121
- | **clear-negotiations** | Удаляет отказы и отменяет заявки, которые долго висят |
122
- | **call-api** | Вызвать произвольный метод API м вывести результат. Я эту команду придумал чтобы собирать ссылки на сайты и загонять их в сканер уязвимостей. |
113
+ | Операция | Описание |
114
+ | ---------------------- | --------------------------------------------------------------------------------------------------- |
115
+ | **add-handler** | Добавляет обработчик протокола `hhandroid` |
116
+ | **authorize** | Открывает сайт hh.ru для авторизации и перехватывает перенаправление на `hhadnroid://oauthresponse` |
117
+ | **whoami** | Выводит информацию об авторизованном пользователе |
118
+ | **list-resumes** | Список резюме |
119
+ | **update-resumes** | Обновить все резюме. Аналогично нажатию кнопки «Обновить дату». |
120
+ | **apply-similar** | Откликнуться на все подходящие вакансии. Лимит = 200 в день |
121
+ | **clear-negotiations** | Удаляет отказы и отменяет заявки, которые долго висят |
122
+ | **call-api** | Вызов произвольного метода API с вводом результата. |
123
+ | **refresh-token** | Обновляет access_token. |
123
124
 
124
125
  Для начала нужно добавить обработчик протокола `hhandroid`, который используется Android-приложением для усложнения жизни честным автоматизаторам:
125
126
 
@@ -223,7 +224,7 @@ datetime.datetime(2023, 3, 23, 6, 36, 15, 596290)
223
224
  >>>
224
225
  ```
225
226
 
226
- После нужно авторизоваться по новой.
227
+ После нужно вызвать `refresh-token`.
227
228
 
228
229
  ![](https://user-images.githubusercontent.com/12753171/222870516-b29f2417-d11a-4122-8291-7d440a422a31.png)
229
230
 
@@ -96,16 +96,17 @@ https://hh.ru/employer/1918903
96
96
  - `-v` используется для вывода отладочной информации. Два таких флага, например, выводят запросы к **API**.
97
97
  - `-c <path>` можно создать путь до конфига. С помощью этого флага можно одновременно использовать несколько профилей.
98
98
 
99
- | Операция | Описание |
100
- | ---------------------- | --------------------------------------------------------------------------------------------------------------------------------------------- |
101
- | **add-handler** | Добавляет обработчик протокола `hhandroid` |
102
- | **authorize** | Открывает сайт hh.ru для авторизации и перехватывает перенаправление на `hhadnroid://oauthresponse` |
103
- | **whoami** | Выводит информацию об авторизованном пользователе |
104
- | **list-resumes** | Список резюме |
105
- | **update-resumes** | Обновить все резюме. Аналогично нажатию кнопки «Обновить дату». |
106
- | **apply-similar** | Откликнуться на все подходящие вакансии. Лимит = 200 в день |
107
- | **clear-negotiations** | Удаляет отказы и отменяет заявки, которые долго висят |
108
- | **call-api** | Вызвать произвольный метод API м вывести результат. Я эту команду придумал чтобы собирать ссылки на сайты и загонять их в сканер уязвимостей. |
99
+ | Операция | Описание |
100
+ | ---------------------- | --------------------------------------------------------------------------------------------------- |
101
+ | **add-handler** | Добавляет обработчик протокола `hhandroid` |
102
+ | **authorize** | Открывает сайт hh.ru для авторизации и перехватывает перенаправление на `hhadnroid://oauthresponse` |
103
+ | **whoami** | Выводит информацию об авторизованном пользователе |
104
+ | **list-resumes** | Список резюме |
105
+ | **update-resumes** | Обновить все резюме. Аналогично нажатию кнопки «Обновить дату». |
106
+ | **apply-similar** | Откликнуться на все подходящие вакансии. Лимит = 200 в день |
107
+ | **clear-negotiations** | Удаляет отказы и отменяет заявки, которые долго висят |
108
+ | **call-api** | Вызов произвольного метода API с вводом результата. |
109
+ | **refresh-token** | Обновляет access_token. |
109
110
 
110
111
  Для начала нужно добавить обработчик протокола `hhandroid`, который используется Android-приложением для усложнения жизни честным автоматизаторам:
111
112
 
@@ -209,7 +210,7 @@ datetime.datetime(2023, 3, 23, 6, 36, 15, 596290)
209
210
  >>>
210
211
  ```
211
212
 
212
- После нужно авторизоваться по новой.
213
+ После нужно вызвать `refresh-token`.
213
214
 
214
215
  ![](https://user-images.githubusercontent.com/12753171/222870516-b29f2417-d11a-4122-8291-7d440a422a31.png)
215
216
 
@@ -1,7 +1,7 @@
1
- #from copy import deepcopy
1
+ # from copy import deepcopy
2
2
  from typing import Any
3
3
 
4
- from requests import Response, Request
4
+ from requests import Request, Response
5
5
  from requests.adapters import CaseInsensitiveDict
6
6
 
7
7
  __all__ = (
@@ -20,7 +20,11 @@ class ApiError(Exception):
20
20
  def __init__(self, response: Response, data: dict[str, Any]) -> None:
21
21
  self._response = response
22
22
  self._raw = data
23
-
23
+
24
+ @property
25
+ def data(self) -> dict:
26
+ return self._raw
27
+
24
28
  @property
25
29
  def request(self) -> Request:
26
30
  return self._response.request
@@ -8,7 +8,7 @@ from ..api import ApiClient, ApiError, BadRequest
8
8
  from ..main import BaseOperation
9
9
  from ..main import Namespace as BaseNamespace
10
10
  from ..types import ApiListResponse, VacancyItem
11
- from ..utils import truncate_string
11
+ from ..utils import print_err, truncate_string
12
12
 
13
13
  logger = logging.getLogger(__package__)
14
14
 
@@ -118,7 +118,7 @@ class Operation(BaseOperation):
118
118
  ")",
119
119
  )
120
120
  except ApiError as ex:
121
- logger.warning(ex)
121
+ print_err("❗ Ошибка:", ex)
122
122
  if isinstance(ex, BadRequest) and ex.limit_exceeded:
123
123
  break
124
124
  print("📝 Отклики на вакансии разосланы!")
@@ -1,11 +1,12 @@
1
1
  # Этот модуль можно использовать как образец для других
2
2
  import argparse
3
+ import json
3
4
  import logging
5
+ import sys
4
6
 
5
7
  from ..api import ApiClient, ApiError
6
8
  from ..main import BaseOperation
7
9
  from ..main import Namespace as BaseNamespace
8
- from ..utils import dumps
9
10
 
10
11
  logger = logging.getLogger(__package__)
11
12
 
@@ -14,6 +15,7 @@ class Namespace(BaseNamespace):
14
15
  method: str
15
16
  endpoint: str
16
17
  params: list[str]
18
+ pretty_print: bool
17
19
 
18
20
 
19
21
  class Operation(BaseOperation):
@@ -24,7 +26,7 @@ class Operation(BaseOperation):
24
26
  parser.add_argument(
25
27
  "param",
26
28
  nargs="*",
27
- help="PARAM=VALUE. Значения можно оборачивать в кавычки.",
29
+ help="PARAM=VALUE",
28
30
  default=[],
29
31
  )
30
32
  parser.add_argument(
@@ -40,7 +42,7 @@ class Operation(BaseOperation):
40
42
  params = dict(x.split("=", 1) for x in args.param)
41
43
  try:
42
44
  result = api.request(args.method, args.endpoint, params=params)
43
- print(dumps(result))
45
+ print(json.dumps(result, ensure_ascii=True))
44
46
  except ApiError as ex:
45
- logger.warning(ex)
47
+ json.dump(ex.data, sys.stderr, ensure_ascii=True)
46
48
  return 1
@@ -8,7 +8,7 @@ from ..constants import INVALID_ISO8601_FORMAT
8
8
  from ..main import BaseOperation
9
9
  from ..main import Namespace as BaseNamespace
10
10
  from ..types import ApiListResponse
11
- from ..utils import truncate_string
11
+ from ..utils import print_err, truncate_string
12
12
 
13
13
  logger = logging.getLogger(__package__)
14
14
 
@@ -100,6 +100,5 @@ class Operation(BaseOperation):
100
100
  ")",
101
101
  )
102
102
  except ClientError as ex:
103
- logger.warning(ex)
104
-
103
+ print_err("❗ Ошибка:", ex)
105
104
  print("🧹 Чистка заявок завершена!")
@@ -0,0 +1,42 @@
1
+ # Этот модуль можно использовать как образец для других
2
+ import argparse
3
+ import logging
4
+
5
+ from ..api import ApiError, OAuthClient
6
+ from ..main import BaseOperation
7
+ from ..main import Namespace as BaseNamespace
8
+ from ..utils import print_err
9
+
10
+ logger = logging.getLogger(__package__)
11
+
12
+
13
+ class Namespace(BaseNamespace):
14
+ pass
15
+
16
+
17
+ class Operation(BaseOperation):
18
+ """Получает новый access_token."""
19
+
20
+ def setup_parser(self, parser: argparse.ArgumentParser) -> None:
21
+ pass
22
+
23
+ def run(self, args: Namespace) -> None:
24
+ if (
25
+ not args.config["token"]
26
+ or not args.config["token"]["refresh_token"]
27
+ ):
28
+ print_err("❗ Необходим refresh_token!")
29
+ return 1
30
+ try:
31
+ oauth = OAuthClient(
32
+ user_agent=(
33
+ args.config["oauth_user_agent"]
34
+ or args.config["user_agent"]
35
+ ),
36
+ )
37
+ token = oauth.refresh_access(args.config["token"]["refresh_token"])
38
+ args.config.save(token=token)
39
+ print("✅ Токен обновлен!")
40
+ except ApiError as ex:
41
+ print_err("❗ Ошибка:", ex)
42
+ return 1
@@ -6,7 +6,7 @@ from ..api import ApiClient, ApiError
6
6
  from ..main import BaseOperation
7
7
  from ..main import Namespace as BaseNamespace
8
8
  from ..types import ApiListResponse
9
- from ..utils import truncate_string
9
+ from ..utils import print_err, truncate_string
10
10
 
11
11
  logger = logging.getLogger(__package__)
12
12
 
@@ -34,4 +34,4 @@ class Operation(BaseOperation):
34
34
  assert res == {}
35
35
  print("✅ Обновлено", truncate_string(resume["title"]))
36
36
  except ApiError as ex:
37
- logger.warning(ex)
37
+ print_err("❗ Ошибка:", ex)
@@ -1,11 +1,11 @@
1
1
  # Этот модуль можно использовать как образец для других
2
2
  import argparse
3
+ import json
3
4
  import logging
4
5
 
5
6
  from ..api import ApiClient
6
7
  from ..main import BaseOperation
7
8
  from ..main import Namespace as BaseNamespace
8
- from ..utils import dumps
9
9
 
10
10
  logger = logging.getLogger(__package__)
11
11
 
@@ -27,4 +27,4 @@ class Operation(BaseOperation):
27
27
  user_agent=args.config["user_agent"],
28
28
  )
29
29
  result = api.get("/me")
30
- print(dumps(result))
30
+ print(json.dumps(result, ensure_ascii=True, indent=2, sort_keys=True))
@@ -9,13 +9,6 @@ from typing import Any
9
9
 
10
10
  print_err = partial(print, file=sys.stderr)
11
11
 
12
- json_dump_kwargs = dict(
13
- indent=2, ensure_ascii=False, sort_keys=True, default=str
14
- )
15
-
16
- dump = partial(json.dump, **json_dump_kwargs)
17
- dumps = partial(json.dumps, **json_dump_kwargs)
18
-
19
12
 
20
13
  class AttrDict(dict):
21
14
  __getattr__ = dict.get
@@ -43,8 +36,10 @@ class Config(dict):
43
36
  self.update(*args, **kwargs)
44
37
  self._config_path.parent.mkdir(exist_ok=True, parents=True)
45
38
  with self._lock:
46
- with self._config_path.open("w+") as f:
47
- dump(self, f)
39
+ with self._config_path.open("w+") as fp:
40
+ json.dump(
41
+ self, fp, ensure_ascii=True, indent=2, sort_keys=True
42
+ )
48
43
 
49
44
  __getitem__ = dict.get
50
45
 
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "hh-applicant-tool"
3
- version = "0.2.0"
3
+ version = "0.2.1"
4
4
  description = ""
5
5
  authors = ["Senior YAML Developer <yamldeveloper@proton.me>"]
6
6
  readme = "README.md"