All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.internal.filewatch.DefaultFileWatcherFactoryTest.groovy Maven / Gradle / Ivy

/*
 * Copyright 2015 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.gradle.internal.filewatch

import org.gradle.api.file.DirectoryTree
import org.gradle.api.internal.file.FileSystemSubset
import org.gradle.api.tasks.util.PatternSet
import org.gradle.internal.Pair
import org.gradle.internal.concurrent.DefaultExecutorFactory
import org.gradle.internal.concurrent.Stoppable
import org.gradle.internal.nativeintegration.filesystem.FileSystem
import org.gradle.util.UsesNativeServices
import spock.lang.Unroll

import java.util.concurrent.CountDownLatch

@UsesNativeServices
class DefaultFileWatcherFactoryTest extends AbstractFileWatcherTest {
    FileWatcher fileWatcher
    FileWatcherFactory fileWatcherFactory

    void setup() {
        fileWatcherFactory = new DefaultFileWatcherFactory(new DefaultExecutorFactory(), Stub(FileSystem))
        fileSystemSubset = FileSystemSubset.builder().add(testDir.testDirectory).build()
    }

    void cleanup() {
        fileWatcher?.stop()
        if (fileWatcherFactory instanceof Stoppable) {
            fileWatcherFactory.stop()
        }
    }

    def "watch service should notify of new files"() {
        given:
        def listener = Mock(FileWatcherListener)
        def listenerCalledLatch = new CountDownLatch(1)
        when:
        fileWatcher = fileWatcherFactory.watch(onError, listener)
        fileWatcher.watch(fileSystemSubset)
        File createdFile = testDir.file("newfile.txt")
        createdFile.text = "Hello world"
        waitOn(listenerCalledLatch)
        then:
        (1.._) * listener.onChange(_, _) >> { FileWatcher watcher, FileWatcherEvent event ->
            handleEvent(event, listenerCalledLatch)
        }
        0 * _._
    }

    def "watch service should notify of new files in subdirectories"() {
        given:
        def listener = Mock(FileWatcherListener)
        def listenerCalledLatch = new CountDownLatch(1)
        def listenerCalledLatch2 = new CountDownLatch(1)
        when:
        fileWatcher = fileWatcherFactory.watch(onError, listener)
        fileWatcher.watch(fileSystemSubset)
        def subdir = testDir.createDir("subdir")
        subdir.createFile("somefile").text = "Hello world"
        waitOn(listenerCalledLatch)
        then:
        (1.._) * listener.onChange(_, _) >> { FileWatcher watcher, FileWatcherEvent event ->
            handleEvent(event, listenerCalledLatch)
        }
        0 * _._
        when:
        subdir.file('someotherfile').text = "Hello world"
        waitOn(listenerCalledLatch2)
        then:
        (1.._) * listener.onChange(_, _) >> { FileWatcher watcher, FileWatcherEvent event ->
            handleEvent(event, listenerCalledLatch2)
        }
        0 * _._
    }

    def handleEvent(FileWatcherEvent event, CountDownLatch listenerCalledLatch) {
        //println "event: $event"
        listenerCalledLatch.countDown()
    }

    def "watch service should register to watch subdirs at startup"() {
        given:
        def listener = Mock(FileWatcherListener)
        def subdir = testDir.createDir("subdir")
        def listenerCalledLatch = new CountDownLatch(1)
        when:
        fileWatcher = fileWatcherFactory.watch(onError, listener)
        fileWatcher.watch(fileSystemSubset)
        subdir.createFile("somefile").text = "Hello world"
        waitOn(listenerCalledLatch)
        then:
        (1.._) * listener.onChange(_, _) >> { FileWatcher watcher, FileWatcherEvent event ->
            handleEvent(event, listenerCalledLatch)
        }
    }

    def "listener can terminate watcher"() {
        given:
        def block = this.> blockingVar()

        when:
        fileWatcher = fileWatcherFactory.watch(onError) { watcher, event ->
            def vals = [watcher.running]
            watcher.stop()
            block.set(vals << watcher.running)
        }
        fileWatcher.watch(fileSystemSubset)

        and:
        testDir.file("new") << "new"

        then:
        block.get() == [true, false]
        !fileWatcher.running
    }

    def "observer can terminate watcher"() {
        given:
        def eventLatch = new CountDownLatch(1)
        def stopLatch = new CountDownLatch(1)
        def result = this. blockingVar()

        when:
        fileWatcher = fileWatcherFactory.watch(onError) { watcher, event ->
            eventLatch.countDown()
            try {
                waitOn(stopLatch)
            } catch(e) {}
            result.set(watcher.running)
        }
        fileWatcher.watch(fileSystemSubset)

        and:
        testDir.file("new") << "new"

        then:
        waitOn(eventLatch)
        fileWatcher.stop()
        stopLatch.countDown()
        !result.get()
    }

    def "can interrupt watcher"() {
        given:
        def watcherThread = this. blockingVar()

        when:
        fileWatcher = fileWatcherFactory.watch(onError) { watcher, event ->
            watcherThread.set(Thread.currentThread())
        }
        fileWatcher.watch(fileSystemSubset)

        and:
        testDir.file("new") << "new"

        then:
        watcherThread.get().interrupt()
        await { assert !fileWatcher.running }
    }

    def "can interrupt watcher event delivery"() {
        when:
        def eventReceivedLatch = new CountDownLatch(1)
        def interruptedLatch = new CountDownLatch(1)
        fileWatcher = fileWatcherFactory.watch(onError) { watcher, event ->
            eventReceivedLatch.countDown()
            try {
                Thread.sleep(100000)
            } catch (InterruptedException e) {
                interruptedLatch.countDown()
                throw e
            }
        }
        fileWatcher.watch(fileSystemSubset)

        and:
        testDir.file("new") << "new"

        then:
        waitOn(eventReceivedLatch)
        fileWatcher.stop()
        waitOn(interruptedLatch)
    }

    def "watcher can detects all files added to watched directory"() {
        when:
        def eventReceivedLatch = new CountDownLatch(1)
        def filesAddedLatch = new CountDownLatch(1)
        def totalLatch = new CountDownLatch(10)

        fileWatcher = fileWatcherFactory.watch(onError) { watcher, event ->
            eventReceivedLatch.countDown()
            waitOn(filesAddedLatch)
            if (event.type == FileWatcherEvent.Type.CREATE) {
                totalLatch.countDown()
            }
        }
        fileWatcher.watch(fileSystemSubset)

        testDir.file("1").createDir()
        waitOn(eventReceivedLatch)

        testDir.file("1/2/3/4/5/6/7/8/9/10").createDir()
        filesAddedLatch.countDown()

        then:
        waitOn(totalLatch)
    }

    def "watcher doesn't add directories that have been deleted after change detection"() {
        when:
        def eventReceivedLatch = new CountDownLatch(1)
        fileWatcher = fileWatcherFactory.watch(onError) { watcher, event ->
            eventReceivedLatch.countDown()
            event.file.delete()
        }
        fileWatcher.watch(fileSystemSubset)

        testDir.file("testdir").createDir()
        waitOn(eventReceivedLatch)
        sleep(500)

        then:
        noExceptionThrown()
        thrownInWatchExecution == null
    }


    def "watcher will stop if listener throws and error is forwarded"() {
        when:
        def onErrorStatus = this.> blockingVar()
        fileWatcher = fileWatcherFactory.watch(
            { onErrorStatus.set(Pair.of(fileWatcher.running, it)) },
            { watcher, event -> throw new RuntimeException("!!") }
        )
        fileWatcher.watch(fileSystemSubset)

        and:
        testDir.file("new") << "new"

        then:
        await { assert !fileWatcher.running }
        !onErrorStatus.get().left
        onErrorStatus.get().right.message == "!!"
    }

    @Unroll
    def "watcher should register to watch all added directories - #scenario"() {
        given:
        def listener = Mock(FileWatcherListener)
        def subdir1 = testDir.file('src/main/java')
        def subdir1File = subdir1.file("SomeFile.java")
        def subdir2 = testDir.file('src/main/groovy')
        def subdir2File = subdir2.file("SomeFile.groovy")
        def filesToSee = ([subdir1File, subdir2File].collect { it.absolutePath } as Set).asSynchronized()
        def fileEventMatchedLatch = new CountDownLatch(filesToSee.size())

        if (subdir1Create) {
            subdir1.mkdirs()
        }
        if (subdir2Create) {
            subdir2.mkdirs()
        }
        if (parentCreate) {
            subdir1.getParentFile().mkdirs()
        }
        when:
        fileWatcher = fileWatcherFactory.watch(onError, listener)
        fileWatcher.watch(FileSystemSubset.builder().add(subdir1).build())
        subdir1.createFile("SomeFile.java").text = "Hello world"
        fileWatcher.watch(FileSystemSubset.builder().add(subdir2).build())
        subdir2.createFile("SomeFile.groovy").text = "Hello world"
        waitOn(fileEventMatchedLatch, false)
        then:
        (1.._) * listener.onChange(_, _) >> { FileWatcher watcher, FileWatcherEvent event ->
            if (event.file.isFile()) {
                if (filesToSee.remove(event.file.absolutePath)) {
                    fileEventMatchedLatch.countDown()
                }
            }
        }
        filesToSee.isEmpty()
        where:
        scenario                         | subdir1Create | subdir2Create | parentCreate
        'both exist'                     | true          | true          | false
        'first exists'                   | true          | false         | false
        //'second exists'                  | false         | true          | false
        //'neither exists - parent exists' | false         | false         | true
        //'neither exists'                 | false         | false         | false
    }

    def "should support watching directory that didn't exist when watching started"() {
        given:
        def listener = Mock(FileWatcherListener)
        def fileEventMatchedLatch = new CountDownLatch(1)
        def filesSeen = ([] as Set).asSynchronized()
        listener.onChange(_, _) >> { FileWatcher watcher, FileWatcherEvent event ->
            if (event.file.isFile()) {
                filesSeen.add(event.file.absolutePath)
                fileEventMatchedLatch.countDown()
            }
        }
        fileWatcher = fileWatcherFactory.watch(onError, listener)
        def subdir = testDir.file('src/main/java')
        def subdirFile = subdir.file("SomeFile.java")

        when: 'Adds watch for non-existing directory'
        fileWatcher.watch(FileSystemSubset.builder().add(subdir).build())

        and: 'Creates file'
        subdirFile.createFile().text = 'Some content'
        waitOn(fileEventMatchedLatch)

        then: 'File should have been noticed'
        filesSeen.size() == 1
        filesSeen[0] == subdirFile.absolutePath
    }

    @Unroll
    def "should watch changes in sub directory when watching first for single file in parent directory where usesDirectoryTree: #usesDirectoryTree"() {
        given:
        def listener = Mock(FileWatcherListener)
        def fileEventMatchedLatch = new CountDownLatch(1)
        def filesSeen = ([] as Set).asSynchronized()
        listener.onChange(_, _) >> { FileWatcher watcher, FileWatcherEvent event ->
            if (event.file.isFile()) {
                filesSeen.add(event.file.absolutePath)
                fileEventMatchedLatch.countDown()
            }
        }
        fileWatcher = fileWatcherFactory.watch(onError, listener)
        def subdir = testDir.file('src/subdirectory')
        def subdirFile = subdir.file("nested.txt")

        // watch for file in parent directory initially
        def singleFile = testDir.file('src/topLevel.txt').createFile()
        fileWatcher.watch(FileSystemSubset.builder().add(singleFile).build())

        when: 'Adds watch for sub directory'
        subdir.mkdirs()
        fileWatcher.watch(FileSystemSubset.builder().add(usesDirectoryTree ? new SimpleDirectoryTree(dir: subdir) : subdir).build())

        and: 'Create file'
        subdirFile.createFile().text = 'Some content'
        waitOn(fileEventMatchedLatch)

        then: 'File should have been noticed'
        filesSeen.size() == 1
        filesSeen[0] == subdirFile.absolutePath

        where:
        usesDirectoryTree << [false, true]
    }

    private static class SimpleDirectoryTree implements DirectoryTree {
        File dir
        PatternSet patterns = new PatternSet()
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy