Compare commits

14 Commits
main ... main

4 changed files with 97 additions and 5 deletions

View File

@@ -0,0 +1,82 @@
name: Android Test
on:
pull_request:
branches: [ main ]
jobs:
validate-and-test:
runs-on: android
steps:
- name: Check ANDROID_SERIAL is set
run: |
if [ -z "$ANDROID_SERIAL" ]; then
echo "❌ Ошибка: Переменная окружения ANDROID_SERIAL не установлена."
exit 1
fi
echo "✅ Переменная ANDROID_SERIAL установлена: $ANDROID_SERIAL"
- name: Check ADB device is connected
run: |
command -v adb >/dev/null 2>&1 || { echo "❌ Ошибка: adb не найден в PATH." >&2; exit 1; }
connected_devices=$(adb devices | grep -c "$ANDROID_SERIAL.*device$")
if [ "$connected_devices" -ne 1 ]; then
echo "❌ Ошибка: Устройство с серийным номером $ANDROID_SERIAL не подключено или не доступно."
echo "Доступные устройства:"
adb devices
exit 1
fi
echo "✅ Устройство с серийным номером $ANDROID_SERIAL подключено и готово к использованию."
- name: Checkout PR code
uses: actions/checkout@v4
with:
fetch-depth: 0
submodules: recursive
- name: Validate allowed changes
run: |
python3 /opt/scripts/validate-changes.py
env:
GITEA_REPOSITORY: ${{ gitea.repository }}
GITEA_BASE_REF: ${{ gitea.event.pull_request.base.ref }}
GITEA_HEAD_REF: ${{ gitea.event.pull_request.head.ref }}
- name: Checkout tests
run: python3 /opt/scripts/copy-tests.py --repo-url "Olympic/NTO-2025-Client-Android-tests" --branch "main"
- name: Uninstall APK before tests (Gradle)
run: |
chmod +x ./gradlew
./gradlew uninstallAll
- name: Run tests
run: |
chmod +x ./gradlew
./gradlew connectedDebugAndroidTest
- name: Uninstall APK after tests (Gradle)
run: |
chmod +x ./gradlew
./gradlew uninstallAll
if: always()
- name: Upload test results
uses: christopherHX/gitea-upload-artifact@v4
with:
name: test-results
path: app/build/outputs/androidTest-results/connected
if-no-files-found: ignore
retention-days: 30
if: always()
- name: Upload test reports
uses: christopherHX/gitea-upload-artifact@v4
with:
name: test-reports
path: app/build/reports/androidTests/connected
if-no-files-found: ignore
retention-days: 30
if: always()

View File

@@ -1,4 +1,4 @@
# НТО 2025. II отборочный этап. Индивдуальное задание — Android # НТО 2025. II отборочный этап. Индивидуальные задания — Android
## 📖 Предыстория ## 📖 Предыстория
В компании S есть возможность бронирования мест в пространствах, предназначенных под общее использование (open-space). На данный момент для бронирования места используются различные способы бронирования, разработанные в каждом офисе индивидуально. В компании S есть возможность бронирования мест в пространствах, предназначенных под общее использование (open-space). На данный момент для бронирования места используются различные способы бронирования, разработанные в каждом офисе индивидуально.
@@ -23,7 +23,7 @@
Поскольку проект только на начальной стадии развития и сервер ещё не был опубликован на внешнем ресурсе, тебе предстоит развернуть его локально. Ссылка на репозиторий с [бэкендом](https://git.sicampus.ru/Olympic/NTO-2025-Client-Android-backend). Рекомендуется открывать в IntelijIDEA. Поскольку проект только на начальной стадии развития и сервер ещё не был опубликован на внешнем ресурсе, тебе предстоит развернуть его локально. Ссылка на репозиторий с [бэкендом](https://git.sicampus.ru/Olympic/NTO-2025-Client-Android-backend). Рекомендуется открывать в IntelijIDEA.
Отметим, что при доработке проекта ui слой и доменные сущности изменять не нужно. Отметим, что при доработке проекта ui слой и доменные сущности изменять не нужно. При решении заданий настоятельно рекомендуем соблюдать [Clean Architecture](https://blog.cleancoder.com/uncle-bob/2012/08/13/the-clean-architecture.html).
### 1. Реализация основного запроса ### 1. Реализация основного запроса
@@ -40,7 +40,7 @@
- если код статуса ответа равен 2xx, то необходимо считать тело ответа и вернуть полученные данные; - если код статуса ответа равен 2xx, то необходимо считать тело ответа и вернуть полученные данные;
- если код статуса ответа не равен 2xx, то необходимо считать тело ответа, содержащее информацию об ошибке, извлечь из него сообщение и на уровне доменной модели вернуть `Result.Failure` (как выполнить мапинг - решаешь ты, в т.ч. можно использовать конструкцию `runCatching { … }`), у которого это сообщение передаётся в свойство message для дальнейшего отображения в интерфейсе. - если код статуса ответа не равен 2xx, то необходимо считать тело ответа, содержащее информацию об ошибке, извлечь из него сообщение и на уровне доменной модели вернуть `Result.Failure` (как выполнить мапинг - решаешь ты, в т.ч. можно использовать конструкцию `runCatching { … }`), у которого это сообщение передаётся в свойство message для дальнейшего отображения в интерфейсе.
Для тестирования можете модифицровать тестовый сервер, все необхоидмые сущности для этого уже созданы. Для тестирования можете модифицировать тестовый сервер, все необходимые сущности для этого уже созданы.
## ✅ Особенности оценивания ## ✅ Особенности оценивания

View File

@@ -1,4 +1,14 @@
version: 1 version: 1
allowed_files: allowed_files:
- "app/src/main/java/ru/myitschool/work/*" - "app/src/main/java/ru/myitschool/work/*"
- "app/src/main/res/*" - "app/src/main/res/*"
forbidden_files:
- "app/src/main/java/ru/myitschool/work/domain/entities/BookingEntity.kt"
- "app/src/main/java/ru/myitschool/work/domain/entities/UserEntity.kt"
- "app/src/main/java/ru/myitschool/work/ui/theme/Color.kt"
- "app/src/main/java/ru/myitschool/work/ui/theme/Theme.kt"
- "app/src/main/java/ru/myitschool/work/ui/theme/Type.kt"
- "app/src/main/java/ru/myitschool/work/ui/root/RootActivity.kt"
- "app/src/main/java/ru/myitschool/work/ui/root/RootIntent.kt"
- "app/src/main/java/ru/myitschool/work/ui/root/RootState.kt"
- "app/src/main/java/ru/myitschool/work/ui/root/RootViewModel.kt"