Skip to content

Implement Bitrix24Partners support (issue #70)#109

Open
KarlsonComplete wants to merge 33 commits into
mesilov:claude/bitrix24-issue-70-01PSzKghSeJKibfs1z3wb8sXfrom
KarlsonComplete:claude/bitrix24-issue-70-01PSzKghSeJKibfs1z3wb8sX
Open

Implement Bitrix24Partners support (issue #70)#109
KarlsonComplete wants to merge 33 commits into
mesilov:claude/bitrix24-issue-70-01PSzKghSeJKibfs1z3wb8sXfrom
KarlsonComplete:claude/bitrix24-issue-70-01PSzKghSeJKibfs1z3wb8sX

Conversation

@KarlsonComplete
Copy link
Copy Markdown
Collaborator

@KarlsonComplete KarlsonComplete commented Apr 19, 2026

This commit implements comprehensive support for Bitrix24Partners following the established DDD/CQRS patterns in the codebase.

Features Implemented:

Core Domain

  • Bitrix24Partner entity with full interface implementation
  • Repository with all required query methods
  • Doctrine XML mapping with unique constraint on bitrix24PartnerId
  • PhoneNumber Doctrine custom type for phone number storage

Use Cases (CQRS)

  • Create: Create new partner records
  • Update: Update partner information
  • Delete: Soft-delete partners (mark as deleted)
  • MarkAsBlocked: Block partner accounts
  • MarkAsActive: Reactivate blocked partners

CLI Tools

  • ScrapePartnersCommand: Web scraper for https://www.bitrix24.ru/partners/
    • Parses partner data from HTML
    • Generates CSV output
    • Configurable URL and output path
  • ImportPartnersCsvCommand: CSV import utility
    • Bulk import partners from CSV files
    • Phone number parsing with libphonenumber
    • Error handling with --skip-errors option

Testing

  • Unit tests for entity (extends SDK contract tests)
  • Unit tests for Create command validation

Technical Details:

  • Follows existing bounded context patterns
  • Uses value objects where appropriate
  • Implements domain events for all state changes
  • Full validation in command constructors
  • Comprehensive error handling

Resolves: #70

Q A
Bug fix? no
New feature? yes
Deprecations? no
Issues Fix #70
License MIT

Copy link
Copy Markdown

@gemini-code-assist gemini-code-assist Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Code Review

This pull request refactors the Bitrix24Partners module by renaming bitrix24PartnerId to bitrix24PartnerNumber and transitioning the repository to a composition-based approach. It also introduces email validation, updates PHP and PHPUnit version requirements, adds a maintainer skill definition, and improves testing infrastructure with a new partner builder. Feedback identifies inconsistent Docker command usage in the Makefile, suggests a more standard PHP version constraint, notes PSR-12 style regressions, recommends better HTML error handling, and identifies a typo in a directory name.

Comment thread Makefile
Comment thread composer.json
Comment thread src/Bitrix24Partners/Console/ScrapePartnersCommand.php Outdated
Comment thread skills/bitrix24-php-lib-maintainer /SKILL.md
Copy link
Copy Markdown
Collaborator

@camaxtly camaxtly left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

19-04-2026

Comment thread config/xml/Bitrix24.Lib.Bitrix24Partners.Entity.Bitrix24Partner.dcm.xml Outdated
Comment thread src/Bitrix24Partners/Entity/Bitrix24Partner.php Outdated
Comment thread src/Bitrix24Partners/Entity/Bitrix24Partner.php Outdated
Comment thread src/Bitrix24Partners/Entity/Bitrix24Partner.php Outdated
Comment thread src/Bitrix24Partners/Entity/Bitrix24Partner.php Outdated
Comment thread src/Bitrix24Partners/Entity/Bitrix24Partner.php
Comment thread src/Bitrix24Partners/UseCase/Delete/Handler.php Outdated
…logic:

- Add `Bitrix24PartnerNotFoundException` handling to UseCase Handlers (`Create`, `Update`, `Delete`, `MarkAsBlocked`, `MarkAsActive`) for better error management and logging.
- Enhance field validation in `Update.Command` to ensure stricter checks.
- Introduce `logoUrl` field in `Bitrix24Partner` entity and adjust related UseCases and tests.
- Refactor tests to include new scenarios and improve data consistency.
- Replace `docker-compose` with `docker compose` in `Makefile`.
- Introduce `bitrix24:partners:scrape:v2` command with updated logic.
- Register `ScrapePartnersCommand_V2` in the console application.
- Integrate Monolog for detailed logging throughout the scraping process.
- Refactor `ScrapePartnersCommand_V2` to enhance parsing logic and improve error handling.
- Introduce support for fetching partner logos and storing results in a CSV file.
- Streamline HTTP requests and implement configurable delays.
…а, изменил сигнатуры методов сущности и добавил поддержку логотипа партнера.
@KarlsonComplete KarlsonComplete changed the title Claude/bitrix24 issue 70 01 p sz kgh se j kibfs1z3wb8s x Implement Bitrix24Partners support (issue #70) Apr 26, 2026
Comment thread src/Bitrix24Partners/Console/ScrapePartnersCommand.php Outdated
Comment thread src/Bitrix24Partners/Console/ScrapePartnersCommand.php
Copy link
Copy Markdown
Collaborator

@camaxtly camaxtly left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

26-04-2026

Comment thread src/Bitrix24Partners/Console/ScrapePartnersCommand.php Outdated
Comment thread src/Bitrix24Partners/Console/ScrapePartnersCommand.php Outdated
Comment thread src/Bitrix24Partners/Console/ScrapePartnersCommand.php Outdated
Comment thread src/Bitrix24Partners/UseCase/Create/Handler.php
Comment thread src/Bitrix24Partners/UseCase/Delete/Handler.php
- Add `Uuid::v7()` support in `Bitrix24Partner` constructors and builders.
- Improve validation for partner creation with `bitrix24PartnerNumber`.
- Update Doctrine mapping for `bitrix24PartnerNumber` to use `unique` attribute.
…erBuilder:

- Remove redundant `withBitrix24PartnerNumber` calls from tests.
- Adjust `Bitrix24PartnerBuilder` to expand possible number range.
- Reorder `clear()` in `Update.HandlerTest` for better test consistency.
…dize database truncation across functional tests.
- Implement stricter non-empty string checks for `site`, `email`, `openLineId`, `externalId`, and `logoUrl` in `Update.Command` and `Bitrix24Partner` entity.
- Update associated tests to cover new validation scenarios.
…s, and refactor `Makefile` and `console` setup.
…aper` services and integrated them into `ScrapePartnersCommandV2` for improved modularity and reusability.
…ScrapePartnersCommandV2: добавлена обработка блокировок, настройки задержек, детектирование пустых страниц и обновлён механизм работы с доменом.
…rtnersCsvCommand, улучшил обработку пустых страниц и блокировок в ScrapePartnersCommandV2, добавил загрузку .env.local.
…er updates, integrate `ScrapeStateManager` into scraping commands, and refactor CSV handling logic for modularity.
…nsole commands, and refactor partner scraping for improved base domain handling and progress tracking.
…r consistency and improved maintainability across infrastructure, use cases, and console commands.
|-------|----------|--------------|
| `--base-url` | URL страницы списка партнёров | `https://www.bitrix24.ru/partners/country__19/` |
| `--output-file` | Путь к выходному CSV файлу | `partners.csv` |
| `--page-delay` | Задержка между страницами (сек) | `2` |
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Отразить какую проблему можно решить

# Кастомный выходной файл и ускоренный парсинг
php bin/console partners:scrape --output-file=partners_kz.csv --page-delay=1 --partner-delay=1

# Продолжить прерванный парсинг
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Нет, пример команды должен быть полный. Тут хорошо пример запуска и пример продолжения.

php bin/console partners:scrape --insecure
```

**Механизм resume:** При прерывании (Ctrl+C) создаётся state-файл `<output>.state.json` с информацией о последней обработанной странице. При запуске с `--resume` парсинг продолжится с этого места.
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Поправить про прерывание


**Механизм resume:** При прерывании (Ctrl+C) создаётся state-файл `<output>.state.json` с информацией о последней обработанной странице. При запуске с `--resume` парсинг продолжится с этого места.

**Обнаружение бана:** Команда автоматически определяет блокировку по двум признакам:
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Бан и продолжение загрузки вынести до команд, как общее описание, что может произойти


**Обязательные колонки CSV:** `title`, `bitrix24_partner_number`

**Опциональные колонки:** `site`, `phone`, `email`, `open_line_id`, `external_id`, `logo_url`
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Зачем поддерживать кастомную структуру файла?

3240,Hoster.KZ,https://b24.kz/,8-727-2-379-284,info@b24.kz,https://.../logo.jpg,/partners/partner/3240/,https://www.bitrix24.kz,2026-05-01T12:27:22+00:00
```

| Колонка | Описание |
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Эти поля должны быть обязательными при импорте.

Copy link
Copy Markdown
Collaborator

@camaxtly camaxtly left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

10-05-2026

Comment thread docs/partner-commands.md Outdated

---

### `partners:update` — Обновление конкретных партнёров
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Переделываем логику работы ---

  1. Можно указать по ИД перечень партнеров, которых надо обновить
  2. Только этих партнеров выгрузаем в файл. Файл и его структура идентична тому, который используется для полной выгрузки.
  3. Дальше используем процедуру импорта и она обновит данные партнеров из файла.


---

## Запуск через Docker (Makefile)
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Актуализровать команды и их описание


---

## Рекомендации
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Рекомендации разнести в разделы, как правильно пользоваться либой.

private const int DEFAULT_PAGE_DELAY = 2;

private const int DEFAULT_PARTNER_DELAY = 2;

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Тут не хватает дефолта для insecure

protected function execute(InputInterface $input, OutputInterface $output): int
{
$io = new SymfonyStyle($input, $output);
$symfonyStyle = new SymfonyStyle($input, $output);
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

назови переменную io, чтоб было одинаково во всех командах, когда ты работает с "разукрашиванием" консоли

/** @var AggregateRootEventsEmitterInterface|Bitrix24PartnerInterface $activePartner */
$activePartner = $this->bitrix24PartnerRepository->findByBitrix24PartnerNumber($command->bitrix24PartnerNumber);

if (null !== $activePartner) {
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

С таким походом у нас этот юзкейз актуален только для 1 импорта, потом 99% будет отваливаться здесь.

use libphonenumber\PhoneNumberUtil;
use Psr\Log\LoggerInterface;

readonly class Handler
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Обхединяем с текущим create, так как нам по сути надо обновить данные в БД из файла и все.
А если партнера нет, то его и добавить.

use Bitrix24\SDK\Application\Contracts\Events\AggregateRootEventsEmitterInterface;
use Psr\Log\LoggerInterface;

readonly class Handler
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Проработать сценарий верхнеуровнено, как и когда будет вызываться этот юзкейс

@@ -0,0 +1,190 @@
# Консольные команды для работы с партнёрами Bitrix24

## Обзор
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Сделлать еще 1 доку и описать сценарии импорта.

  1. пропуск,
  2. обновление данных,
  3. добавление несуществующего и
  4. удаление лишнего.

*
* @return bool true if the Bitrix24Partner are equal, false otherwise
*/
public function equals(Bitrix24PartnerInterface $other): bool
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Если нужно только для тестов, так и оставь в комментах

…ментацию по сценарию импорта партнёров, переработал ScrapePartnersCommand и UpdatePartnersCommand для синхронизации с новой спецификацией.
- Implement `Command` and `Handler` classes for upsert functionality.
- Add validation for required fields in `Upsert\Command`.
- Introduce `Upsert\Handler` to handle create, update, or skip logic based on partner data.
- Update documentation to include new upsert scenarios and sync modes (`full`, `partial`).
- Add unit and functional tests to cover all upsert scenarios, including creation, update, and skip cases.
- Refactor partner import logic to align with the new upsert behavior.
…arser dependency:

- Move PartnerHtmlParser logic into PartnerPageScraper as a dependency.
- Simplify scraper command implementations (`UpdatePartnersCommand`, `ScrapePartnersCommand`) by using the consolidated `fetchPartnerData` method.
- Introduce verbosity-based output control in `ScrapePartnersCommand`.
- Add `PartnerData` DTO to structure scraped partner data.
- Update tests to align with new scraping logic and enhance maintainability.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants