Compare commits
	
		
			1 Commits
		
	
	
		
			master
			...
			9797468e31
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 9797468e31 | 
							
								
								
									
										65
									
								
								.gitea/workflows/common_coverage.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								.gitea/workflows/common_coverage.yml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,65 @@
 | 
			
		||||
name: Coverage
 | 
			
		||||
on:
 | 
			
		||||
  workflow_call:
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  BuildAndTestAndCoverage:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Check out repository code
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
        with:
 | 
			
		||||
          fetch-depth: 0
 | 
			
		||||
      - name: Fetch tags
 | 
			
		||||
        run: git fetch --tags -p
 | 
			
		||||
      - uses: actions/setup-java@v4
 | 
			
		||||
        with:
 | 
			
		||||
          distribution: 'temurin'
 | 
			
		||||
          java-version: '17'
 | 
			
		||||
          cache: gradle
 | 
			
		||||
      - uses: gradle/actions/setup-gradle@v3
 | 
			
		||||
      - uses: android-actions/setup-android@v3
 | 
			
		||||
      - name: Configure gradle...
 | 
			
		||||
        run: mkdir -p ~/.gradle && echo "ignoreGitVersion=true" >> ~/.gradle/gradle.properties
 | 
			
		||||
      - uses: KengoTODA/actions-setup-docker-compose@v1
 | 
			
		||||
        with:
 | 
			
		||||
          version: "2.23.3"
 | 
			
		||||
      - name: run selfoss
 | 
			
		||||
        run: |
 | 
			
		||||
          docker compose -f .gitea/workflows/assets/docker-compose.yml up -d
 | 
			
		||||
      - name: Set env url
 | 
			
		||||
        run: |
 | 
			
		||||
          export SELFOSS_URL=172.17.0.1:8888
 | 
			
		||||
      # https://github.com/ReactiveCircus/android-emulator-runner/issues/385
 | 
			
		||||
      - name: Kill crashpad_handler processes
 | 
			
		||||
        if: always()
 | 
			
		||||
        run: |
 | 
			
		||||
          pkill -SIGTERM crashpad_handler || true
 | 
			
		||||
          sleep 5
 | 
			
		||||
          pkill -SIGKILL crashpad_handler || true
 | 
			
		||||
      - name: Tests
 | 
			
		||||
        uses: reactivecircus/android-emulator-runner@v2
 | 
			
		||||
        with:
 | 
			
		||||
          api-level: 29
 | 
			
		||||
          script: |
 | 
			
		||||
            ./gradlew androidApp:connectedAndroidTest
 | 
			
		||||
            killall -INT crashpad_handler || true
 | 
			
		||||
      - uses: actions/upload-artifact@v3
 | 
			
		||||
        if: failure()
 | 
			
		||||
        with:
 | 
			
		||||
          name: failure-espresso
 | 
			
		||||
          path: build/reports/androidTests/connected/screenshots
 | 
			
		||||
          retention-days: 2
 | 
			
		||||
          overwrite: true
 | 
			
		||||
          include-hidden-files: true
 | 
			
		||||
      - uses: actions/upload-artifact@v3
 | 
			
		||||
        with:
 | 
			
		||||
          name: coverage-espresso
 | 
			
		||||
          path: build/reports/coverage/androidTest/githubConfig/debug/connected
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
          overwrite: true
 | 
			
		||||
          include-hidden-files: true
 | 
			
		||||
      - name: Clean
 | 
			
		||||
        if: always()
 | 
			
		||||
        run: |
 | 
			
		||||
          docker compose -f .gitea/workflows/assets/docker-compose.yml stop
 | 
			
		||||
@@ -3,89 +3,148 @@ on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
    branches:
 | 
			
		||||
      - master
 | 
			
		||||
      - chore-crowdin-ci
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  Lint:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Check out repository code
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
      - uses: actions/setup-java@v4
 | 
			
		||||
        with:
 | 
			
		||||
          distribution: 'temurin'
 | 
			
		||||
          java-version: '17'
 | 
			
		||||
          cache: gradle
 | 
			
		||||
      - name: Install klint
 | 
			
		||||
        run: curl -sSLO https://github.com/pinterest/ktlint/releases/download/1.5.0/ktlint && chmod a+x ktlint && mv ktlint /usr/local/bin/
 | 
			
		||||
      - name: Install detekt
 | 
			
		||||
        run: curl -sSLO https://github.com/detekt/detekt/releases/download/v1.23.7/detekt-cli-1.23.7.zip && unzip detekt-cli-1.23.7.zip
 | 
			
		||||
      - name: Linting...
 | 
			
		||||
        run: ktlint 'shared/**/*.kt' 'androidApp/**/*.kt' '!shared/build'
 | 
			
		||||
      - name: Detecting...
 | 
			
		||||
        run: ./detekt-cli-1.23.7/bin/detekt-cli -c detekt.yml --excludes '**/shared/build/**/*.kt'
 | 
			
		||||
  translations:
 | 
			
		||||
  BuildAndTestAndCoverage:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Check out repository code
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
        with:
 | 
			
		||||
          fetch-depth: 0
 | 
			
		||||
      - name: "Check translations changes"
 | 
			
		||||
        id: check-translations-changes
 | 
			
		||||
        uses: tj-actions/changed-files@v45
 | 
			
		||||
      - name: Fetch tags
 | 
			
		||||
        run: git fetch --tags -p
 | 
			
		||||
      - uses: actions/setup-java@v4
 | 
			
		||||
        with:
 | 
			
		||||
          files: |
 | 
			
		||||
            androidApp/src/main/res/values/strings.xml
 | 
			
		||||
      - name: upload translation sources
 | 
			
		||||
        if: steps.check-api-changes.outputs.any_modified == 'true'
 | 
			
		||||
        uses: crowdin/github-action@v2
 | 
			
		||||
          distribution: 'temurin'
 | 
			
		||||
          java-version: '17'
 | 
			
		||||
      - uses: gradle/actions/setup-gradle@v3
 | 
			
		||||
      - uses: android-actions/setup-android@v3
 | 
			
		||||
      - name: Configure gradle...
 | 
			
		||||
        run: mkdir -p ~/.gradle && echo "ignoreGitVersion=true" >> ~/.gradle/gradle.properties
 | 
			
		||||
      - uses: KengoTODA/actions-setup-docker-compose@v1
 | 
			
		||||
        with:
 | 
			
		||||
          config: './.gitea/workflows/assets/crowdin.yml'
 | 
			
		||||
          upload_sources: true
 | 
			
		||||
          upload_translations: false
 | 
			
		||||
          download_translations: false
 | 
			
		||||
          create_pull_request: false
 | 
			
		||||
          push_translations: false
 | 
			
		||||
        env:
 | 
			
		||||
          CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }}
 | 
			
		||||
          CROWDIN_PERSONAL_TOKEN: ${{ secrets.CROWDIN_PERSONAL_TOKEN }}
 | 
			
		||||
      - name: wait
 | 
			
		||||
        if: steps.check-api-changes.outputs.any_modified == 'true'
 | 
			
		||||
        run: sleep 10s
 | 
			
		||||
      - name: download translations
 | 
			
		||||
        if: steps.check-api-changes.outputs.any_modified == 'true'
 | 
			
		||||
        uses: crowdin/github-action@v2
 | 
			
		||||
        with:
 | 
			
		||||
          config: './.gitea/workflows/assets/crowdin.yml'
 | 
			
		||||
          upload_sources: false
 | 
			
		||||
          upload_translations: false
 | 
			
		||||
          download_translations: true
 | 
			
		||||
          create_pull_request: false
 | 
			
		||||
          push_translations: false
 | 
			
		||||
        env:
 | 
			
		||||
          CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }}
 | 
			
		||||
          CROWDIN_PERSONAL_TOKEN: ${{ secrets.CROWDIN_PERSONAL_TOKEN }}
 | 
			
		||||
      - name: Check for uncommitted changes
 | 
			
		||||
        if: steps.check-api-changes.outputs.any_modified == 'true'
 | 
			
		||||
        id: check-changes
 | 
			
		||||
        uses: mskri/check-uncommitted-changes-action@v1.0.1
 | 
			
		||||
      - name: Commit Changes
 | 
			
		||||
        if: steps.check-api-changes.outputs.any_modified == 'true' && steps.check-changes.outputs.changes != ''
 | 
			
		||||
          version: "2.23.3"
 | 
			
		||||
      - name: run selfoss
 | 
			
		||||
        run: |
 | 
			
		||||
          git config --global user.email aminecmi+giteadrone@pm.me
 | 
			
		||||
          git config --global user.name giteadrone
 | 
			
		||||
          git add ./androidApp/src/main/res/*
 | 
			
		||||
          git commit -m "translation: translation files"
 | 
			
		||||
      - name: Push changes
 | 
			
		||||
        if: steps.check-api-changes.outputs.any_modified == 'true' && steps.check-changes.outputs.changes != ''
 | 
			
		||||
        uses: appleboy/git-push-action@v1.0.0
 | 
			
		||||
          docker compose -f .gitea/workflows/assets/docker-compose.yml up -d
 | 
			
		||||
      - name: Set env url
 | 
			
		||||
        run: |
 | 
			
		||||
          export SELFOSS_URL=172.17.0.1:8888
 | 
			
		||||
      # https://github.com/ReactiveCircus/android-emulator-runner/issues/385
 | 
			
		||||
      - name: Kill crashpad_handler processes
 | 
			
		||||
        if: always()
 | 
			
		||||
        run: |
 | 
			
		||||
          pkill -SIGTERM crashpad_handler || true
 | 
			
		||||
          sleep 5
 | 
			
		||||
          pkill -SIGKILL crashpad_handler || true
 | 
			
		||||
      - name: Tests
 | 
			
		||||
        uses: reactivecircus/android-emulator-runner@v2
 | 
			
		||||
        with:
 | 
			
		||||
          author_name: giteadrone
 | 
			
		||||
          author_email: aminecmi+giteadrone@pm.me
 | 
			
		||||
          remote: ${{ secrets.REMOTE_URL }}
 | 
			
		||||
          ssh_key: ${{ secrets.PRIVATE_KEY }}
 | 
			
		||||
          branch: ${{ github.head_ref || github.ref_name }}
 | 
			
		||||
  build:
 | 
			
		||||
    needs: Lint
 | 
			
		||||
    uses: ./.gitea/workflows/common_build.yml
 | 
			
		||||
          api-level: 29
 | 
			
		||||
          script: |
 | 
			
		||||
            ./gradlew androidApp:connectedAndroidTest || true
 | 
			
		||||
            ./gradlew androidApp:fetchScreenshots
 | 
			
		||||
      - uses: actions/upload-artifact@v3
 | 
			
		||||
        if: always()
 | 
			
		||||
        with:
 | 
			
		||||
          name: failure-espresso
 | 
			
		||||
          path: build/reports/androidTests/connected/screenshots
 | 
			
		||||
          retention-days: 2
 | 
			
		||||
          overwrite: true
 | 
			
		||||
          include-hidden-files: true
 | 
			
		||||
      - uses: actions/upload-artifact@v3
 | 
			
		||||
        with:
 | 
			
		||||
          name: coverage-espresso
 | 
			
		||||
          path: build/reports/coverage/androidTest/githubConfig/debug/connected
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
          overwrite: true
 | 
			
		||||
          include-hidden-files: true
 | 
			
		||||
      - name: Clean
 | 
			
		||||
        if: always()
 | 
			
		||||
        run: |
 | 
			
		||||
          docker compose -f .gitea/workflows/assets/docker-compose.yml stop
 | 
			
		||||
 | 
			
		||||
#  Lint:
 | 
			
		||||
#    runs-on: ubuntu-latest
 | 
			
		||||
#    steps:
 | 
			
		||||
#      - name: Check out repository code
 | 
			
		||||
#        uses: actions/checkout@v4
 | 
			
		||||
#      - uses: actions/setup-java@v4
 | 
			
		||||
#        with:
 | 
			
		||||
#          distribution: 'temurin'
 | 
			
		||||
#          java-version: '17'
 | 
			
		||||
#          cache: gradle
 | 
			
		||||
#      - name: Install klint
 | 
			
		||||
#        run: curl -sSLO https://github.com/pinterest/ktlint/releases/download/1.5.0/ktlint && chmod a+x ktlint && mv ktlint /usr/local/bin/
 | 
			
		||||
#      - name: Install detekt
 | 
			
		||||
#        run: curl -sSLO https://github.com/detekt/detekt/releases/download/v1.23.7/detekt-cli-1.23.7.zip && unzip detekt-cli-1.23.7.zip
 | 
			
		||||
#      - name: Linting...
 | 
			
		||||
#        run: ktlint 'shared/**/*.kt' 'androidApp/**/*.kt' '!shared/build'
 | 
			
		||||
#      - name: Detecting...
 | 
			
		||||
#        run: ./detekt-cli-1.23.7/bin/detekt-cli -c detekt.yml --excludes '**/shared/build/**/*.kt'
 | 
			
		||||
#  translations:
 | 
			
		||||
#    runs-on: ubuntu-latest
 | 
			
		||||
#    steps:
 | 
			
		||||
#      - name: Check out repository code
 | 
			
		||||
#        uses: actions/checkout@v4
 | 
			
		||||
#        with:
 | 
			
		||||
#          fetch-depth: 0
 | 
			
		||||
#      - name: "Check translations changes"
 | 
			
		||||
#        id: check-translations-changes
 | 
			
		||||
#        uses: tj-actions/changed-files@v45
 | 
			
		||||
#        with:
 | 
			
		||||
#          files: |
 | 
			
		||||
#            androidApp/src/main/res/values/strings.xml
 | 
			
		||||
#      - name: upload translation sources
 | 
			
		||||
#        if: steps.check-api-changes.outputs.any_modified == 'true'
 | 
			
		||||
#        uses: crowdin/github-action@v2
 | 
			
		||||
#        with:
 | 
			
		||||
#          config: './.gitea/workflows/assets/crowdin.yml'
 | 
			
		||||
#          upload_sources: true
 | 
			
		||||
#          upload_translations: false
 | 
			
		||||
#          download_translations: false
 | 
			
		||||
#          create_pull_request: false
 | 
			
		||||
#          push_translations: false
 | 
			
		||||
#        env:
 | 
			
		||||
#          CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }}
 | 
			
		||||
#          CROWDIN_PERSONAL_TOKEN: ${{ secrets.CROWDIN_PERSONAL_TOKEN }}
 | 
			
		||||
#      - name: wait
 | 
			
		||||
#        if: steps.check-api-changes.outputs.any_modified == 'true'
 | 
			
		||||
#        run: sleep 10s
 | 
			
		||||
#      - name: download translations
 | 
			
		||||
#        if: steps.check-api-changes.outputs.any_modified == 'true'
 | 
			
		||||
#        uses: crowdin/github-action@v2
 | 
			
		||||
#        with:
 | 
			
		||||
#          config: './.gitea/workflows/assets/crowdin.yml'
 | 
			
		||||
#          upload_sources: false
 | 
			
		||||
#          upload_translations: false
 | 
			
		||||
#          download_translations: true
 | 
			
		||||
#          create_pull_request: false
 | 
			
		||||
#          push_translations: false
 | 
			
		||||
#        env:
 | 
			
		||||
#          CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }}
 | 
			
		||||
#          CROWDIN_PERSONAL_TOKEN: ${{ secrets.CROWDIN_PERSONAL_TOKEN }}
 | 
			
		||||
#      - name: Check for uncommitted changes
 | 
			
		||||
#        if: steps.check-api-changes.outputs.any_modified == 'true'
 | 
			
		||||
#        id: check-changes
 | 
			
		||||
#        uses: mskri/check-uncommitted-changes-action@v1.0.1
 | 
			
		||||
#      - name: Commit Changes
 | 
			
		||||
#        if: steps.check-api-changes.outputs.any_modified == 'true' && steps.check-changes.outputs.changes != ''
 | 
			
		||||
#        run: |
 | 
			
		||||
#          git config --global user.email aminecmi+giteadrone@pm.me
 | 
			
		||||
#          git config --global user.name giteadrone
 | 
			
		||||
#          git add ./androidApp/src/main/res/*
 | 
			
		||||
#          git commit -m "translation: translation files"
 | 
			
		||||
#      - name: Push changes
 | 
			
		||||
#        if: steps.check-api-changes.outputs.any_modified == 'true' && steps.check-changes.outputs.changes != ''
 | 
			
		||||
#        uses: appleboy/git-push-action@v1.0.0
 | 
			
		||||
#        with:
 | 
			
		||||
#          author_name: giteadrone
 | 
			
		||||
#          author_email: aminecmi+giteadrone@pm.me
 | 
			
		||||
#          remote: ${{ secrets.REMOTE_URL }}
 | 
			
		||||
#          ssh_key: ${{ secrets.PRIVATE_KEY }}
 | 
			
		||||
#          branch: ${{ github.head_ref || github.ref_name }}
 | 
			
		||||
#  build:
 | 
			
		||||
#    needs: Lint
 | 
			
		||||
#    uses: ./.gitea/workflows/common_build.yml
 | 
			
		||||
 
 | 
			
		||||
@@ -95,7 +95,7 @@ android {
 | 
			
		||||
 | 
			
		||||
        // tests
 | 
			
		||||
        testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
 | 
			
		||||
        testInstrumentationRunnerArguments["clearPackageData"] = "true"
 | 
			
		||||
        testInstrumentationRunnerArguments["useTestStorageService"] = "true"
 | 
			
		||||
    }
 | 
			
		||||
    packaging {
 | 
			
		||||
        resources {
 | 
			
		||||
@@ -109,6 +109,11 @@ android {
 | 
			
		||||
            proguardFiles(getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro")
 | 
			
		||||
        }
 | 
			
		||||
        getByName("debug") {
 | 
			
		||||
            isTestCoverageEnabled = true
 | 
			
		||||
            enableAndroidTestCoverage = true
 | 
			
		||||
            installation {
 | 
			
		||||
                installOptions("-g", "-r")
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    flavorDimensions.add("build")
 | 
			
		||||
@@ -121,7 +126,6 @@ android {
 | 
			
		||||
    namespace = "bou.amine.apps.readerforselfossv2.android"
 | 
			
		||||
    testOptions {
 | 
			
		||||
        animationsDisabled = true
 | 
			
		||||
        execution = "ANDROIDX_TEST_ORCHESTRATOR"
 | 
			
		||||
        unitTests {
 | 
			
		||||
            isIncludeAndroidResources = true
 | 
			
		||||
        }
 | 
			
		||||
@@ -154,8 +158,8 @@ dependencies {
 | 
			
		||||
    implementation("androidx.multidex:multidex:2.0.1")
 | 
			
		||||
 | 
			
		||||
    // About
 | 
			
		||||
    implementation("com.mikepenz:aboutlibraries-core:10.5.1")
 | 
			
		||||
    implementation("com.mikepenz:aboutlibraries:10.5.1")
 | 
			
		||||
    implementation("com.mikepenz:aboutlibraries-core:11.6.3")
 | 
			
		||||
    implementation("com.mikepenz:aboutlibraries:11.6.3")
 | 
			
		||||
 | 
			
		||||
    // Material-ish things
 | 
			
		||||
    implementation("com.ashokvarma.android:bottom-navigation-bar:2.2.0")
 | 
			
		||||
@@ -197,14 +201,15 @@ dependencies {
 | 
			
		||||
    testImplementation("io.mockk:mockk:1.13.14")
 | 
			
		||||
    testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.10.1")
 | 
			
		||||
    implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.6.1")
 | 
			
		||||
    androidTestImplementation("androidx.test:runner:1.6.2")
 | 
			
		||||
    androidTestImplementation("androidx.test:rules:1.6.1")
 | 
			
		||||
    androidTestImplementation("androidx.test:runner:1.7.0-alpha01")
 | 
			
		||||
    androidTestImplementation("androidx.test:rules:1.7.0-alpha01")
 | 
			
		||||
    androidTestImplementation("androidx.test.espresso:espresso-core:3.6.1")
 | 
			
		||||
    implementation("androidx.test.espresso:espresso-idling-resource:3.6.1")
 | 
			
		||||
    androidTestImplementation("androidx.test.ext:junit-ktx:1.2.1")
 | 
			
		||||
    androidTestUtil("androidx.test:orchestrator:1.5.1")
 | 
			
		||||
    androidTestUtil("androidx.test.services:test-services:1.6.0-alpha02")
 | 
			
		||||
    testImplementation("org.robolectric:robolectric:4.14.1")
 | 
			
		||||
    testImplementation("androidx.test:core-ktx:1.6.1")
 | 
			
		||||
    testImplementation("androidx.test:core-ktx:1.7.0-alpha01")
 | 
			
		||||
    androidTestImplementation("androidx.test.uiautomator:uiautomator:2.3.0")
 | 
			
		||||
 | 
			
		||||
    implementation("ch.acra:acra-http:$acraVersion")
 | 
			
		||||
    implementation("ch.acra:acra-toast:$acraVersion")
 | 
			
		||||
@@ -227,6 +232,7 @@ tasks.withType<Test> {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
aboutLibraries {
 | 
			
		||||
    excludeFields = arrayOf("generated")
 | 
			
		||||
    offlineMode = true
 | 
			
		||||
    fetchRemoteLicense = false
 | 
			
		||||
    fetchRemoteFunding = false
 | 
			
		||||
@@ -235,3 +241,33 @@ aboutLibraries {
 | 
			
		||||
    duplicationMode = com.mikepenz.aboutlibraries.plugin.DuplicateMode.MERGE
 | 
			
		||||
    duplicationRule = com.mikepenz.aboutlibraries.plugin.DuplicateRule.GROUP
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Screenshot failure handling
 | 
			
		||||
val reportsDirectory = file("$buildDir/reports/androidTests/connected")
 | 
			
		||||
 | 
			
		||||
val clearScreenshotsTask =
 | 
			
		||||
    tasks.register<Exec>("clearScreenshots") {
 | 
			
		||||
        println("AMINE : clear")
 | 
			
		||||
        commandLine = listOf("adb", "shell", "rm", "-r", "/sdcard/Pictures/selfoss_tests")
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
val createScreenshotDirectoryTask =
 | 
			
		||||
    tasks.register<Exec>("createScreenshotDirectory") {
 | 
			
		||||
        println("AMINE : create directory")
 | 
			
		||||
        group = "reporting"
 | 
			
		||||
        commandLine = listOf("adb", "shell", "mkdir", "-p", "/sdcard/Pictures/selfoss_tests")
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
tasks.register<Exec>("fetchScreenshots") {
 | 
			
		||||
    println("AMINE : fetch")
 | 
			
		||||
    group = "reporting"
 | 
			
		||||
    executable(android.adbExecutable.toString())
 | 
			
		||||
    commandLine = listOf("adb", "pull", "/sdcard/Pictures/selfoss_tests/.", reportsDirectory.toString())
 | 
			
		||||
 | 
			
		||||
    finalizedBy(clearScreenshotsTask)
 | 
			
		||||
    dependsOn(createScreenshotDirectoryTask)
 | 
			
		||||
 | 
			
		||||
    doFirst {
 | 
			
		||||
        reportsDirectory.mkdirs()
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -15,16 +15,28 @@ import androidx.test.filters.LargeTest
 | 
			
		||||
import bou.amine.apps.readerforselfossv2.android.testing.CountingIdlingResourceSingleton
 | 
			
		||||
import org.junit.After
 | 
			
		||||
import org.junit.Before
 | 
			
		||||
import org.junit.FixMethodOrder
 | 
			
		||||
import org.junit.Rule
 | 
			
		||||
import org.junit.Test
 | 
			
		||||
import org.junit.rules.RuleChain
 | 
			
		||||
import org.junit.runner.RunWith
 | 
			
		||||
import org.junit.runners.MethodSorters
 | 
			
		||||
 | 
			
		||||
@Suppress("ktlint:standard:class-naming", "detekt:ClassNaming")
 | 
			
		||||
@RunWith(AndroidJUnit4::class)
 | 
			
		||||
@FixMethodOrder(MethodSorters.NAME_ASCENDING)
 | 
			
		||||
@LargeTest
 | 
			
		||||
class LoginActivityTest {
 | 
			
		||||
class `1-LoginActivityTest` : WithANRException() {
 | 
			
		||||
    @get:Rule
 | 
			
		||||
    val activityRule = ActivityScenarioRule(LoginActivity::class.java)
 | 
			
		||||
 | 
			
		||||
    @JvmField
 | 
			
		||||
    @Rule
 | 
			
		||||
    val ruleChain: RuleChain =
 | 
			
		||||
        RuleChain
 | 
			
		||||
            .outerRule(activityRule)
 | 
			
		||||
            .around(ScreenshotTakingRule())
 | 
			
		||||
 | 
			
		||||
    @Before
 | 
			
		||||
    fun registerIdlingResource() {
 | 
			
		||||
        IdlingRegistry
 | 
			
		||||
@@ -40,7 +52,7 @@ class LoginActivityTest {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun viewIsInitialized() {
 | 
			
		||||
    fun `1-viewIsInitialized`() {
 | 
			
		||||
        onView(withId(R.id.urlView)).check(matches(isDisplayed()))
 | 
			
		||||
        onView(withId(R.id.selfSigned))
 | 
			
		||||
            .check(matches(isDisplayed()))
 | 
			
		||||
@@ -57,28 +69,28 @@ class LoginActivityTest {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun urlError() {
 | 
			
		||||
    fun `2-urlError`() {
 | 
			
		||||
        performLogin("10.0.2.2:8888")
 | 
			
		||||
        onView(withId(R.id.urlView)).perform(click())
 | 
			
		||||
        onView(withId(R.id.urlView)).check(matches(withError(R.string.login_url_problem)))
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun connectError() {
 | 
			
		||||
        performLogin("http://10.0.2.2:8889")
 | 
			
		||||
        onView(withId(R.id.urlView)).perform(click())
 | 
			
		||||
        onView(withId(R.id.urlView)).check(matches(withError(R.string.wrong_infos)))
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun urlSlashError() {
 | 
			
		||||
    fun `3-urlSlashError`() {
 | 
			
		||||
        performLogin("https://google.fr/toto")
 | 
			
		||||
        onView(withId(R.id.urlView)).perform(click())
 | 
			
		||||
        onView(withId(R.id.urlView)).check(matches(withError(R.string.login_url_problem)))
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun multiError() {
 | 
			
		||||
    fun `4-connectError`() {
 | 
			
		||||
        performLogin("http://10.0.2.2:8889")
 | 
			
		||||
        onView(withId(R.id.urlView)).perform(click())
 | 
			
		||||
        onView(withId(R.id.urlView)).check(matches(withError(R.string.wrong_infos)))
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun `5-multiError`() {
 | 
			
		||||
        onView(withId(R.id.signInButton)).perform(click())
 | 
			
		||||
        onView(withId(R.id.signInButton)).perform(click())
 | 
			
		||||
        onView(withId(R.id.signInButton)).perform(click())
 | 
			
		||||
@@ -86,8 +98,9 @@ class LoginActivityTest {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun connect() {
 | 
			
		||||
    fun `6-connect`() {
 | 
			
		||||
        performLogin()
 | 
			
		||||
        onView(withText(R.string.gdpr_dialog_title)).check(matches(isDisplayed()))
 | 
			
		||||
        onView(withText("OK")).perform(click())
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -15,21 +15,27 @@ import androidx.test.ext.junit.rules.ActivityScenarioRule
 | 
			
		||||
import androidx.test.ext.junit.runners.AndroidJUnit4
 | 
			
		||||
import androidx.test.filters.LargeTest
 | 
			
		||||
import org.hamcrest.CoreMatchers.not
 | 
			
		||||
import org.junit.Before
 | 
			
		||||
import org.junit.FixMethodOrder
 | 
			
		||||
import org.junit.Rule
 | 
			
		||||
import org.junit.Test
 | 
			
		||||
import org.junit.rules.RuleChain
 | 
			
		||||
import org.junit.runner.RunWith
 | 
			
		||||
import org.junit.runners.MethodSorters
 | 
			
		||||
 | 
			
		||||
@Suppress("ktlint:standard:class-naming", "detekt:ClassNaming")
 | 
			
		||||
@RunWith(AndroidJUnit4::class)
 | 
			
		||||
@FixMethodOrder(MethodSorters.NAME_ASCENDING)
 | 
			
		||||
@LargeTest
 | 
			
		||||
class HomeActivityTest {
 | 
			
		||||
class `2-HomeActivityTest` : WithANRException() {
 | 
			
		||||
    @get:Rule
 | 
			
		||||
    val activityRule = ActivityScenarioRule(LoginActivity::class.java)
 | 
			
		||||
    val activityRule = ActivityScenarioRule(HomeActivity::class.java)
 | 
			
		||||
 | 
			
		||||
    @Before
 | 
			
		||||
    fun init() {
 | 
			
		||||
        loginAndInitHome()
 | 
			
		||||
    }
 | 
			
		||||
    @JvmField
 | 
			
		||||
    @Rule
 | 
			
		||||
    val ruleChain: RuleChain =
 | 
			
		||||
        RuleChain
 | 
			
		||||
            .outerRule(activityRule)
 | 
			
		||||
            .around(ScreenshotTakingRule())
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun testMenu() {
 | 
			
		||||
@@ -15,13 +15,23 @@ import org.hamcrest.CoreMatchers.not
 | 
			
		||||
import org.junit.Before
 | 
			
		||||
import org.junit.Rule
 | 
			
		||||
import org.junit.Test
 | 
			
		||||
import org.junit.rules.RuleChain
 | 
			
		||||
import org.junit.runner.RunWith
 | 
			
		||||
 | 
			
		||||
@RunWith(AndroidJUnit4::class)
 | 
			
		||||
@LargeTest
 | 
			
		||||
class SettingsActivityTest {
 | 
			
		||||
@Suppress("ktlint:standard:class-naming", "detekt:ClassNaming")
 | 
			
		||||
class `3-SettingsActivityTest` : WithANRException() {
 | 
			
		||||
    @get:Rule
 | 
			
		||||
    val activityRule = ActivityScenarioRule(LoginActivity::class.java)
 | 
			
		||||
    val activityRule = ActivityScenarioRule(HomeActivity::class.java)
 | 
			
		||||
 | 
			
		||||
    @JvmField
 | 
			
		||||
    @Rule
 | 
			
		||||
    val ruleChain: RuleChain =
 | 
			
		||||
        RuleChain
 | 
			
		||||
            .outerRule(activityRule)
 | 
			
		||||
            .around(ScreenshotTakingRule())
 | 
			
		||||
 | 
			
		||||
    lateinit var context: Context
 | 
			
		||||
 | 
			
		||||
    @Before
 | 
			
		||||
@@ -29,7 +39,6 @@ class SettingsActivityTest {
 | 
			
		||||
        activityRule.scenario.onActivity { activity ->
 | 
			
		||||
            context = activity.window.context
 | 
			
		||||
        }
 | 
			
		||||
        loginAndInitHome()
 | 
			
		||||
        openMenu()
 | 
			
		||||
        onView(withText(R.string.title_activity_settings)).perform(click())
 | 
			
		||||
    }
 | 
			
		||||
@@ -68,6 +77,9 @@ class SettingsActivityTest {
 | 
			
		||||
        changeAndSaveSetting("", "10") {
 | 
			
		||||
            onView(withText(R.string.pref_api_timeout)).perform(click())
 | 
			
		||||
        }
 | 
			
		||||
        changeAndSaveSetting("", "60") {
 | 
			
		||||
            onView(withText(R.string.pref_api_timeout)).perform(click())
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
@@ -22,19 +22,30 @@ import androidx.test.filters.LargeTest
 | 
			
		||||
import org.hamcrest.CoreMatchers.allOf
 | 
			
		||||
import org.hamcrest.CoreMatchers.not
 | 
			
		||||
import org.junit.Before
 | 
			
		||||
import org.junit.FixMethodOrder
 | 
			
		||||
import org.junit.Rule
 | 
			
		||||
import org.junit.Test
 | 
			
		||||
import org.junit.rules.RuleChain
 | 
			
		||||
import org.junit.runner.RunWith
 | 
			
		||||
import org.junit.runners.MethodSorters
 | 
			
		||||
 | 
			
		||||
@Suppress("ktlint:standard:class-naming", "detekt:ClassNaming")
 | 
			
		||||
@RunWith(AndroidJUnit4::class)
 | 
			
		||||
@FixMethodOrder(MethodSorters.NAME_ASCENDING)
 | 
			
		||||
@LargeTest
 | 
			
		||||
class SettingsActivityGeneralTest {
 | 
			
		||||
class `4-SettingsActivityGeneralTest` : WithANRException() {
 | 
			
		||||
    @get:Rule
 | 
			
		||||
    val activityRule = ActivityScenarioRule(LoginActivity::class.java)
 | 
			
		||||
    val activityRule = ActivityScenarioRule(HomeActivity::class.java)
 | 
			
		||||
 | 
			
		||||
    @JvmField
 | 
			
		||||
    @Rule
 | 
			
		||||
    val ruleChain: RuleChain =
 | 
			
		||||
        RuleChain
 | 
			
		||||
            .outerRule(activityRule)
 | 
			
		||||
            .around(ScreenshotTakingRule())
 | 
			
		||||
 | 
			
		||||
    @Before
 | 
			
		||||
    fun init() {
 | 
			
		||||
        loginAndInitHome()
 | 
			
		||||
        openActionBarOverflowOrOptionsMenu(
 | 
			
		||||
            ApplicationProvider.getApplicationContext(),
 | 
			
		||||
        )
 | 
			
		||||
@@ -1,29 +1,40 @@
 | 
			
		||||
package bou.amine.apps.readerforselfossv2.android
 | 
			
		||||
 | 
			
		||||
import android.content.Context
 | 
			
		||||
import androidx.test.core.app.ApplicationProvider
 | 
			
		||||
import androidx.test.espresso.Espresso.onView
 | 
			
		||||
import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu
 | 
			
		||||
import androidx.test.espresso.action.ViewActions
 | 
			
		||||
import androidx.test.espresso.action.ViewActions.click
 | 
			
		||||
import androidx.test.espresso.assertion.ViewAssertions.matches
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isChecked
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isDisplayed
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isRoot
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.withText
 | 
			
		||||
import androidx.test.ext.junit.rules.ActivityScenarioRule
 | 
			
		||||
import androidx.test.ext.junit.runners.AndroidJUnit4
 | 
			
		||||
import androidx.test.filters.LargeTest
 | 
			
		||||
import bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity
 | 
			
		||||
import org.hamcrest.CoreMatchers.allOf
 | 
			
		||||
import org.hamcrest.CoreMatchers.not
 | 
			
		||||
import org.junit.After
 | 
			
		||||
import org.junit.Before
 | 
			
		||||
import org.junit.Rule
 | 
			
		||||
import org.junit.Test
 | 
			
		||||
import org.junit.rules.RuleChain
 | 
			
		||||
import org.junit.runner.RunWith
 | 
			
		||||
 | 
			
		||||
@Suppress("ktlint:standard:class-naming", "detekt:ClassNaming")
 | 
			
		||||
@RunWith(AndroidJUnit4::class)
 | 
			
		||||
@LargeTest
 | 
			
		||||
class SettingsActivityReaderTest {
 | 
			
		||||
class `5-SettingsActivityReaderTest` : WithANRException() {
 | 
			
		||||
    @get:Rule
 | 
			
		||||
    val activityRule = ActivityScenarioRule(LoginActivity::class.java)
 | 
			
		||||
    val activityRule = ActivityScenarioRule(SettingsActivity::class.java)
 | 
			
		||||
 | 
			
		||||
    @JvmField
 | 
			
		||||
    @Rule
 | 
			
		||||
    val ruleChain: RuleChain =
 | 
			
		||||
        RuleChain
 | 
			
		||||
            .outerRule(activityRule)
 | 
			
		||||
            .around(ScreenshotTakingRule())
 | 
			
		||||
 | 
			
		||||
    lateinit var context: Context
 | 
			
		||||
 | 
			
		||||
@@ -32,14 +43,14 @@ class SettingsActivityReaderTest {
 | 
			
		||||
        activityRule.scenario.onActivity { activity ->
 | 
			
		||||
            context = activity.window.context
 | 
			
		||||
        }
 | 
			
		||||
        loginAndInitHome()
 | 
			
		||||
        openActionBarOverflowOrOptionsMenu(
 | 
			
		||||
            ApplicationProvider.getApplicationContext(),
 | 
			
		||||
        )
 | 
			
		||||
        onView(withText(R.string.title_activity_settings)).perform(click())
 | 
			
		||||
        onView(withText(R.string.pref_header_viewer)).perform(click())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @After
 | 
			
		||||
    fun back() {
 | 
			
		||||
        onView(isRoot()).perform(ViewActions.pressBack())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Test
 | 
			
		||||
    fun testReader() {
 | 
			
		||||
        onView(withSettingsCheckboxFrame(R.string.pref_switch_actions_pager_scroll)).check(
 | 
			
		||||
@@ -1,31 +1,42 @@
 | 
			
		||||
package bou.amine.apps.readerforselfossv2.android
 | 
			
		||||
 | 
			
		||||
import android.content.Context
 | 
			
		||||
import androidx.test.core.app.ApplicationProvider
 | 
			
		||||
import androidx.test.espresso.Espresso.onView
 | 
			
		||||
import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu
 | 
			
		||||
import androidx.test.espresso.action.ViewActions
 | 
			
		||||
import androidx.test.espresso.action.ViewActions.click
 | 
			
		||||
import androidx.test.espresso.assertion.ViewAssertions.matches
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isChecked
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isDisplayed
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isEnabled
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isNotEnabled
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isRoot
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.withText
 | 
			
		||||
import androidx.test.ext.junit.rules.ActivityScenarioRule
 | 
			
		||||
import androidx.test.ext.junit.runners.AndroidJUnit4
 | 
			
		||||
import androidx.test.filters.LargeTest
 | 
			
		||||
import bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity
 | 
			
		||||
import org.hamcrest.CoreMatchers.allOf
 | 
			
		||||
import org.hamcrest.CoreMatchers.not
 | 
			
		||||
import org.junit.After
 | 
			
		||||
import org.junit.Before
 | 
			
		||||
import org.junit.Rule
 | 
			
		||||
import org.junit.Test
 | 
			
		||||
import org.junit.rules.RuleChain
 | 
			
		||||
import org.junit.runner.RunWith
 | 
			
		||||
 | 
			
		||||
@Suppress("ktlint:standard:class-naming", "detekt:ClassNaming")
 | 
			
		||||
@RunWith(AndroidJUnit4::class)
 | 
			
		||||
@LargeTest
 | 
			
		||||
class SettingsActivityOfflineTest {
 | 
			
		||||
class `6-SettingsActivityOfflineTest` : WithANRException() {
 | 
			
		||||
    @get:Rule
 | 
			
		||||
    val activityRule = ActivityScenarioRule(LoginActivity::class.java)
 | 
			
		||||
    val activityRule = ActivityScenarioRule(SettingsActivity::class.java)
 | 
			
		||||
 | 
			
		||||
    @JvmField
 | 
			
		||||
    @Rule
 | 
			
		||||
    val ruleChain: RuleChain =
 | 
			
		||||
        RuleChain
 | 
			
		||||
            .outerRule(activityRule)
 | 
			
		||||
            .around(ScreenshotTakingRule())
 | 
			
		||||
 | 
			
		||||
    lateinit var context: Context
 | 
			
		||||
 | 
			
		||||
@@ -34,14 +45,14 @@ class SettingsActivityOfflineTest {
 | 
			
		||||
        activityRule.scenario.onActivity { activity ->
 | 
			
		||||
            context = activity.window.context
 | 
			
		||||
        }
 | 
			
		||||
        loginAndInitHome()
 | 
			
		||||
        openActionBarOverflowOrOptionsMenu(
 | 
			
		||||
            ApplicationProvider.getApplicationContext(),
 | 
			
		||||
        )
 | 
			
		||||
        onView(withText(R.string.title_activity_settings)).perform(click())
 | 
			
		||||
        onView(withText(R.string.pref_header_offline)).perform(click())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @After
 | 
			
		||||
    fun back() {
 | 
			
		||||
        onView(isRoot()).perform(ViewActions.pressBack())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Suppress("detekt:LongMethod")
 | 
			
		||||
    @Test
 | 
			
		||||
    fun testOffline() {
 | 
			
		||||
@@ -18,14 +18,23 @@ import org.junit.After
 | 
			
		||||
import org.junit.Before
 | 
			
		||||
import org.junit.Rule
 | 
			
		||||
import org.junit.Test
 | 
			
		||||
import org.junit.rules.RuleChain
 | 
			
		||||
import org.junit.runner.RunWith
 | 
			
		||||
import java.util.UUID
 | 
			
		||||
 | 
			
		||||
@Suppress("ktlint:standard:class-naming", "detekt:ClassNaming")
 | 
			
		||||
@RunWith(AndroidJUnit4::class)
 | 
			
		||||
@LargeTest
 | 
			
		||||
class SourcesActivityTest {
 | 
			
		||||
class `7-SourcesActivityTest` : WithANRException() {
 | 
			
		||||
    @get:Rule
 | 
			
		||||
    val activityRule = ActivityScenarioRule(LoginActivity::class.java)
 | 
			
		||||
    val activityRule = ActivityScenarioRule(HomeActivity::class.java)
 | 
			
		||||
 | 
			
		||||
    @JvmField
 | 
			
		||||
    @Rule
 | 
			
		||||
    val ruleChain: RuleChain =
 | 
			
		||||
        RuleChain
 | 
			
		||||
            .outerRule(activityRule)
 | 
			
		||||
            .around(ScreenshotTakingRule())
 | 
			
		||||
 | 
			
		||||
    lateinit var sourceName: String
 | 
			
		||||
 | 
			
		||||
@@ -33,7 +42,6 @@ class SourcesActivityTest {
 | 
			
		||||
    fun init() {
 | 
			
		||||
        sourceName = UUID.randomUUID().toString().substring(0, 15)
 | 
			
		||||
 | 
			
		||||
        loginAndInitHome()
 | 
			
		||||
        goToSources()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@@ -75,10 +83,4 @@ class SourcesActivityTest {
 | 
			
		||||
        onView(withId(android.R.id.button1)).perform(click())
 | 
			
		||||
        onView(withText(sourceName)).check(doesNotExist())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun goToSources() {
 | 
			
		||||
        openMenu()
 | 
			
		||||
        onView(withText(R.string.menu_home_sources))
 | 
			
		||||
            .perform(click())
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,7 +1,11 @@
 | 
			
		||||
package bou.amine.apps.readerforselfossv2.android
 | 
			
		||||
 | 
			
		||||
import android.content.Context
 | 
			
		||||
import android.os.Environment.DIRECTORY_PICTURES
 | 
			
		||||
import android.os.Environment.getExternalStoragePublicDirectory
 | 
			
		||||
import android.util.Log
 | 
			
		||||
import androidx.annotation.ArrayRes
 | 
			
		||||
import androidx.test.espresso.Espresso
 | 
			
		||||
import androidx.test.espresso.Espresso.onData
 | 
			
		||||
import androidx.test.espresso.Espresso.onView
 | 
			
		||||
import androidx.test.espresso.action.ViewActions.click
 | 
			
		||||
@@ -9,29 +13,38 @@ import androidx.test.espresso.action.ViewActions.replaceText
 | 
			
		||||
import androidx.test.espresso.action.ViewActions.typeTextIntoFocusedView
 | 
			
		||||
import androidx.test.espresso.assertion.ViewAssertions.doesNotExist
 | 
			
		||||
import androidx.test.espresso.assertion.ViewAssertions.matches
 | 
			
		||||
import androidx.test.espresso.base.DefaultFailureHandler
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isChecked
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isDisplayed
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.isNotChecked
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.withId
 | 
			
		||||
import androidx.test.espresso.matcher.ViewMatchers.withText
 | 
			
		||||
import androidx.test.platform.app.InstrumentationRegistry.getInstrumentation
 | 
			
		||||
import androidx.test.runner.screenshot.BasicScreenCaptureProcessor
 | 
			
		||||
import androidx.test.runner.screenshot.Screenshot
 | 
			
		||||
import androidx.test.uiautomator.UiDevice
 | 
			
		||||
import androidx.test.uiautomator.UiSelector
 | 
			
		||||
import org.hamcrest.CoreMatchers.allOf
 | 
			
		||||
import org.hamcrest.Matchers.hasToString
 | 
			
		||||
import org.junit.BeforeClass
 | 
			
		||||
import org.junit.rules.TestWatcher
 | 
			
		||||
import org.junit.runner.Description
 | 
			
		||||
import java.io.File
 | 
			
		||||
import java.io.IOException
 | 
			
		||||
import java.util.Locale
 | 
			
		||||
 | 
			
		||||
val defaultUrl = (System.getenv("SELFOSS_URL") ?: "").ifEmpty { "http://10.0.2.2:8888" }
 | 
			
		||||
 | 
			
		||||
fun performLogin(someUrl: String? = null) {
 | 
			
		||||
    Log.i("AUTOMATION", "The url used will be ${if (!someUrl.isNullOrEmpty()) someUrl else defaultUrl}")
 | 
			
		||||
    onView(withId(R.id.urlView)).perform(click()).perform(
 | 
			
		||||
        typeTextIntoFocusedView(
 | 
			
		||||
            if (!someUrl.isNullOrEmpty()) someUrl else "http://10.0.2.2:8888",
 | 
			
		||||
            if (!someUrl.isNullOrEmpty()) someUrl else defaultUrl,
 | 
			
		||||
        ),
 | 
			
		||||
    )
 | 
			
		||||
    onView(withId(R.id.signInButton)).perform(click())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fun loginAndInitHome() {
 | 
			
		||||
    performLogin()
 | 
			
		||||
    onView(withText(R.string.gdpr_dialog_title)).check(matches(isDisplayed()))
 | 
			
		||||
    onView(withText("OK")).perform(click())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fun changeAndCancelSetting(
 | 
			
		||||
    oldValue: String,
 | 
			
		||||
    newValue: String,
 | 
			
		||||
@@ -97,6 +110,12 @@ fun testPreferencesFromArray(
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fun goToSources() {
 | 
			
		||||
    openMenu()
 | 
			
		||||
    onView(withText(R.string.menu_home_sources))
 | 
			
		||||
        .perform(click())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fun testAddSourceWithUrl(
 | 
			
		||||
    url: String,
 | 
			
		||||
    sourceName: String,
 | 
			
		||||
@@ -119,3 +138,87 @@ fun testAddSourceWithUrl(
 | 
			
		||||
        .perform(click())
 | 
			
		||||
    onView(withText(sourceName)).check(matches(isDisplayed()))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@Suppress("detekt:UtilityClassWithPublicConstructor")
 | 
			
		||||
open class WithANRException {
 | 
			
		||||
    companion object {
 | 
			
		||||
        // Running count of the number of Android Not Responding dialogues to prevent endless dismissal.
 | 
			
		||||
        private var anrCount = 0
 | 
			
		||||
 | 
			
		||||
        // `RootViewWithoutFocusException` class is private, need to match the message (instead of using type matching).
 | 
			
		||||
        private val rootViewWithoutFocusExceptionMsg =
 | 
			
		||||
            java.lang.String.format(
 | 
			
		||||
                Locale.ROOT,
 | 
			
		||||
                "Waited for the root of the view hierarchy to have " +
 | 
			
		||||
                    "window focus and not request layout for 10 seconds. If you specified a non " +
 | 
			
		||||
                    "default root matcher, it may be picking a root that never takes focus. " +
 | 
			
		||||
                    "Root:",
 | 
			
		||||
            )
 | 
			
		||||
 | 
			
		||||
        private fun handleAnrDialogue() {
 | 
			
		||||
            val device = UiDevice.getInstance(getInstrumentation())
 | 
			
		||||
            // If running the device in English Locale
 | 
			
		||||
            val waitButton = device.findObject(UiSelector().textContains("wait"))
 | 
			
		||||
            if (waitButton.exists()) waitButton.click()
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        @JvmStatic
 | 
			
		||||
        @BeforeClass
 | 
			
		||||
        fun setUpHandler() {
 | 
			
		||||
            Espresso.setFailureHandler { error, viewMatcher ->
 | 
			
		||||
 | 
			
		||||
                if (error.message!!.contains(rootViewWithoutFocusExceptionMsg) && anrCount < 3) {
 | 
			
		||||
                    anrCount++
 | 
			
		||||
                    handleAnrDialogue()
 | 
			
		||||
                } else { // chain all failures down to the default espresso handler
 | 
			
		||||
                    DefaultFailureHandler(getInstrumentation().targetContext).handle(error, viewMatcher)
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class MyScreenCaptureProcessor(
 | 
			
		||||
    parentFolderPath: String,
 | 
			
		||||
) : BasicScreenCaptureProcessor() {
 | 
			
		||||
    init {
 | 
			
		||||
        this.mDefaultScreenshotPath =
 | 
			
		||||
            File(
 | 
			
		||||
                File(
 | 
			
		||||
                    getExternalStoragePublicDirectory(DIRECTORY_PICTURES),
 | 
			
		||||
                    "selfoss_tests",
 | 
			
		||||
                ).absolutePath,
 | 
			
		||||
                "screenshots/$parentFolderPath",
 | 
			
		||||
            )
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun getFilename(prefix: String): String = prefix
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fun takeScreenshot(
 | 
			
		||||
    parentFolderPath: String = "",
 | 
			
		||||
    screenShotName: String,
 | 
			
		||||
) {
 | 
			
		||||
    Log.d("Screenshots", "Taking screenshot of '$screenShotName'")
 | 
			
		||||
    val screenCapture = Screenshot.capture()
 | 
			
		||||
    val processors = setOf(MyScreenCaptureProcessor(parentFolderPath))
 | 
			
		||||
    try {
 | 
			
		||||
        screenCapture.apply {
 | 
			
		||||
            name = screenShotName
 | 
			
		||||
            process(processors)
 | 
			
		||||
        }
 | 
			
		||||
        Log.d("Screenshots", "Screenshot taken")
 | 
			
		||||
    } catch (ex: IOException) {
 | 
			
		||||
        Log.e("Screenshots", "Could not take the screenshot", ex)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class ScreenshotTakingRule : TestWatcher() {
 | 
			
		||||
    override fun failed(
 | 
			
		||||
        e: Throwable?,
 | 
			
		||||
        description: Description,
 | 
			
		||||
    ) {
 | 
			
		||||
        val parentFolderPath = "failures/${description.className}"
 | 
			
		||||
        takeScreenshot(parentFolderPath = parentFolderPath, screenShotName = description.methodName)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -27,3 +27,4 @@ org.gradle.caching=true
 | 
			
		||||
ignoreGitVersion=false
 | 
			
		||||
kotlin.native.cacheKind.iosX64=none
 | 
			
		||||
org.gradle.configureondemand=true
 | 
			
		||||
kotlin.jvm.target.validation.mode=IGNORE
 | 
			
		||||
 
 | 
			
		||||
@@ -224,16 +224,25 @@ class Repository(
 | 
			
		||||
            appSettingsService.isItemCachingEnabled() || !appSettingsService.isUpdateSourcesEnabled()
 | 
			
		||||
        val shouldFetch = if (!appSettingsService.isUpdateSourcesEnabled()) !fetchedSources else true
 | 
			
		||||
        if (shouldFetch && connectivityService.isNetworkAvailable()) {
 | 
			
		||||
            val apiSources = api.sourcesDetailed()
 | 
			
		||||
            if (apiSources.success && apiSources.data != null) {
 | 
			
		||||
                fetchedSources = true
 | 
			
		||||
                sources = apiSources.data
 | 
			
		||||
                if (isDatabaseEnabled) {
 | 
			
		||||
                    resetDBSourcesWithData(sources)
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            sources = sourceDetails(isDatabaseEnabled)
 | 
			
		||||
        } else if (isDatabaseEnabled) {
 | 
			
		||||
            sources = getDBSources().map { it.toView() } as ArrayList<SelfossModel.SourceDetail>
 | 
			
		||||
            if (sources.isEmpty() && !connectivityService.isNetworkAvailable() && !fetchedSources) {
 | 
			
		||||
                sources = sourceDetails(isDatabaseEnabled)
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return sources
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private suspend fun sourceDetails(isDatabaseEnabled: Boolean): ArrayList<SelfossModel.SourceDetail> {
 | 
			
		||||
        var sources = ArrayList<SelfossModel.SourceDetail>()
 | 
			
		||||
        val apiSources = api.sourcesDetailed()
 | 
			
		||||
        if (apiSources.success && apiSources.data != null) {
 | 
			
		||||
            fetchedSources = true
 | 
			
		||||
            sources = apiSources.data
 | 
			
		||||
            if (isDatabaseEnabled) {
 | 
			
		||||
                resetDBSourcesWithData(sources)
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return sources
 | 
			
		||||
    }
 | 
			
		||||
@@ -371,6 +380,7 @@ class Repository(
 | 
			
		||||
    ): Boolean {
 | 
			
		||||
        var response = false
 | 
			
		||||
        if (connectivityService.isNetworkAvailable()) {
 | 
			
		||||
            fetchedSources = false
 | 
			
		||||
            response = api
 | 
			
		||||
                .createSourceForVersion(
 | 
			
		||||
                    title,
 | 
			
		||||
@@ -392,6 +402,7 @@ class Repository(
 | 
			
		||||
    ): Boolean {
 | 
			
		||||
        var response = false
 | 
			
		||||
        if (connectivityService.isNetworkAvailable()) {
 | 
			
		||||
            fetchedSources = false
 | 
			
		||||
            response = api.updateSourceForVersion(id, title, url, spout, tags).isSuccess == true
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
@@ -406,6 +417,7 @@ class Repository(
 | 
			
		||||
        if (connectivityService.isNetworkAvailable()) {
 | 
			
		||||
            val response = api.deleteSource(id)
 | 
			
		||||
            success = response.isSuccess
 | 
			
		||||
            fetchedSources = false
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // We filter on success or if the network isn't available
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user